mirror of
https://github.com/EQEmu/Server.git
synced 2026-05-30 11:45:46 +00:00
Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b7eeb3b2f9 | |||
| d2f8eead1f |
@@ -1,6 +0,0 @@
|
|||||||
!Makefile
|
|
||||||
base/*.sql
|
|
||||||
base/*.zip
|
|
||||||
base/db/
|
|
||||||
base/maps/
|
|
||||||
!base/expansion/Makefile
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
# This is a "dev" image for running eqemu in development, also for dev containers
|
|
||||||
ARG USERNAME=user-name-goes-here
|
|
||||||
ARG USER_UID=1000
|
|
||||||
ARG USER_GID=$USER_UID
|
|
||||||
|
|
||||||
FROM --platform=linux/arm64 mcr.microsoft.com/devcontainers/base:debian
|
|
||||||
RUN sudo apt update && sudo apt install -y --no-install-recommends build-essential libtool cmake curl debconf-utils git libluabind-dev libsodium-dev liblua5.2-0 liblua5.2-dev libmariadb-dev libssl-dev minizip make mariadb-client locales nano open-vm-tools unzip uuid-dev iputils-ping wget libcurl4-openssl-dev gdb libyaml-cpp-dev ccache ninja-build pv mariadb-server libperl-dev libjson-perl libio-stringy-perl liblua5.1-dev libluabind-dev libboost-dev mariadb-server valgrind telnet libgoogle-perftools-dev google-perftools
|
|
||||||
|
|
||||||
USER $USERNAME
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
# This is a "dev" image for running eqemu in development, also for dev containers
|
|
||||||
ARG USERNAME=user-name-goes-here
|
|
||||||
ARG USER_UID=1000
|
|
||||||
ARG USER_GID=$USER_UID
|
|
||||||
|
|
||||||
FROM mcr.microsoft.com/devcontainers/base:debian
|
|
||||||
RUN sudo apt update && sudo apt install -y --no-install-recommends build-essential libtool cmake curl debconf-utils git libluabind-dev libsodium-dev liblua5.2-0 liblua5.2-dev libmariadb-dev libssl-dev minizip make mariadb-client locales nano open-vm-tools unzip uuid-dev iputils-ping wget libcurl4-openssl-dev gdb libyaml-cpp-dev ccache ninja-build pv mariadb-server libperl-dev libjson-perl libio-stringy-perl liblua5.1-dev libluabind-dev libboost-dev mariadb-server valgrind telnet libgoogle-perftools-dev google-perftools
|
|
||||||
|
|
||||||
USER $USERNAME
|
|
||||||
@@ -1,284 +0,0 @@
|
|||||||
NAME := eqemu-server
|
|
||||||
.ONESHELL:
|
|
||||||
|
|
||||||
DOCKER_ARGS := --rm --name ${NAME} -v $$PWD:/src -w /src ${NAME}
|
|
||||||
DOCKER_ARM64_ARGS := --rm --platform linux/arm64 --name ${NAME}-arm64 -v $$PWD:/src -w /src ${NAME}-arm64
|
|
||||||
|
|
||||||
.PHONY: build
|
|
||||||
build:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile build --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build$$BUILD_SUFFIX && cmake --build . --config Release --target all --
|
|
||||||
|
|
||||||
.PHONY: cmake
|
|
||||||
cmake:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile cmake --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@echo "working directory: $$PWD"
|
|
||||||
mkdir -p build$$BUILD_SUFFIX
|
|
||||||
@cd build$$BUILD_SUFFIX && cmake -DEQEMU_BUILD_LOGIN=ON \
|
|
||||||
-DEQEMU_BUILD_TESTS=ON \
|
|
||||||
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache -G Ninja ..
|
|
||||||
|
|
||||||
clean:
|
|
||||||
ifneq (,$(findstring .devcontainer,$$PWD))
|
|
||||||
@make -C ../ -f .devcontainer/Makefile clean --no-print-directory
|
|
||||||
endif
|
|
||||||
rm -rf build
|
|
||||||
|
|
||||||
docker-cmake:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile docker-cmake --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@echo "working directory: $$PWD"
|
|
||||||
git submodule update --init --recursive
|
|
||||||
docker run ${DOCKER_ARGS} make cmake
|
|
||||||
|
|
||||||
docker-build:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile docker-build --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
docker run ${DOCKER_ARGS} make build
|
|
||||||
|
|
||||||
# Build image if it doesn't exist
|
|
||||||
docker-image-build:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile docker-image-build --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
ifeq ($(shell docker images -q ${NAME} 2> /dev/null),)
|
|
||||||
@echo "Docker image not found. Building..."
|
|
||||||
docker build -f Dockerfile.debian.dev -t ${NAME} .
|
|
||||||
endif
|
|
||||||
|
|
||||||
docker-arm-cmake: docker-arm-image-build
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile docker-arm-cmake --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
git submodule update --init --recursive
|
|
||||||
docker run ${DOCKER_ARM64_ARGS} make cmake BUILD_SUFFIX=arm64
|
|
||||||
|
|
||||||
docker-arm-build: docker-arm-image-build
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile docker-arm-build --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
docker run ${DOCKER_ARM64_ARGS} make build BUILD_SUFFIX=arm64
|
|
||||||
|
|
||||||
docker-arm-image-build:
|
|
||||||
ifeq ($(shell docker images -q ${NAME}-arm64 2> /dev/null),)
|
|
||||||
@echo "Docker image not found. Building..."
|
|
||||||
docker build -f Dockerfile.debian.arm.dev -t ${NAME}-arm64 .
|
|
||||||
endif
|
|
||||||
|
|
||||||
docker-clean: clean
|
|
||||||
|
|
||||||
.PHONY: prep
|
|
||||||
prep:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile prep --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@echo "Preparing build/bin for usage..."
|
|
||||||
mkdir -p build/bin/assets/patches
|
|
||||||
cp -R -u -p .devcontainer/base/eqemu_config.json build/bin/eqemu_config.json
|
|
||||||
cp -R -u -p .devcontainer/base/login.json build/bin/login.json
|
|
||||||
cp -R -u -p loginserver/login_util/* build/bin/assets/patches/
|
|
||||||
mkdir -p build/bin/assets
|
|
||||||
cp -R -u -p utils/patches build/bin/assets/
|
|
||||||
-unlink build/bin/lua_modules
|
|
||||||
cd build/bin && ln -s quests/lua_modules lua_modules
|
|
||||||
-unlink build/bin/mods
|
|
||||||
cd build/bin && ln -s quests/mods mods
|
|
||||||
-unlink build/bin/maps
|
|
||||||
cd build/bin && ln -s ../../base/maps maps
|
|
||||||
mkdir -p build/bin/logs
|
|
||||||
mkdir -p build/bin/shared
|
|
||||||
@echo "Eqemu is prepared. Edit build/bin/eqemu_config.json to configure."
|
|
||||||
|
|
||||||
maps:
|
|
||||||
@echo "Downloading maps..."
|
|
||||||
@mkdir -p base/maps
|
|
||||||
@cd base/maps && wget -nc https://github.com/Akkadius/eqemu-maps/archive/refs/heads/master.zip
|
|
||||||
@cd base/maps && unzip -o master.zip
|
|
||||||
@cd base/maps && mv eqemu-maps-master/* .
|
|
||||||
@cd base/maps && rm -rf eqemu-maps-master
|
|
||||||
@echo "Maps downloaded."
|
|
||||||
|
|
||||||
quests:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile quests --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@cd build/bin && git clone https://github.com/ProjectEQ/projecteqquests.git quests
|
|
||||||
|
|
||||||
# Runs tests
|
|
||||||
.PHONY: test
|
|
||||||
test:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile test --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && ./tests
|
|
||||||
|
|
||||||
# Runs login binary
|
|
||||||
.PHONY: login
|
|
||||||
login:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile login --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && ./loginserver
|
|
||||||
|
|
||||||
# Runs shared_memory binary
|
|
||||||
.PHONY: shared
|
|
||||||
shared:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile shared --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && ./shared_memory
|
|
||||||
|
|
||||||
# Runs zone binary
|
|
||||||
.PHONY: zone
|
|
||||||
zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@-rm build/bin/logs/zone/zone*.log
|
|
||||||
cd build/bin && ./zone
|
|
||||||
|
|
||||||
# Runs world binary
|
|
||||||
.PHONY: world
|
|
||||||
world:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile world --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@-rm build/bin/logs/world*.log
|
|
||||||
cd build/bin && ./world
|
|
||||||
|
|
||||||
# Runs ucs binary
|
|
||||||
.PHONY: ucs
|
|
||||||
ucs:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile ucs --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@-rm build/bin/logs/ucs*.log
|
|
||||||
cd build/bin && ./ucs
|
|
||||||
|
|
||||||
# Runs queryserv binary
|
|
||||||
.PHONY: queryserv
|
|
||||||
queryserv:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile queryserv --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@-rm build/bin/logs/query_server*.log
|
|
||||||
cd build/bin && ./queryserv
|
|
||||||
|
|
||||||
valgrind-%:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile valgrind --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && valgrind --leak-check=full --show-leak-kinds=all --track-origins=yes --verbose --log-file=logs/$*.valgrind.log ./$*
|
|
||||||
|
|
||||||
# Start mariaDB standalone
|
|
||||||
.PHONY: mariadb
|
|
||||||
mariadb:
|
|
||||||
@sudo service mariadb start
|
|
||||||
|
|
||||||
.PHONY: inject-mariadb
|
|
||||||
inject-mariadb:
|
|
||||||
-sudo service mariadb start
|
|
||||||
-mkdir -p base/db/
|
|
||||||
-sudo mariadb -e 'DROP DATABASE IF EXISTS peq;'
|
|
||||||
-sudo mariadb -e 'CREATE DATABASE peq;'
|
|
||||||
-sudo mariadb -e "CREATE USER 'peq'@'127.0.0.1' IDENTIFIED BY 'peqpass';"
|
|
||||||
-sudo mariadb -e "GRANT ALL PRIVILEGES ON *.* TO 'peq'@'127.0.0.1';"
|
|
||||||
ifeq (,$(wildcard base/db/db.sql.zip))
|
|
||||||
@echo "base/db.sql.zip not found. Downloading latest from https://db.projecteq.net/"
|
|
||||||
wget -nc https://db.projecteq.net/latest -O base/db/db.sql.zip
|
|
||||||
-cd base/db && unzip db.sql.zip
|
|
||||||
endif
|
|
||||||
@echo "Sourcing db may take a while, please wait..."
|
|
||||||
@cd base/db/peq-dump && sudo mariadb --database peq -e "source create_all_tables.sql"
|
|
||||||
@echo "MariaDB is now injected."
|
|
||||||
|
|
||||||
.PHONY: gm-%
|
|
||||||
gm-%:
|
|
||||||
sudo mariadb --database peq -e "UPDATE account SET status=255 WHERE name = '$*';"
|
|
||||||
@echo "Account $* is now a GM. /camp to have it go into effect."
|
|
||||||
|
|
||||||
depends:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile depends --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
sudo apt install graphviz pip time
|
|
||||||
pip3 install graphviz
|
|
||||||
mkdir -p build/depends
|
|
||||||
@if [ ! -f "build/depends/dependency_graph.py" ]; then \
|
|
||||||
wget https://raw.githubusercontent.com/pvigier/dependency-graph/master/dependency_graph.py -O build/depends/dependency_graph.py; \
|
|
||||||
fi
|
|
||||||
@echo "Generating dependency graphs (This may take a while)..."
|
|
||||||
@echo "Login..."
|
|
||||||
time python3 build/depends/dependency_graph.py -f png login build/depends/login.dot
|
|
||||||
@echo "World..."
|
|
||||||
time python3 build/depends/dependency_graph.py -f png world build/depends/world.dot
|
|
||||||
@echo "Zone..."
|
|
||||||
time python3 build/depends/dependency_graph.py -f png zone build/depends/zone.dot
|
|
||||||
@echo "Common..."
|
|
||||||
time python3 build/depends/dependency_graph.py -f png common build/depends/common.dot
|
|
||||||
|
|
||||||
backup:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile backup --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@mkdir -p build/bin/backup
|
|
||||||
cd build/bin && ./world database:dump --compress --player-tables --state-tables --system-tables --query-serv-tables
|
|
||||||
|
|
||||||
cpu-zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile cpu-zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@cd build/bin && mkdir -p tmp
|
|
||||||
cd build/bin && CPUPROFILE=prof.out ./zone
|
|
||||||
|
|
||||||
pprof-zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile pprof-zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && google-pprof --pdf zone prof.out > prof.pdf
|
|
||||||
pprof-web-zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile pprof-web-zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && google-pprof --web zone prof.out
|
|
||||||
pprof-gv-zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile pprof-gv-zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
cd build/bin && google-pprof --gv zone prof.out > prof.gv
|
|
||||||
heap-zone:
|
|
||||||
ifeq ($(findstring .devcontainer,$(CURDIR)),.devcontainer)
|
|
||||||
@make -C ../ -f .devcontainer/Makefile heap-zone --no-print-directory
|
|
||||||
exit
|
|
||||||
endif
|
|
||||||
@cd build/bin && mkdir -p tmp
|
|
||||||
cd build/bin && HEAPPROFILE=prof.out ./zone
|
|
||||||
@@ -1,73 +0,0 @@
|
|||||||
{
|
|
||||||
"server": {
|
|
||||||
"zones": {
|
|
||||||
"defaultstatus": "0",
|
|
||||||
"ports": {
|
|
||||||
"low": "7000",
|
|
||||||
"high": "7400"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"qsdatabase": {
|
|
||||||
"host": "127.0.0.1",
|
|
||||||
"port": "3306",
|
|
||||||
"username": "peq",
|
|
||||||
"password": "peqpass",
|
|
||||||
"db": "peq"
|
|
||||||
},
|
|
||||||
"chatserver": {
|
|
||||||
"port": "7778",
|
|
||||||
"host": ""
|
|
||||||
},
|
|
||||||
"mailserver": {
|
|
||||||
"host": "",
|
|
||||||
"port": "7778"
|
|
||||||
},
|
|
||||||
"webinterface": {
|
|
||||||
"port": "9081"
|
|
||||||
},
|
|
||||||
"world": {
|
|
||||||
"longname": "New Devbox",
|
|
||||||
"address": "192.168.1.100",
|
|
||||||
"localaddress": "192.168.1.100",
|
|
||||||
"loginserver1": {
|
|
||||||
"account": "",
|
|
||||||
"password": "",
|
|
||||||
"legacy": 0,
|
|
||||||
"host": "login.projecteq.net",
|
|
||||||
"port": "5998"
|
|
||||||
},
|
|
||||||
"tcp": {
|
|
||||||
"ip": "127.0.0.1",
|
|
||||||
"port": "9001"
|
|
||||||
},
|
|
||||||
"telnet": {
|
|
||||||
"ip": "0.0.0.0",
|
|
||||||
"port": "9000",
|
|
||||||
"enabled": "true"
|
|
||||||
},
|
|
||||||
"key": "random-generate-here",
|
|
||||||
"http": {
|
|
||||||
"port": "9080",
|
|
||||||
"enabled": "true",
|
|
||||||
"mimefile": "mime.types"
|
|
||||||
},
|
|
||||||
"shortname": "dev"
|
|
||||||
},
|
|
||||||
"database": {
|
|
||||||
"db": "peq",
|
|
||||||
"host": "127.0.0.1",
|
|
||||||
"port": "3306",
|
|
||||||
"username": "peq",
|
|
||||||
"password": "peqpass"
|
|
||||||
},
|
|
||||||
"files": {
|
|
||||||
"opcodes": "assets/patches/opcodes.conf",
|
|
||||||
"mail_opcodes": "assets/patches/mail_opcodes.conf"
|
|
||||||
},
|
|
||||||
"directories": {
|
|
||||||
"patches": "assets/patches/",
|
|
||||||
"opcodes": "assets/patches/",
|
|
||||||
"plugins": "quests/plugins/"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
{
|
|
||||||
"database": {
|
|
||||||
"host": "127.0.0.1",
|
|
||||||
"port": "3306",
|
|
||||||
"db": "peq",
|
|
||||||
"user": "peq",
|
|
||||||
"password": "peqpass"
|
|
||||||
},
|
|
||||||
"account": {
|
|
||||||
"auto_create_accounts": true
|
|
||||||
},
|
|
||||||
"worldservers": {
|
|
||||||
"unregistered_allowed": true,
|
|
||||||
"reject_duplicate_servers": false
|
|
||||||
},
|
|
||||||
"web_api": {
|
|
||||||
"enabled": true,
|
|
||||||
"port": 6000
|
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"mode": 14,
|
|
||||||
"allow_password_login": true,
|
|
||||||
"allow_token_login": true
|
|
||||||
},
|
|
||||||
"logging": {
|
|
||||||
"trace": false,
|
|
||||||
"world_trace": false,
|
|
||||||
"dump_packets_in": false,
|
|
||||||
"dump_packets_out": false
|
|
||||||
},
|
|
||||||
"client_configuration": {
|
|
||||||
"titanium_port": 5998,
|
|
||||||
"titanium_opcodes": "assets/patches/login_opcodes.conf",
|
|
||||||
"sod_port": 5999,
|
|
||||||
"sod_opcodes": "assets/patches/login_opcodes_sod.conf"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,49 +1,21 @@
|
|||||||
// For format details, see https://aka.ms/devcontainer.json. For config options, see the
|
// For format details, see https://aka.ms/vscode-remote/devcontainer.json or this file's README at:
|
||||||
// README at: https://github.com/devcontainers/templates/tree/main/src/ubuntu
|
// https://github.com/microsoft/vscode-dev-containers/tree/v0.101.1/containers/ubuntu-18.04-git
|
||||||
{
|
{
|
||||||
"name": "eqemu",
|
"name": "Ubuntu 18.04 EQEMU",
|
||||||
"build": {
|
// Moved from dockerfile to image so it builds faster
|
||||||
"dockerfile": "Dockerfile.debian.dev"
|
"image": "eqemu/devcontainer:0.0.2",
|
||||||
|
|
||||||
|
// Set *default* container specific settings.json values on container create.
|
||||||
|
"settings": {
|
||||||
|
"terminal.integrated.shell.linux": "/bin/bash"
|
||||||
},
|
},
|
||||||
"appPort": [
|
|
||||||
"5998:5998/udp",
|
"runArgs": [ "--cap-add=SYS_PTRACE", "--security-opt", "seccomp=unconfined" ],
|
||||||
"7000:7000/udp",
|
|
||||||
"7001:7001/udp",
|
// Add the IDs of extensions you want installed when the container is created.
|
||||||
"7002:7002/udp",
|
"extensions": ["ms-vscode.cpptools", "ms-azuretools.vscode-docker"],
|
||||||
"7003:7003/udp",
|
"mounts": ["source=/var/run/docker.sock,target=/var/run/docker.sock,type=bind"],
|
||||||
"7004:7004/udp",
|
|
||||||
"7005:7005/udp",
|
|
||||||
"9000:9000/udp",
|
|
||||||
"9001:9001/udp"
|
|
||||||
],
|
|
||||||
"forwardPorts": [
|
|
||||||
3306
|
|
||||||
],
|
|
||||||
"remoteEnv": {
|
"remoteEnv": {
|
||||||
"LOCALWSF": "${localWorkspaceFolder}",
|
"HOST_PROJECT_PATH": "${localWorkspaceFolder}"
|
||||||
"CONTAINERWSF": "${containerWorkspaceFolder}"
|
}
|
||||||
},
|
}
|
||||||
"containerUser": "vscode",
|
|
||||||
// add ptrace
|
|
||||||
"runArgs": [
|
|
||||||
"--cap-add=SYS_PTRACE",
|
|
||||||
"--security-opt",
|
|
||||||
"seccomp=unconfined"
|
|
||||||
],
|
|
||||||
"customizations": {
|
|
||||||
"vscode": {
|
|
||||||
"extensions": [
|
|
||||||
"ms-vscode.cpptools-extension-pack",
|
|
||||||
"vilicvane.sensitive-replace",
|
|
||||||
"maattdd.gitless",
|
|
||||||
"bibhasdn.unique-lines",
|
|
||||||
"GitHub.copilot",
|
|
||||||
"xackery.make-magic",
|
|
||||||
"Gruntfuggly.todo-tree",
|
|
||||||
"ms-vscode.cmake-tools"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"workspaceFolder": "/src",
|
|
||||||
"workspaceMount": "source=${localWorkspaceFolder},target=/src,type=bind,consistency=cached"
|
|
||||||
}
|
|
||||||
|
|||||||
+6
-88
@@ -1,98 +1,16 @@
|
|||||||
---
|
---
|
||||||
|
|
||||||
kind: pipeline
|
kind: pipeline
|
||||||
type: docker
|
type: docker
|
||||||
name: Build Linux
|
name: EQEmulator Server Linux CI
|
||||||
|
|
||||||
# Limits how many of these builds can run on the drone runner at a time, this isn't about cores
|
# Limits how many of these builds can run on the drone runner at a time, this isn't about cores
|
||||||
concurrency:
|
concurrency:
|
||||||
limit: 1
|
limit: 1
|
||||||
|
|
||||||
volumes:
|
|
||||||
- name: cache
|
|
||||||
host:
|
|
||||||
path: /var/lib/cache-release
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Build Linux X64
|
- name: server-build
|
||||||
image: akkadius/eqemu-server:v14
|
# Source build script https://github.com/Akkadius/akk-stack/blob/master/containers/eqemu-server/Dockerfile#L20
|
||||||
environment:
|
image: akkadius/eqemu-server:latest
|
||||||
GITHUB_TOKEN:
|
|
||||||
from_secret: GH_RELEASE_GITHUB_API_TOKEN
|
|
||||||
RCLONE_CONFIG_REMOTE_TYPE: ftp
|
|
||||||
RCLONE_FTP_HOST: drone.akkadius.com
|
|
||||||
RCLONE_FTP_USER: artifacts
|
|
||||||
RCLONE_FTP_PASS:
|
|
||||||
from_secret: RCLONE_FTP_PASS
|
|
||||||
commands:
|
commands:
|
||||||
- ./utils/scripts/build/linux-build.sh
|
- sudo chown eqemu:eqemu /drone/src/ * -R
|
||||||
volumes:
|
- git submodule init && git submodule update && mkdir -p build && cd build && cmake -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_BOTS=ON -DEQEMU_BUILD_LUA=ON -G 'Unix Makefiles' .. && make -j$((`nproc`-4))
|
||||||
- name: cache
|
|
||||||
path: /home/eqemu/.ccache/
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
kind: pipeline
|
|
||||||
type: exec
|
|
||||||
name: Build Windows
|
|
||||||
|
|
||||||
# Limits how many of these builds can run on the drone runner at a time, this isn't about cores
|
|
||||||
concurrency:
|
|
||||||
limit: 1
|
|
||||||
|
|
||||||
platform:
|
|
||||||
os: windows
|
|
||||||
arch: amd64
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Build Windows X64
|
|
||||||
environment:
|
|
||||||
RCLONE_CONFIG_REMOTE_TYPE: ftp
|
|
||||||
RCLONE_FTP_HOST: drone.akkadius.com
|
|
||||||
RCLONE_FTP_USER: artifacts
|
|
||||||
RCLONE_FTP_PASS:
|
|
||||||
from_secret: RCLONE_FTP_PASS
|
|
||||||
GITHUB_TOKEN:
|
|
||||||
from_secret: GH_RELEASE_GITHUB_API_TOKEN
|
|
||||||
commands:
|
|
||||||
- .\utils\scripts\build\windows-build.ps1
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
kind: pipeline
|
|
||||||
type: docker
|
|
||||||
name: Publish Artifacts to Github
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Upload Artifacts
|
|
||||||
image: akkadius/eqemu-build-releaser:v3
|
|
||||||
environment:
|
|
||||||
RCLONE_CONFIG_REMOTE_TYPE: ftp
|
|
||||||
RCLONE_FTP_HOST: drone.akkadius.com
|
|
||||||
RCLONE_FTP_USER: artifacts
|
|
||||||
RCLONE_FTP_PASS:
|
|
||||||
from_secret: RCLONE_FTP_PASS
|
|
||||||
GH_RELEASE_GITHUB_API_TOKEN:
|
|
||||||
from_secret: GH_RELEASE_GITHUB_API_TOKEN
|
|
||||||
GITHUB_TOKEN:
|
|
||||||
from_secret: GH_RELEASE_GITHUB_API_TOKEN
|
|
||||||
commands:
|
|
||||||
- ./utils/scripts/build/should-release/should-release
|
|
||||||
- rclone config create remote ftp env_auth true > /dev/null
|
|
||||||
- |
|
|
||||||
rclone copy remote: --include "eqemu-server*.zip" .
|
|
||||||
- gh-release --assets=eqemu-server-linux-x64.zip,eqemu-server-windows-x64.zip -y
|
|
||||||
- |
|
|
||||||
rclone delete remote: --include "eqemu-server*.zip"
|
|
||||||
|
|
||||||
trigger:
|
|
||||||
branch:
|
|
||||||
- master
|
|
||||||
event:
|
|
||||||
- push
|
|
||||||
|
|
||||||
depends_on:
|
|
||||||
- Build Windows
|
|
||||||
- Build Linux
|
|
||||||
|
|
||||||
|
|
||||||
-15
@@ -54,18 +54,3 @@ bin/
|
|||||||
/Win32
|
/Win32
|
||||||
/x64
|
/x64
|
||||||
/client_files/**/CMakeFiles/
|
/client_files/**/CMakeFiles/
|
||||||
|
|
||||||
.idea
|
|
||||||
|
|
||||||
# Clangd Generated Files.
|
|
||||||
compile_flags.txt
|
|
||||||
.cache/
|
|
||||||
|
|
||||||
# Build pipeline
|
|
||||||
!utils/scripts/build/
|
|
||||||
!utils/scripts/build/should-release/should-release
|
|
||||||
!utils/scripts/build/should-release/should-release.exe
|
|
||||||
|
|
||||||
# CMake Files
|
|
||||||
cmake-build-relwithdebinfo/*
|
|
||||||
skill-caps.diff
|
|
||||||
|
|||||||
+19
@@ -0,0 +1,19 @@
|
|||||||
|
language: cpp
|
||||||
|
compiler: gcc
|
||||||
|
dist: bionic
|
||||||
|
|
||||||
|
addons:
|
||||||
|
apt:
|
||||||
|
packages:
|
||||||
|
- libmysqlclient-dev
|
||||||
|
- libperl-dev
|
||||||
|
- libboost-dev
|
||||||
|
- liblua5.1-0-dev
|
||||||
|
- zlib1g-dev
|
||||||
|
- uuid-dev
|
||||||
|
- libssl-dev
|
||||||
|
|
||||||
|
script:
|
||||||
|
- cmake -G "Unix Makefiles" -DEQEMU_BUILD_TESTS=ON -DEQEMU_ENABLE_BOTS=ON -DEQEMU_BUILD_LOGIN=ON
|
||||||
|
- make -j2
|
||||||
|
- ./bin/tests
|
||||||
Vendored
+5
-12
@@ -3,20 +3,13 @@
|
|||||||
{
|
{
|
||||||
"name": "Linux",
|
"name": "Linux",
|
||||||
"includePath": [
|
"includePath": [
|
||||||
"${default}",
|
"${workspaceFolder}/**",
|
||||||
"${workspaceFolder}/submodules/fmt/include",
|
"/usr/include/mysql"
|
||||||
"${workspaceFolder}/submodules/cereal/include",
|
|
||||||
"${workspaceFolder}/submodules/glm",
|
|
||||||
"${workspaceFolder}/submodules/libuv/include"
|
|
||||||
],
|
|
||||||
"defines": [
|
|
||||||
"LUA_EQEMU=1"
|
|
||||||
],
|
],
|
||||||
|
"defines": [],
|
||||||
"compilerPath": "/usr/bin/gcc",
|
"compilerPath": "/usr/bin/gcc",
|
||||||
"cStandard": "c17",
|
"cStandard": "c11",
|
||||||
"cppStandard": "gnu++17",
|
"cppStandard": "c++17"
|
||||||
"intelliSenseMode": "linux-gcc-x64",
|
|
||||||
"configurationProvider": "ms-vscode.cmake-tools"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"version": 4
|
"version": 4
|
||||||
|
|||||||
Vendored
-173
@@ -1,173 +0,0 @@
|
|||||||
{
|
|
||||||
// Use IntelliSense to learn about possible attributes.
|
|
||||||
// Hover to view descriptions of existing attributes.
|
|
||||||
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
|
|
||||||
"version": "0.2.0",
|
|
||||||
"configurations": [
|
|
||||||
{
|
|
||||||
"name": "(gdb) attach",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "attach",
|
|
||||||
"program": "${workspaceFolder}/build/bin/world",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) shared_memory",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/shared_memory",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) world",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/world",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) zone",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/zone",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"args": [
|
|
||||||
"",
|
|
||||||
],
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) zone neriakb",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/zone",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"args": [
|
|
||||||
"neriakb",
|
|
||||||
],
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) login",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/loginserver",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) ucs",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/ucs",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "(gdb) queryserv",
|
|
||||||
"type": "cppdbg",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/build/bin/queryserv",
|
|
||||||
"cwd": "${workspaceFolder}/build/bin",
|
|
||||||
"MIMode": "gdb",
|
|
||||||
"setupCommands": [
|
|
||||||
{
|
|
||||||
"description": "Enable pretty-printing for gdb",
|
|
||||||
"text": "-enable-pretty-printing",
|
|
||||||
"ignoreFailures": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Set Disassembly Flavor to Intel",
|
|
||||||
"text": "-gdb-set disassembly-flavor intel",
|
|
||||||
"ignoreFailures": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
Vendored
-136
@@ -1,136 +0,0 @@
|
|||||||
{
|
|
||||||
"editor.detectIndentation": false,
|
|
||||||
"editor.insertSpaces": false,
|
|
||||||
"editor.tabSize": 4,
|
|
||||||
"editor.autoIndent": "full",
|
|
||||||
"editor.trimAutoWhitespace": true,
|
|
||||||
"files.trimTrailingWhitespace": true,
|
|
||||||
//"editor.formatOnSave": true,
|
|
||||||
"search.exclude": {
|
|
||||||
"dependencies": false,
|
|
||||||
},
|
|
||||||
"C_Cpp.default.includePath": [
|
|
||||||
"/usr/include/x86_64-linux-gnu",
|
|
||||||
"/usr/include/lua5.2",
|
|
||||||
"/usr/include/mariadb",
|
|
||||||
"${workspaceFolder}/dependencies/curl_x64/include",
|
|
||||||
"${workspaceFolder}/dependencies/fmt/include",
|
|
||||||
"${workspaceFolder}/dependencies/glm",
|
|
||||||
"${workspaceFolder}/dependencies/libuv/include",
|
|
||||||
"${workspaceFolder}/dependencies/sol2",
|
|
||||||
"${workspaceFolder}/dependencies/zlibng"
|
|
||||||
],
|
|
||||||
"telemetry.enableTelemetry": false,
|
|
||||||
"cmake.buildDirectory": "${workspaceFolder}/build",
|
|
||||||
"cmake.configureArgs": [
|
|
||||||
"-DEQEMU_BUILD_LOGIN=ON",
|
|
||||||
"-DEQEMU_BUILD_TESTS=ON",
|
|
||||||
"-DCMAKE_CXX_COMPILER_LAUNCHER=ccache",
|
|
||||||
"-DEQEMU_ADD_PROFILER=ON",
|
|
||||||
"Ninja"
|
|
||||||
],
|
|
||||||
"cmake.skipConfigureIfCachePresent": true,
|
|
||||||
"cmake.configureOnOpen": false,
|
|
||||||
"files.associations": {
|
|
||||||
"*.ipp": "cpp",
|
|
||||||
"functional": "cpp",
|
|
||||||
"string": "cpp",
|
|
||||||
"iostream": "cpp",
|
|
||||||
"map": "cpp",
|
|
||||||
"fstream": "cpp",
|
|
||||||
"type_traits": "cpp",
|
|
||||||
"utility": "cpp",
|
|
||||||
"cstring": "cpp",
|
|
||||||
"*.tcc": "cpp",
|
|
||||||
"cctype": "cpp",
|
|
||||||
"clocale": "cpp",
|
|
||||||
"cmath": "cpp",
|
|
||||||
"csignal": "cpp",
|
|
||||||
"cstdarg": "cpp",
|
|
||||||
"cstddef": "cpp",
|
|
||||||
"cstdio": "cpp",
|
|
||||||
"cstdlib": "cpp",
|
|
||||||
"ctime": "cpp",
|
|
||||||
"cwchar": "cpp",
|
|
||||||
"cwctype": "cpp",
|
|
||||||
"any": "cpp",
|
|
||||||
"array": "cpp",
|
|
||||||
"atomic": "cpp",
|
|
||||||
"strstream": "cpp",
|
|
||||||
"bit": "cpp",
|
|
||||||
"bitset": "cpp",
|
|
||||||
"chrono": "cpp",
|
|
||||||
"codecvt": "cpp",
|
|
||||||
"compare": "cpp",
|
|
||||||
"complex": "cpp",
|
|
||||||
"concepts": "cpp",
|
|
||||||
"condition_variable": "cpp",
|
|
||||||
"coroutine": "cpp",
|
|
||||||
"cstdint": "cpp",
|
|
||||||
"deque": "cpp",
|
|
||||||
"forward_list": "cpp",
|
|
||||||
"list": "cpp",
|
|
||||||
"set": "cpp",
|
|
||||||
"unordered_map": "cpp",
|
|
||||||
"unordered_set": "cpp",
|
|
||||||
"vector": "cpp",
|
|
||||||
"exception": "cpp",
|
|
||||||
"algorithm": "cpp",
|
|
||||||
"iterator": "cpp",
|
|
||||||
"memory": "cpp",
|
|
||||||
"memory_resource": "cpp",
|
|
||||||
"numeric": "cpp",
|
|
||||||
"optional": "cpp",
|
|
||||||
"random": "cpp",
|
|
||||||
"ratio": "cpp",
|
|
||||||
"regex": "cpp",
|
|
||||||
"source_location": "cpp",
|
|
||||||
"string_view": "cpp",
|
|
||||||
"system_error": "cpp",
|
|
||||||
"tuple": "cpp",
|
|
||||||
"future": "cpp",
|
|
||||||
"initializer_list": "cpp",
|
|
||||||
"iomanip": "cpp",
|
|
||||||
"iosfwd": "cpp",
|
|
||||||
"istream": "cpp",
|
|
||||||
"limits": "cpp",
|
|
||||||
"mutex": "cpp",
|
|
||||||
"new": "cpp",
|
|
||||||
"numbers": "cpp",
|
|
||||||
"ostream": "cpp",
|
|
||||||
"semaphore": "cpp",
|
|
||||||
"sstream": "cpp",
|
|
||||||
"stdexcept": "cpp",
|
|
||||||
"stop_token": "cpp",
|
|
||||||
"streambuf": "cpp",
|
|
||||||
"thread": "cpp",
|
|
||||||
"cfenv": "cpp",
|
|
||||||
"cinttypes": "cpp",
|
|
||||||
"typeindex": "cpp",
|
|
||||||
"typeinfo": "cpp",
|
|
||||||
"valarray": "cpp",
|
|
||||||
"variant": "cpp",
|
|
||||||
"csetjmp": "cpp",
|
|
||||||
"charconv": "cpp",
|
|
||||||
"format": "cpp",
|
|
||||||
"ranges": "cpp",
|
|
||||||
"span": "cpp"
|
|
||||||
},
|
|
||||||
"cmake.statusbar.advanced": {
|
|
||||||
"kit": {
|
|
||||||
"visibility": "hidden",
|
|
||||||
},
|
|
||||||
"debug": {
|
|
||||||
"visibility": "hidden",
|
|
||||||
},
|
|
||||||
"buildTarget": {
|
|
||||||
"visibility": "hidden",
|
|
||||||
},
|
|
||||||
"launch": {
|
|
||||||
"visibility": "hidden",
|
|
||||||
},
|
|
||||||
"ctest": {
|
|
||||||
"visibility": "icon",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Vendored
+164
@@ -0,0 +1,164 @@
|
|||||||
|
{
|
||||||
|
// See https://go.microsoft.com/fwlink/?LinkId=733558
|
||||||
|
// for the documentation about the tasks.json format
|
||||||
|
"version": "2.0.0",
|
||||||
|
"tasks": [
|
||||||
|
{
|
||||||
|
"label": "make",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "cd bin && make",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "make clean",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "cd bin && make clean",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "cmake",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "mkdir -p bin && cd bin && rm CMakeCache.txt && cmake -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_LUA=ON -G 'Unix Makefiles' ..",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher":{
|
||||||
|
"owner": "cpp",
|
||||||
|
"fileLocation": "relative",
|
||||||
|
"pattern":[
|
||||||
|
{
|
||||||
|
"regexp": "([\\w+|\\\\]*\\.\\w+)\\((\\d+)\\)\\: (warning|error) (.*)$",
|
||||||
|
"file": 1,
|
||||||
|
"location": 2,
|
||||||
|
"severity": 3,
|
||||||
|
"message": 4
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "download maps",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "mkdir -p bin && cd bin && wget https://codeload.github.com/Akkadius/EQEmuMaps/zip/master -O maps.zip && unzip -o maps.zip && rm ./maps -rf && mv EQEmuMaps-master maps && rm maps.zip",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "download quests",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "mkdir -p bin && cd bin && cd server && git -C ./quests pull 2> /dev/null || git clone https://github.com/ProjectEQ/projecteqquests.git quests",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "download eqemu_config",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "mkdir -p bin && cd bin && wget --no-check-certificate https://raw.githubusercontent.com/Akkadius/EQEmuInstall/master/eqemu_config_docker.json -O eqemu_config.json",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "rebuild database (mariadb must be started)",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "mkdir -p bin && cd bin && docker run -i --rm --privileged -v ${HOST_PROJECT_PATH}/bin:/src --network=eqemu -it eqemu/server:0.0.3 bash -c './eqemu_server.pl source_peq_db && ./eqemu_server.pl check_db_updates && ./eqemu_server.pl linux_login_server_setup'",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
"$gcc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "zone 7000",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop zone7000 | true && docker network create eqemu | true && docker run -i --rm --name zone7000 --cap-add=SYS_PTRACE --security-opt seccomp=unconfined --privileged -v ${HOST_PROJECT_PATH}/bin:/src --ulimit core=10000000 --network=eqemu -p 7000:7000/udp -e LD_LIBRARY_PATH=/src/ eqemu/server:0.0.3 gdb -ex run --args ./zone dynamic_zone7000:7000",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "zone 7001",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop zone7001 | true && docker network create eqemu | true && docker run -i --rm --name zone7001 --cap-add=SYS_PTRACE --security-opt seccomp=unconfined --privileged -v ${HOST_PROJECT_PATH}/bin:/src --ulimit core=10000000 --network=eqemu -p 7001:7001/udp -e LD_LIBRARY_PATH=/src/ eqemu/server:0.0.3 gdb -ex run --args ./zone dynamic_zone7001:7001",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "loginserver",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop loginserver | true && docker network create eqemu | true && docker run -i --rm --cap-add=SYS_PTRACE --security-opt seccomp=unconfined --privileged -v ${HOST_PROJECT_PATH}/bin:/src --ulimit core=10000000 --network=eqemu --name loginserver -p 5999:5999/udp -p 5998:5998/udp -e LD_LIBRARY_PATH=/src/ eqemu/server:0.0.3 gdb -ex run --args ./loginserver",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "shared_memory, world",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop sharedmemory | true && docker stop world | true && docker network create eqemu | true && docker run --rm -v ${HOST_PROJECT_PATH}/bin:/src --network=eqemu --name sharedmemory eqemu/server:0.0.3 ./shared_memory && docker run --rm -v ${HOST_PROJECT_PATH}/bin:/src --ulimit core=10000000 -e LD_LIBRARY_PATH=/src/ --network=eqemu --name world -p 9000:9000 -p 9000:9000/udp -p 9001:9001 -p 9080:9080 eqemu/server:0.0.3 gdb -ex run ./world",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "queryserv",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop queryserv | true && docker run --rm -v ${HOST_PROJECT_PATH}/bin:/src --ulimit core=10000000 -e LD_LIBRARY_PATH=/src/ --network=eqemu --name queryserv eqemu/server:0.0.3 gdb -ex run ./queryserv",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "mariadb",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop mariadb | true && cd bin && docker network create eqemu | true && docker run --rm -v ${HOST_PROJECT_PATH}/bin/db:/bitnami/mariadb -p 3306:3306 -e MARIADB_DATABASE=peq -e MARIADB_USER=eqemu -e MARIADB_PASSWORD=eqemupass -e ALLOW_EMPTY_PASSWORD=yes --name mariadb --network=eqemu bitnami/mariadb:latest",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "ucs",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "docker stop ucs | true && cd bin && docker network create eqemu | true && docker run --rm -v ${HOST_PROJECT_PATH}/bin:/src -p 7778:7778 --name ucs --network=eqemu eqemu/server:0.0.3 gdb -ex run ./ucs",
|
||||||
|
"group": {
|
||||||
|
"kind": "test",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -40,14 +40,14 @@ Assuming it is starting in c:/projects/eqemu and the x64 dependencies were extra
|
|||||||
|
|
||||||
mkdir build
|
mkdir build
|
||||||
cd build
|
cd build
|
||||||
cmake -G "Visual Studio 15 2017 Win64" -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_ZLIB=ON -DCMAKE_TOOLCHAIN_FILE="c:/projects/eqemu/vcpkg/vcpkg-export-20180828-145455/scripts/buildsystems/vcpkg.cmake" ..
|
cmake -G "Visual Studio 15 2017 Win64" -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_LOGIN=ON -DEQEMU_BUILD_ZLIB=ON -DEQEMU_ENABLE_BOTS=ON -DCMAKE_TOOLCHAIN_FILE="c:/projects/eqemu/vcpkg/vcpkg-export-20180828-145455/scripts/buildsystems/vcpkg.cmake" ..
|
||||||
|
|
||||||
##### Linux
|
##### Linux
|
||||||
Similarly to Windows running CMake on Linux is simple it just omits the toolchain file and uses a different generator.
|
Similarly to Windows running CMake on Linux is simple it just omits the toolchain file and uses a different generator.
|
||||||
|
|
||||||
mkdir build
|
mkdir build
|
||||||
cd build
|
cd build
|
||||||
cmake -G "Unix Makefiles" -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_LOGIN=ON ..
|
cmake -G "Unix Makefiles" -DEQEMU_BUILD_TESTS=ON -DEQEMU_ENABLE_BOTS=ON -DEQEMU_BUILD_LOGIN=ON ..
|
||||||
|
|
||||||
### Building
|
### Building
|
||||||
|
|
||||||
|
|||||||
-4340
File diff suppressed because it is too large
Load Diff
+20
-105
@@ -1,4 +1,4 @@
|
|||||||
CMAKE_MINIMUM_REQUIRED(VERSION 3.12)
|
CMAKE_MINIMUM_REQUIRED(VERSION 3.2)
|
||||||
|
|
||||||
SET(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake/" ${CMAKE_MODULE_PATH})
|
SET(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake/" ${CMAKE_MODULE_PATH})
|
||||||
|
|
||||||
@@ -12,47 +12,16 @@ IF(NOT CMAKE_BUILD_TYPE)
|
|||||||
SET(CMAKE_BUILD_TYPE RelWithDebInfo CACHE STRING "Choose the type of build." FORCE)
|
SET(CMAKE_BUILD_TYPE RelWithDebInfo CACHE STRING "Choose the type of build." FORCE)
|
||||||
ENDIF(NOT CMAKE_BUILD_TYPE)
|
ENDIF(NOT CMAKE_BUILD_TYPE)
|
||||||
|
|
||||||
SET(CMAKE_CXX_STANDARD 20)
|
SET(CMAKE_CXX_STANDARD 14)
|
||||||
SET(CMAKE_CXX_STANDARD_REQUIRED ON)
|
SET(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
SET(CMAKE_CXX_EXTENSIONS OFF)
|
SET(CMAKE_CXX_EXTENSIONS OFF)
|
||||||
|
|
||||||
OPTION(EQEMU_BUILD_STATIC "Build with static linking" OFF)
|
|
||||||
OPTION(EQEMU_BUILD_PCH "Build with precompiled headers (Windows)" ON)
|
|
||||||
|
|
||||||
IF (EQEMU_BUILD_STATIC)
|
|
||||||
SET(BUILD_SHARED_LIBS OFF)
|
|
||||||
SET(CMAKE_FIND_LIBRARY_SUFFIXES ".lib" ".a")
|
|
||||||
MESSAGE(STATUS "Building with static linking")
|
|
||||||
SET(CMAKE_EXE_LINKER_FLAGS "-static-libgcc -static-libstdc++")
|
|
||||||
IF (UNIX)
|
|
||||||
SET(PERL_LIBRARY "/opt/eqemu-perl/lib/5.32.1/x86_64-linux-thread-multi/CORE/libperl.so")
|
|
||||||
SET(PERL_INCLUDE_PATH "/opt/eqemu-perl/lib/5.32.1/x86_64-linux-thread-multi/CORE/")
|
|
||||||
SET(PERL_EXECUTABLE "/opt/eqemu-perl/bin/perl")
|
|
||||||
ENDIF ()
|
|
||||||
ENDIF (EQEMU_BUILD_STATIC)
|
|
||||||
|
|
||||||
|
|
||||||
# Requires libgoogle-perftools-dev google-perftools packages for linux (debian)
|
|
||||||
IF(EQEMU_ADD_PROFILER)
|
|
||||||
SET(CMAKE_EXE_LINKER_FLAGS "-Wl,--no-as-needed,-lprofiler,--as-needed")
|
|
||||||
ENDIF(EQEMU_ADD_PROFILER)
|
|
||||||
|
|
||||||
IF(USE_MAP_MMFS)
|
|
||||||
ADD_DEFINITIONS(-DUSE_MAP_MMFS)
|
|
||||||
ENDIF (USE_MAP_MMFS)
|
|
||||||
|
|
||||||
IF(MSVC)
|
IF(MSVC)
|
||||||
ADD_DEFINITIONS(-D_CRT_SECURE_NO_WARNINGS)
|
ADD_DEFINITIONS(-D_CRT_SECURE_NO_WARNINGS)
|
||||||
ADD_DEFINITIONS(-DNOMINMAX)
|
ADD_DEFINITIONS(-DNOMINMAX)
|
||||||
ADD_DEFINITIONS(-DCRASH_LOGGING)
|
ADD_DEFINITIONS(-DCRASH_LOGGING)
|
||||||
ADD_DEFINITIONS(-D_HAS_AUTO_PTR_ETC) # for Luabind on C++17
|
|
||||||
|
|
||||||
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /MP")
|
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /MP")
|
||||||
|
|
||||||
OPTION(EQEMU_DISABLE_MSVC_WARNINGS "Disable MSVC compile warnings." ON)
|
|
||||||
IF(EQEMU_DISABLE_MSVC_WARNINGS)
|
|
||||||
ADD_DEFINITIONS( "/W0 /D_CRT_SECURE_NO_WARNINGS /wd4005 /wd4996 /nologo /Os")
|
|
||||||
ENDIF(EQEMU_DISABLE_MSVC_WARNINGS)
|
|
||||||
ELSE(MSVC)
|
ELSE(MSVC)
|
||||||
ADD_DEFINITIONS(-DHAS_UNION_SEMUN)
|
ADD_DEFINITIONS(-DHAS_UNION_SEMUN)
|
||||||
ENDIF(MSVC)
|
ENDIF(MSVC)
|
||||||
@@ -149,16 +118,11 @@ ELSE()
|
|||||||
MESSAGE(STATUS "* mbedTLS: MISSING *")
|
MESSAGE(STATUS "* mbedTLS: MISSING *")
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
MESSAGE(STATUS "PERL_INCLUDE_PATH: ${PERL_INCLUDE_PATH}")
|
|
||||||
MESSAGE(STATUS "PERL_LIBRARY: ${PERL_LIBRARY}")
|
|
||||||
MESSAGE(STATUS "PERL_INCLUDE_DIR: ${PERL_INCLUDE_DIR}")
|
|
||||||
MESSAGE(STATUS "PERL_INCLUDE_DIRS: ${PERL_INCLUDE_DIRS}")
|
|
||||||
MESSAGE(STATUS "PERL_LIBRARIES: ${PERL_LIBRARIES}")
|
|
||||||
MESSAGE(STATUS "PERL_VERSION: ${PERL_VERSION}")
|
|
||||||
|
|
||||||
MESSAGE(STATUS "**************************************************")
|
MESSAGE(STATUS "**************************************************")
|
||||||
|
|
||||||
#options
|
#options
|
||||||
|
OPTION(EQEMU_DEPOP_INVALIDATES_CACHE "#repop invalidates the npc_types cache (will cause a larger database hit on #repop but is more convienent)." ON)
|
||||||
|
OPTION(EQEMU_ENABLE_BOTS "Enable Bots" OFF)
|
||||||
OPTION(EQEMU_COMMANDS_LOGGING "Enable GM Command logs" ON)
|
OPTION(EQEMU_COMMANDS_LOGGING "Enable GM Command logs" ON)
|
||||||
OPTION(EQEMU_BUILD_SERVER "Build the game server." ON)
|
OPTION(EQEMU_BUILD_SERVER "Build the game server." ON)
|
||||||
OPTION(EQEMU_BUILD_LOGIN "Build the login server." ON)
|
OPTION(EQEMU_BUILD_LOGIN "Build the login server." ON)
|
||||||
@@ -167,58 +131,21 @@ OPTION(EQEMU_BUILD_TESTS "Build utility tests." OFF)
|
|||||||
OPTION(EQEMU_BUILD_CLIENT_FILES "Build Client Import/Export Data Programs." ON)
|
OPTION(EQEMU_BUILD_CLIENT_FILES "Build Client Import/Export Data Programs." ON)
|
||||||
OPTION(EQEMU_PREFER_LUA "Build with normal Lua even if LuaJIT is found." OFF)
|
OPTION(EQEMU_PREFER_LUA "Build with normal Lua even if LuaJIT is found." OFF)
|
||||||
|
|
||||||
#PRNG options
|
|
||||||
OPTION(EQEMU_ADDITIVE_LFIB_PRNG "Use Additive LFib for PRNG." OFF)
|
|
||||||
MARK_AS_ADVANCED(EQEMU_ADDITIVE_LFIB_PRNG)
|
|
||||||
OPTION(EQEMU_BIASED_INT_DIST "Use biased int dist instead of uniform." OFF)
|
|
||||||
MARK_AS_ADVANCED(EQEMU_BIASED_INT_DIST)
|
|
||||||
SET(EQEMU_CUSTOM_PRNG_ENGINE "" CACHE STRING "Custom random engine. (ex. std::default_random_engine)")
|
|
||||||
MARK_AS_ADVANCED(EQEMU_CUSTOM_PRNG_ENGINE)
|
|
||||||
|
|
||||||
IF(CMAKE_COMPILER_IS_GNUCXX)
|
|
||||||
OPTION(EQEMU_SFMT19937 "Use GCC's extention for SIMD Fast MT19937." OFF)
|
|
||||||
MARK_AS_ADVANCED(EQEMU_SFMT19937)
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
IF(EQEMU_ADDITIVE_LFIB_PRNG)
|
|
||||||
ADD_DEFINITIONS(-DUSE_ADDITIVE_LFIB_PRNG)
|
|
||||||
IF(EQEMU_SFMT19937)
|
|
||||||
MESSAGE(STATUS "SFMT19937 and ADDITITVE_LFIB_PRNG both set, SFMT19937 ignored.")
|
|
||||||
SET(EQEMU_SFMT19937 OFF)
|
|
||||||
ENDIF()
|
|
||||||
IF(NOT EQEMU_CUSTOM_PRNG_ENGINE STREQUAL "")
|
|
||||||
MESSAGE(STATUS "CUSTOM_PRNG_ENGINE and ADDITITVE_LFIB_PRNG both set, CUSTOM_PRNG_ENGINE ignored.")
|
|
||||||
SET(EQEMU_CUSTOM_PRNG_ENGINE "")
|
|
||||||
ENDIF()
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
IF(EQEMU_SFMT19937)
|
|
||||||
ADD_DEFINITIONS(-DUSE_SFMT19937)
|
|
||||||
IF(NOT EQEMU_CUSTOM_PRNG_ENGINE STREQUAL "")
|
|
||||||
MESSAGE(STATUS "CUSTOM_PRNG_ENGINE and SFMT19937 both set, CUSTOM_PRNG_ENGINE ignored.")
|
|
||||||
SET(EQEMU_CUSTOM_PRNG_ENGINE "")
|
|
||||||
ENDIF()
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
IF(NOT EQEMU_CUSTOM_PRNG_ENGINE STREQUAL "")
|
|
||||||
ADD_DEFINITIONS(-DUSE_CUSTOM_PRNG_ENGINE=${EQEMU_CUSTOM_PRNG_ENGINE})
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
IF(EQEMU_BIASED_INT_DIST)
|
|
||||||
ADD_DEFINITIONS(-DBIASED_INT_DIST)
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
IF(EQEMU_COMMANDS_LOGGING)
|
IF(EQEMU_COMMANDS_LOGGING)
|
||||||
ADD_DEFINITIONS(-DCOMMANDS_LOGGING)
|
ADD_DEFINITIONS(-DCOMMANDS_LOGGING)
|
||||||
ENDIF(EQEMU_COMMANDS_LOGGING)
|
ENDIF(EQEMU_COMMANDS_LOGGING)
|
||||||
|
|
||||||
|
IF(EQEMU_ENABLE_BOTS)
|
||||||
|
ADD_DEFINITIONS(-DBOTS)
|
||||||
|
ENDIF(EQEMU_ENABLE_BOTS)
|
||||||
|
|
||||||
#database
|
#database
|
||||||
IF(MySQL_FOUND AND MariaDB_FOUND)
|
IF(MySQL_FOUND AND MariaDB_FOUND)
|
||||||
SET(DATABASE_LIBRARY_SELECTION MariaDB CACHE STRING "Database library to use:
|
SET(DATABASE_LIBRARY_SELECTION MariaDB CACHE STRING "Database library to use:
|
||||||
MySQL
|
MySQL
|
||||||
MariaDB"
|
MariaDB"
|
||||||
)
|
)
|
||||||
|
|
||||||
IF(DATABASE_LIBRARY_SELECTION STREQUAL "MySQL")
|
IF(DATABASE_LIBRARY_SELECTION STREQUAL "MySQL")
|
||||||
SET(DATABASE_LIBRARY_TYPE " MySQL")
|
SET(DATABASE_LIBRARY_TYPE " MySQL")
|
||||||
SET(DATABASE_LIBRARY_LIBS ${MySQL_LIBRARIES})
|
SET(DATABASE_LIBRARY_LIBS ${MySQL_LIBRARIES})
|
||||||
@@ -249,16 +176,13 @@ IF(OpenSSL_FOUND AND MBEDTLS_FOUND)
|
|||||||
OpenSSL
|
OpenSSL
|
||||||
mbedTLS"
|
mbedTLS"
|
||||||
)
|
)
|
||||||
|
|
||||||
IF(TLS_LIBRARY_SELECTION STREQUAL "OpenSSL")
|
IF(TLS_LIBRARY_SELECTION STREQUAL "OpenSSL")
|
||||||
SET(TLS_LIBRARY_TYPE " OpenSSL")
|
SET(TLS_LIBRARY_TYPE " OpenSSL")
|
||||||
SET(TLS_LIBRARY_ENABLED ON)
|
SET(TLS_LIBRARY_ENABLED ON)
|
||||||
SET(TLS_LIBRARY_LIBS ${OPENSSL_LIBRARIES})
|
SET(TLS_LIBRARY_LIBS ${OPENSSL_LIBRARIES})
|
||||||
SET(TLS_LIBRARY_INCLUDE ${OPENSSL_INCLUDE_DIR})
|
SET(TLS_LIBRARY_INCLUDE ${OPENSSL_INCLUDE_DIR})
|
||||||
ADD_DEFINITIONS(-DEQEMU_USE_OPENSSL)
|
ADD_DEFINITIONS(-DEQEMU_USE_OPENSSL)
|
||||||
IF(${OPENSSL_VERSION} VERSION_GREATER_EQUAL "1.1.1")
|
|
||||||
ADD_DEFINITIONS(-DCPPHTTPLIB_OPENSSL_SUPPORT)
|
|
||||||
ENDIF()
|
|
||||||
ELSEIF(TLS_LIBRARY_SELECTION STREQUAL "mbedTLS")
|
ELSEIF(TLS_LIBRARY_SELECTION STREQUAL "mbedTLS")
|
||||||
SET(TLS_LIBRARY_TYPE " mbedTLS")
|
SET(TLS_LIBRARY_TYPE " mbedTLS")
|
||||||
SET(TLS_LIBRARY_ENABLED ON)
|
SET(TLS_LIBRARY_ENABLED ON)
|
||||||
@@ -274,9 +198,6 @@ ELSEIF(OpenSSL_FOUND)
|
|||||||
SET(TLS_LIBRARY_LIBS ${OPENSSL_LIBRARIES})
|
SET(TLS_LIBRARY_LIBS ${OPENSSL_LIBRARIES})
|
||||||
SET(TLS_LIBRARY_INCLUDE ${OPENSSL_INCLUDE_DIR})
|
SET(TLS_LIBRARY_INCLUDE ${OPENSSL_INCLUDE_DIR})
|
||||||
ADD_DEFINITIONS(-DEQEMU_USE_OPENSSL)
|
ADD_DEFINITIONS(-DEQEMU_USE_OPENSSL)
|
||||||
IF(${OPENSSL_VERSION} VERSION_GREATER_EQUAL "1.1.1")
|
|
||||||
ADD_DEFINITIONS(-DCPPHTTPLIB_OPENSSL_SUPPORT)
|
|
||||||
ENDIF()
|
|
||||||
ELSEIF(MBEDTLS_FOUND)
|
ELSEIF(MBEDTLS_FOUND)
|
||||||
SET(TLS_LIBRARY_TYPE " mbedTLS")
|
SET(TLS_LIBRARY_TYPE " mbedTLS")
|
||||||
SET(TLS_LIBRARY_ENABLED ON)
|
SET(TLS_LIBRARY_ENABLED ON)
|
||||||
@@ -343,10 +264,6 @@ ELSE()
|
|||||||
SET(ZLIB_LIBRARY_INCLUDE "${CMAKE_CURRENT_SOURCE_DIR}/libs/zlibng")
|
SET(ZLIB_LIBRARY_INCLUDE "${CMAKE_CURRENT_SOURCE_DIR}/libs/zlibng")
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
IF (EQEMU_BUILD_STATIC)
|
|
||||||
SET(ZLIB_LIBRARY_LIBS libz.a)
|
|
||||||
ENDIF(EQEMU_BUILD_STATIC)
|
|
||||||
|
|
||||||
MESSAGE(STATUS "")
|
MESSAGE(STATUS "")
|
||||||
MESSAGE(STATUS "**************************************************")
|
MESSAGE(STATUS "**************************************************")
|
||||||
MESSAGE(STATUS "* Library Usage *")
|
MESSAGE(STATUS "* Library Usage *")
|
||||||
@@ -376,8 +293,10 @@ INCLUDE_DIRECTORIES(SYSTEM "${CMAKE_CURRENT_SOURCE_DIR}/submodules/recastnavigat
|
|||||||
INCLUDE_DIRECTORIES(SYSTEM "${CMAKE_CURRENT_SOURCE_DIR}/submodules/recastnavigation/Recast/Include")
|
INCLUDE_DIRECTORIES(SYSTEM "${CMAKE_CURRENT_SOURCE_DIR}/submodules/recastnavigation/Recast/Include")
|
||||||
INCLUDE_DIRECTORIES(SYSTEM "${CMAKE_CURRENT_SOURCE_DIR}/submodules/websocketpp")
|
INCLUDE_DIRECTORIES(SYSTEM "${CMAKE_CURRENT_SOURCE_DIR}/submodules/websocketpp")
|
||||||
|
|
||||||
# silence obnoxious deprecation message
|
OPTION(EQEMU_BUILD_LOGGING "Build Logging (To speed up compilation)" ON)
|
||||||
ADD_DEFINITIONS(-DBOOST_BIND_GLOBAL_PLACEHOLDERS)
|
IF(EQEMU_BUILD_LOGGING)
|
||||||
|
ADD_DEFINITIONS(-DBUILD_LOGGING)
|
||||||
|
ENDIF()
|
||||||
|
|
||||||
IF(TLS_LIBRARY_ENABLED)
|
IF(TLS_LIBRARY_ENABLED)
|
||||||
SET(SERVER_LIBS ${SERVER_LIBS} ${TLS_LIBRARY_LIBS})
|
SET(SERVER_LIBS ${SERVER_LIBS} ${TLS_LIBRARY_LIBS})
|
||||||
@@ -391,12 +310,12 @@ ENDIF()
|
|||||||
|
|
||||||
IF(LUA_LIBRARY_ENABLED)
|
IF(LUA_LIBRARY_ENABLED)
|
||||||
OPTION(EQEMU_BUILD_LUA "Build Lua parser." ON)
|
OPTION(EQEMU_BUILD_LUA "Build Lua parser." ON)
|
||||||
|
|
||||||
IF(EQEMU_BUILD_LUA)
|
IF(EQEMU_BUILD_LUA)
|
||||||
ADD_DEFINITIONS(-DLUA_EQEMU)
|
ADD_DEFINITIONS(-DLUA_EQEMU)
|
||||||
SET(ZONE_LIBS ${LUA_LIBRARY_LIBS})
|
SET(ZONE_LIBS ${LUA_LIBRARY_LIBS})
|
||||||
INCLUDE_DIRECTORIES(SYSTEM "${LUA_LIBRARY_INCLUDE}")
|
INCLUDE_DIRECTORIES(SYSTEM "${LUA_LIBRARY_INCLUDE}")
|
||||||
|
|
||||||
OPTION(EQEMU_SANITIZE_LUA_LIBS "Sanitize Lua Libraries (Remove OS and IO standard libraries from being able to run)." ON)
|
OPTION(EQEMU_SANITIZE_LUA_LIBS "Sanitize Lua Libraries (Remove OS and IO standard libraries from being able to run)." ON)
|
||||||
IF(EQEMU_SANITIZE_LUA_LIBS)
|
IF(EQEMU_SANITIZE_LUA_LIBS)
|
||||||
ADD_DEFINITIONS(-DSANITIZE_LUA_LIBS)
|
ADD_DEFINITIONS(-DSANITIZE_LUA_LIBS)
|
||||||
@@ -407,14 +326,10 @@ ENDIF()
|
|||||||
IF(PERL_LIBRARY_ENABLED)
|
IF(PERL_LIBRARY_ENABLED)
|
||||||
OPTION(EQEMU_BUILD_PERL "Build Perl parser." ON)
|
OPTION(EQEMU_BUILD_PERL "Build Perl parser." ON)
|
||||||
IF(EQEMU_BUILD_PERL)
|
IF(EQEMU_BUILD_PERL)
|
||||||
SET(SERVER_LIBS ${SERVER_LIBS} ${PERL_LIBRARY_LIBS} perlbind)
|
SET(SERVER_LIBS ${SERVER_LIBS} ${PERL_LIBRARY_LIBS})
|
||||||
INCLUDE_DIRECTORIES(SYSTEM "${PERL_LIBRARY_INCLUDE}")
|
INCLUDE_DIRECTORIES(SYSTEM "${PERL_LIBRARY_INCLUDE}")
|
||||||
ADD_DEFINITIONS(-DEMBPERL)
|
ADD_DEFINITIONS(-DEMBPERL)
|
||||||
ADD_DEFINITIONS(-DEMBPERL_PLUGIN)
|
ADD_DEFINITIONS(-DEMBPERL_PLUGIN)
|
||||||
ADD_DEFINITIONS(-DPERLBIND_NO_STRICT_SCALAR_TYPES)
|
|
||||||
IF (UNIX AND EQEMU_BUILD_STATIC)
|
|
||||||
SET(SERVER_LIBS ${SERVER_LIBS} libcrypt.a)
|
|
||||||
ENDIF ()
|
|
||||||
ENDIF()
|
ENDIF()
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
@@ -448,13 +363,13 @@ IF(EQEMU_BUILD_SERVER OR EQEMU_BUILD_LOGIN OR EQEMU_BUILD_TESTS OR EQEMU_BUILD_H
|
|||||||
ADD_SUBDIRECTORY(libs)
|
ADD_SUBDIRECTORY(libs)
|
||||||
ADD_SUBDIRECTORY(submodules/fmt)
|
ADD_SUBDIRECTORY(submodules/fmt)
|
||||||
ADD_SUBDIRECTORY(submodules/libuv)
|
ADD_SUBDIRECTORY(submodules/libuv)
|
||||||
|
|
||||||
IF(EQEMU_BUILD_ZLIB)
|
IF(EQEMU_BUILD_ZLIB)
|
||||||
SET(ZLIB_COMPAT ON CACHE BOOL "Compile with zlib compatible API")
|
SET(ZLIB_COMPAT ON CACHE BOOL "Compile with zlib compatible API")
|
||||||
SET(ZLIB_ENABLE_TESTS OFF CACHE BOOL "Build test binaries")
|
SET(ZLIB_ENABLE_TESTS OFF CACHE BOOL "Build test binaries")
|
||||||
ADD_SUBDIRECTORY(libs/zlibng)
|
ADD_SUBDIRECTORY(libs/zlibng)
|
||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
SET(RECASTNAVIGATION_DEMO OFF CACHE BOOL "Build demo")
|
SET(RECASTNAVIGATION_DEMO OFF CACHE BOOL "Build demo")
|
||||||
SET(RECASTNAVIGATION_TESTS OFF CACHE BOOL "Build tests")
|
SET(RECASTNAVIGATION_TESTS OFF CACHE BOOL "Build tests")
|
||||||
SET(RECASTNAVIGATION_EXAMPLES OFF CACHE BOOL "Build examples")
|
SET(RECASTNAVIGATION_EXAMPLES OFF CACHE BOOL "Build examples")
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# EQEmulator Core Server
|
# EQEmulator Core Server
|
||||||
| Drone (Linux x64) | Drone (Windows x64) |
|
|Travis CI (Linux)|Appveyor (Windows x86) |Appveyor (Windows x64) |
|
||||||
|:---:|:---:|
|
|:---:|:---:|:---:|
|
||||||
|[](http://drone.akkadius.com/EQEmu/Server) |[](http://drone.akkadius.com/EQEmu/Server) |
|
|[](https://travis-ci.org/EQEmu/Server) |[](https://ci.appveyor.com/project/KimLS/server) |[](https://ci.appveyor.com/project/KimLS/server-87crp) |
|
||||||
|
|
||||||
***
|
***
|
||||||
|
|
||||||
@@ -18,16 +18,16 @@
|
|||||||
|**Install Count**|||
|
|**Install Count**|||
|
||||||
### > Windows
|
### > Windows
|
||||||
|
|
||||||
* [Install Guide](https://docs.eqemu.io/server/installation/server-installation-windows/)
|
* [Install Guide](https://eqemu.gitbook.io/server/categories/installation/server-installation-windows)
|
||||||
|
|
||||||
### > Debian/Ubuntu/CentOS/Fedora
|
### > Debian/Ubuntu/CentOS/Fedora
|
||||||
|
|
||||||
* [Install Guide](https://docs.eqemu.io/server/installation/server-installation-linux/)
|
* [Install Guide](https://eqemu.gitbook.io/server/categories/installation/server-installation-linux)
|
||||||
|
|
||||||
* You can use curl or wget to kick off the installer (whichever your OS has)
|
* You can use curl or wget to kick off the installer (whichever your OS has)
|
||||||
> curl -O https://raw.githubusercontent.com/EQEmu/Server/master/utils/scripts/linux_installer/install.sh install.sh && chmod 755 install.sh && ./install.sh
|
> curl -O https://raw.githubusercontent.com/EQEmu/Server/master/utils/scripts/linux_installer/install.sh install.sh && chmod 755 install.sh && ./install.sh
|
||||||
|
|
||||||
> wget --no-check-certificate https://raw.githubusercontent.com/EQEmu/Server/master/utils/scripts/linux_installer/install.sh -O install.sh && chmod 755 install.sh && ./install.sh
|
> wget --no-check-certificate https://raw.githubusercontent.com/EQEmu/Server/master/utils/scripts/linux_installer/install.sh -O install.sh && chmod 755 install.sh && ./install.sh
|
||||||
|
|
||||||
## Supported Clients
|
## Supported Clients
|
||||||
|
|
||||||
@@ -56,7 +56,7 @@ forum, although pull requests will be much quicker and easier on all parties.
|
|||||||
|
|
||||||
## Resources
|
## Resources
|
||||||
- [EQEmulator Forums](http://www.eqemulator.org/forums)
|
- [EQEmulator Forums](http://www.eqemulator.org/forums)
|
||||||
- [EQEmulator Wiki](https://docs.eqemu.io/)
|
- [EQEmulator Wiki](https://eqemu.gitbook.io/)
|
||||||
|
|
||||||
## Related Repositories
|
## Related Repositories
|
||||||
* [ProjectEQ Quests](https://github.com/ProjectEQ/projecteqquests)
|
* [ProjectEQ Quests](https://github.com/ProjectEQ/projecteqquests)
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
version: 1.0.{build}
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- master
|
||||||
|
image: Visual Studio 2017
|
||||||
|
configuration: RelWithDebInfo
|
||||||
|
clone_folder: c:\projects\eqemu
|
||||||
|
init:
|
||||||
|
- ps: git config --global core.autocrlf input
|
||||||
|
cache: c:\tools\vcpkg\installed\
|
||||||
|
before_build:
|
||||||
|
- ps: "$wc = New-Object System.Net.WebClient\n$wc.DownloadFile(\"http://strawberryperl.com/download/5.26.2.1/strawberry-perl-5.26.2.1-64bit-portable.zip\", \"c:\\projects\\eqemu\\strawberry-perl-5.26.2.1-64bit-portable.zip\")\ncd c:\\projects\\eqemu\n7z x c:/projects/eqemu/strawberry-perl-5.26.2.1-64bit-portable.zip -oc:/projects/eqemu/strawberry-perl-portable -y\n(Get-Content C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/config.h).replace('#define PERL_STATIC_INLINE static __inline__', '#define PERL_STATIC_INLINE static __inline') | Set-Content C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/config.h\nvcpkg install boost-geometry:x64-windows boost-dynamic-bitset:x64-windows luajit:x64-windows libsodium:x64-windows libmysql:x64-windows openssl:x64-windows zlib:x64-windows \nmkdir build\ncd build\ncmake -G \"Visual Studio 15 2017 Win64\" -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_LOGIN=ON -EQEMU_ENABLE_BOTS=ON -DPERL_EXECUTABLE=\"C:/projects/eqemu/strawberry-perl-portable/perl/bin/perl.exe\" -DPERL_INCLUDE_PATH=\"C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE\" -DPERL_LIBRARY=\"C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/libperl526.a\" -DCMAKE_TOOLCHAIN_FILE=\"c:/tools/vcpkg/scripts/buildsystems/vcpkg.cmake\" .."
|
||||||
|
build:
|
||||||
|
project: C:\projects\eqemu\build\EQEmu.sln
|
||||||
|
parallel: true
|
||||||
|
verbosity: minimal
|
||||||
|
after_build:
|
||||||
|
- cmd: >-
|
||||||
|
7z a build_x64-bots.zip C:\projects\eqemu\build\bin\RelWithDebInfo\*.exe C:\projects\eqemu\build\bin\RelWithDebInfo\*.dll C:\projects\eqemu\build\bin\RelWithDebInfo\*.pdb C:\projects\eqemu\build\libs\zlibng\RelWithDebInfo\*.dll
|
||||||
|
|
||||||
|
appveyor PushArtifact build_x64-bots.zip
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
version: 1.0.{build}
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- master
|
||||||
|
image: Visual Studio 2017
|
||||||
|
configuration: RelWithDebInfo
|
||||||
|
clone_folder: c:\projects\eqemu
|
||||||
|
init:
|
||||||
|
- ps: git config --global core.autocrlf input
|
||||||
|
cache: c:\tools\vcpkg\installed\
|
||||||
|
before_build:
|
||||||
|
- ps: "$wc = New-Object System.Net.WebClient\n$wc.DownloadFile(\"http://strawberryperl.com/download/5.26.2.1/strawberry-perl-5.26.2.1-64bit-portable.zip\", \"c:\\projects\\eqemu\\strawberry-perl-5.26.2.1-64bit-portable.zip\")\ncd c:\\projects\\eqemu\n7z x c:/projects/eqemu/strawberry-perl-5.26.2.1-64bit-portable.zip -oc:/projects/eqemu/strawberry-perl-portable -y\n(Get-Content C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/config.h).replace('#define PERL_STATIC_INLINE static __inline__', '#define PERL_STATIC_INLINE static __inline') | Set-Content C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/config.h\nvcpkg install boost-geometry:x64-windows boost-dynamic-bitset:x64-windows luajit:x64-windows libsodium:x64-windows libmysql:x64-windows openssl:x64-windows zlib:x64-windows \nmkdir build\ncd build\ncmake -G \"Visual Studio 15 2017 Win64\" -DEQEMU_BUILD_TESTS=ON -DEQEMU_BUILD_LOGIN=ON -EQEMU_ENABLE_BOTS=OFF -DPERL_EXECUTABLE=\"C:/projects/eqemu/strawberry-perl-portable/perl/bin/perl.exe\" -DPERL_INCLUDE_PATH=\"C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE\" -DPERL_LIBRARY=\"C:/projects/eqemu/strawberry-perl-portable/perl/lib/CORE/libperl526.a\" -DCMAKE_TOOLCHAIN_FILE=\"c:/tools/vcpkg/scripts/buildsystems/vcpkg.cmake\" .."
|
||||||
|
build:
|
||||||
|
project: C:\projects\eqemu\build\EQEmu.sln
|
||||||
|
parallel: true
|
||||||
|
verbosity: minimal
|
||||||
|
after_build:
|
||||||
|
- cmd: >-
|
||||||
|
7z a build_x64-no-bots.zip C:\projects\eqemu\build\bin\RelWithDebInfo\*.exe C:\projects\eqemu\build\bin\RelWithDebInfo\*.dll C:\projects\eqemu\build\bin\RelWithDebInfo\*.pdb C:\projects\eqemu\build\libs\zlibng\RelWithDebInfo\*.dll
|
||||||
|
|
||||||
|
appveyor PushArtifact build_x64-no-bots.zip
|
||||||
+1
-1
@@ -3,7 +3,7 @@
|
|||||||
############################################
|
############################################
|
||||||
#
|
#
|
||||||
# New changelog can be found here
|
# New changelog can be found here
|
||||||
# https://docs.eqemu.io/server/changelog/server
|
# https://eqemu.gitbook.io/changelog/
|
||||||
#
|
#
|
||||||
############################################
|
############################################
|
||||||
# Deprecated
|
# Deprecated
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
CMAKE_MINIMUM_REQUIRED(VERSION 3.5)
|
CMAKE_MINIMUM_REQUIRED(VERSION 3.2)
|
||||||
|
|
||||||
add_subdirectory(import)
|
add_subdirectory(import)
|
||||||
add_subdirectory(export)
|
add_subdirectory(export)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
CMAKE_MINIMUM_REQUIRED(VERSION 3.12)
|
CMAKE_MINIMUM_REQUIRED(VERSION 3.2)
|
||||||
|
|
||||||
SET(export_sources
|
SET(export_sources
|
||||||
main.cpp
|
main.cpp
|
||||||
|
|||||||
+152
-64
@@ -25,25 +25,11 @@
|
|||||||
#include "../../common/platform.h"
|
#include "../../common/platform.h"
|
||||||
#include "../../common/crash.h"
|
#include "../../common/crash.h"
|
||||||
#include "../../common/rulesys.h"
|
#include "../../common/rulesys.h"
|
||||||
#include "../../common/strings.h"
|
#include "../../common/string_util.h"
|
||||||
#include "../../common/content/world_content_service.h"
|
#include "../../common/content/world_content_service.h"
|
||||||
#include "../../common/zone_store.h"
|
|
||||||
#include "../../common/path_manager.h"
|
|
||||||
#include "../../common/repositories/base_data_repository.h"
|
|
||||||
#include "../../common/repositories/db_str_repository.h"
|
|
||||||
#include "../../common/repositories/skill_caps_repository.h"
|
|
||||||
#include "../../common/repositories/spells_new_repository.h"
|
|
||||||
#include "../../common/file.h"
|
|
||||||
#include "../../common/events/player_event_logs.h"
|
|
||||||
#include "../../common/skill_caps.h"
|
|
||||||
#include "../../common/evolving_items.h"
|
|
||||||
|
|
||||||
EQEmuLogSys LogSys;
|
EQEmuLogSys LogSys;
|
||||||
WorldContentService content_service;
|
WorldContentService content_service;
|
||||||
ZoneStore zone_store;
|
|
||||||
PathManager path;
|
|
||||||
PlayerEventLogs player_event_logs;
|
|
||||||
EvolvingItemsManager evolving_items_manager;
|
|
||||||
|
|
||||||
void ExportSpells(SharedDatabase *db);
|
void ExportSpells(SharedDatabase *db);
|
||||||
void ExportSkillCaps(SharedDatabase *db);
|
void ExportSkillCaps(SharedDatabase *db);
|
||||||
@@ -56,8 +42,6 @@ int main(int argc, char **argv)
|
|||||||
LogSys.LoadLogSettingsDefaults();
|
LogSys.LoadLogSettingsDefaults();
|
||||||
set_exception_handler();
|
set_exception_handler();
|
||||||
|
|
||||||
path.LoadPaths();
|
|
||||||
|
|
||||||
LogInfo("Client Files Export Utility");
|
LogInfo("Client Files Export Utility");
|
||||||
if (!EQEmuConfig::LoadConfig()) {
|
if (!EQEmuConfig::LoadConfig()) {
|
||||||
LogError("Unable to load configuration file");
|
LogError("Unable to load configuration file");
|
||||||
@@ -96,30 +80,31 @@ int main(int argc, char **argv)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
content_db.SetMySQL(database);
|
content_db.SetMysql(database.getMySQL());
|
||||||
}
|
}
|
||||||
|
|
||||||
LogSys.SetDatabase(&database)
|
database.LoadLogSettings(LogSys.log_settings);
|
||||||
->SetLogPath(path.GetLogPath())
|
LogSys.StartFileLogs();
|
||||||
->LoadLogDatabaseSettings()
|
|
||||||
->StartFileLogs();
|
|
||||||
|
|
||||||
std::string export_type;
|
std::string arg_1;
|
||||||
|
|
||||||
if (argv[1]) {
|
if (argv[1]) {
|
||||||
export_type = argv[1];
|
arg_1 = argv[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Strings::EqualFold(export_type, "spells")) {
|
if (arg_1 == "spells") {
|
||||||
ExportSpells(&content_db);
|
ExportSpells(&content_db);
|
||||||
return 0;
|
return 0;
|
||||||
} else if (Strings::EqualFold(export_type, "skills")) {
|
}
|
||||||
|
if (arg_1 == "skills") {
|
||||||
ExportSkillCaps(&content_db);
|
ExportSkillCaps(&content_db);
|
||||||
return 0;
|
return 0;
|
||||||
} else if (Strings::EqualFold(export_type, "basedata") || Strings::EqualFold(export_type, "base_data")) {
|
}
|
||||||
|
if (arg_1 == "basedata") {
|
||||||
ExportBaseData(&content_db);
|
ExportBaseData(&content_db);
|
||||||
return 0;
|
return 0;
|
||||||
} else if (Strings::EqualFold(export_type, "dbstr") || Strings::EqualFold(export_type, "dbstring")) {
|
}
|
||||||
|
if (arg_1 == "dbstring") {
|
||||||
ExportDBStrings(&database);
|
ExportDBStrings(&database);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -134,79 +119,182 @@ int main(int argc, char **argv)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportSpells(SharedDatabase* db)
|
void ExportSpells(SharedDatabase *db)
|
||||||
{
|
{
|
||||||
std::ofstream file(fmt::format("{}/export/spells_us.txt", path.GetServerPath()));
|
LogInfo("Exporting Spells");
|
||||||
if (!file || !file.is_open()) {
|
|
||||||
|
FILE *f = fopen("export/spells_us.txt", "w");
|
||||||
|
if (!f) {
|
||||||
LogError("Unable to open export/spells_us.txt to write, skipping.");
|
LogError("Unable to open export/spells_us.txt to write, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& lines = SpellsNewRepository::GetSpellFileLines(*db);
|
const std::string query = "SELECT * FROM spells_new ORDER BY id";
|
||||||
|
auto results = db->QueryDatabase(query);
|
||||||
|
|
||||||
const std::string& file_string = Strings::Implode("\n", lines);
|
if (results.Success()) {
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
std::string line;
|
||||||
|
unsigned int fields = results.ColumnCount();
|
||||||
|
for (unsigned int i = 0; i < fields; ++i) {
|
||||||
|
if (i != 0) {
|
||||||
|
line.push_back('^');
|
||||||
|
}
|
||||||
|
|
||||||
file << file_string;
|
if (row[i] != nullptr) {
|
||||||
|
line += row[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file.close();
|
fprintf(f, "%s\n", line.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
}
|
||||||
|
|
||||||
LogInfo("Exported [{}] Spell{}", lines.size(), lines.size() != 1 ? "s" : "");
|
fclose(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportSkillCaps(SharedDatabase* db)
|
bool SkillUsable(SharedDatabase *db, int skill_id, int class_id)
|
||||||
{
|
{
|
||||||
std::ofstream file(fmt::format("{}/export/SkillCaps.txt", path.GetServerPath()));
|
|
||||||
if (!file || !file.is_open()) {
|
bool res = false;
|
||||||
|
|
||||||
|
std::string query = StringFormat(
|
||||||
|
"SELECT max(cap) FROM skill_caps WHERE class=%d AND skillID=%d",
|
||||||
|
class_id, skill_id
|
||||||
|
);
|
||||||
|
auto results = db->QueryDatabase(query);
|
||||||
|
if (!results.Success()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (results.RowCount() == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
if (row[0] && atoi(row[0]) > 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
int GetSkill(SharedDatabase *db, int skill_id, int class_id, int level)
|
||||||
|
{
|
||||||
|
|
||||||
|
std::string query = StringFormat(
|
||||||
|
"SELECT cap FROM skill_caps WHERE class=%d AND skillID=%d AND level=%d",
|
||||||
|
class_id, skill_id, level
|
||||||
|
);
|
||||||
|
auto results = db->QueryDatabase(query);
|
||||||
|
if (!results.Success()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (results.RowCount() == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
return atoi(row[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExportSkillCaps(SharedDatabase *db)
|
||||||
|
{
|
||||||
|
LogInfo("Exporting Skill Caps");
|
||||||
|
|
||||||
|
FILE *f = fopen("export/SkillCaps.txt", "w");
|
||||||
|
if (!f) {
|
||||||
LogError("Unable to open export/SkillCaps.txt to write, skipping.");
|
LogError("Unable to open export/SkillCaps.txt to write, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& lines = SkillCapsRepository::GetSkillCapFileLines(*db);
|
for (int cl = 1; cl <= 16; ++cl) {
|
||||||
|
for (int skill = 0; skill <= 77; ++skill) {
|
||||||
|
if (SkillUsable(db, skill, cl)) {
|
||||||
|
int previous_cap = 0;
|
||||||
|
for (int level = 1; level <= 100; ++level) {
|
||||||
|
int cap = GetSkill(db, skill, cl, level);
|
||||||
|
if (cap < previous_cap) {
|
||||||
|
cap = previous_cap;
|
||||||
|
}
|
||||||
|
|
||||||
const std::string& file_string = Strings::Implode("\n", lines);
|
fprintf(f, "%d^%d^%d^%d^0\n", cl, skill, level, cap);
|
||||||
|
previous_cap = cap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file << file_string;
|
fclose(f);
|
||||||
|
|
||||||
file.close();
|
|
||||||
|
|
||||||
LogInfo("Exported [{}] Skill Cap{}", lines.size(), lines.size() != 1 ? "s" : "");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportBaseData(SharedDatabase *db)
|
void ExportBaseData(SharedDatabase *db)
|
||||||
{
|
{
|
||||||
std::ofstream file(fmt::format("{}/export/BaseData.txt", path.GetServerPath()));
|
LogInfo("Exporting Base Data");
|
||||||
if (!file || !file.is_open()) {
|
|
||||||
|
FILE *f = fopen("export/BaseData.txt", "w");
|
||||||
|
if (!f) {
|
||||||
LogError("Unable to open export/BaseData.txt to write, skipping.");
|
LogError("Unable to open export/BaseData.txt to write, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& lines = BaseDataRepository::GetBaseDataFileLines(*db);
|
const std::string query = "SELECT * FROM base_data ORDER BY level, class";
|
||||||
|
auto results = db->QueryDatabase(query);
|
||||||
|
if (results.Success()) {
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
std::string line;
|
||||||
|
unsigned int fields = results.ColumnCount();
|
||||||
|
for (unsigned int rowIndex = 0; rowIndex < fields; ++rowIndex) {
|
||||||
|
if (rowIndex != 0) {
|
||||||
|
line.push_back('^');
|
||||||
|
}
|
||||||
|
|
||||||
const std::string& file_string = Strings::Implode("\n", lines);
|
if (row[rowIndex] != nullptr) {
|
||||||
|
line += row[rowIndex];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file << file_string;
|
fprintf(f, "%s\n", line.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file.close();
|
fclose(f);
|
||||||
|
|
||||||
LogInfo("Exported [{}] Base Data Entr{}", lines.size(), lines.size() != 1 ? "ies" : "y");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportDBStrings(SharedDatabase *db)
|
void ExportDBStrings(SharedDatabase *db)
|
||||||
{
|
{
|
||||||
std::ofstream file(fmt::format("{}/export/dbstr_us.txt", path.GetServerPath()));
|
LogInfo("Exporting DB Strings");
|
||||||
if (!file || !file.is_open()) {
|
|
||||||
|
FILE *f = fopen("export/dbstr_us.txt", "w");
|
||||||
|
if (!f) {
|
||||||
LogError("Unable to open export/dbstr_us.txt to write, skipping.");
|
LogError("Unable to open export/dbstr_us.txt to write, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& lines = DbStrRepository::GetDBStrFileLines(*db);
|
fprintf(f, "Major^Minor^String(New)\n");
|
||||||
|
const std::string query = "SELECT * FROM db_str ORDER BY id, type";
|
||||||
|
auto results = db->QueryDatabase(query);
|
||||||
|
if (results.Success()) {
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
std::string line;
|
||||||
|
unsigned int fields = results.ColumnCount();
|
||||||
|
for (unsigned int rowIndex = 0; rowIndex < fields; ++rowIndex) {
|
||||||
|
if (rowIndex != 0) {
|
||||||
|
line.push_back('^');
|
||||||
|
}
|
||||||
|
|
||||||
const std::string& file_string = Strings::Implode("\n", lines);
|
if (row[rowIndex] != nullptr) {
|
||||||
|
line += row[rowIndex];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file << file_string;
|
fprintf(f, "%s\n", line.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
file.close();
|
fclose(f);
|
||||||
|
|
||||||
LogInfo("Exported [{}] Database String{}", lines.size(), lines.size() != 1 ? "s" : "");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
CMAKE_MINIMUM_REQUIRED(VERSION 3.12)
|
CMAKE_MINIMUM_REQUIRED(VERSION 3.2)
|
||||||
|
|
||||||
SET(import_sources
|
SET(import_sources
|
||||||
main.cpp
|
main.cpp
|
||||||
|
|||||||
@@ -23,21 +23,11 @@
|
|||||||
#include "../../common/platform.h"
|
#include "../../common/platform.h"
|
||||||
#include "../../common/crash.h"
|
#include "../../common/crash.h"
|
||||||
#include "../../common/rulesys.h"
|
#include "../../common/rulesys.h"
|
||||||
#include "../../common/strings.h"
|
#include "../../common/string_util.h"
|
||||||
#include "../../common/content/world_content_service.h"
|
#include "../../common/content/world_content_service.h"
|
||||||
#include "../../common/zone_store.h"
|
|
||||||
#include "../../common/path_manager.h"
|
|
||||||
#include "../../common/repositories/base_data_repository.h"
|
|
||||||
#include "../../common/file.h"
|
|
||||||
#include "../../common/events/player_event_logs.h"
|
|
||||||
#include "../../common/evolving_items.h"
|
|
||||||
|
|
||||||
EQEmuLogSys LogSys;
|
EQEmuLogSys LogSys;
|
||||||
WorldContentService content_service;
|
WorldContentService content_service;
|
||||||
ZoneStore zone_store;
|
|
||||||
PathManager path;
|
|
||||||
PlayerEventLogs player_event_logs;
|
|
||||||
EvolvingItemsManager evolving_items_manager;
|
|
||||||
|
|
||||||
void ImportSpells(SharedDatabase *db);
|
void ImportSpells(SharedDatabase *db);
|
||||||
void ImportSkillCaps(SharedDatabase *db);
|
void ImportSkillCaps(SharedDatabase *db);
|
||||||
@@ -49,8 +39,6 @@ int main(int argc, char **argv) {
|
|||||||
LogSys.LoadLogSettingsDefaults();
|
LogSys.LoadLogSettingsDefaults();
|
||||||
set_exception_handler();
|
set_exception_handler();
|
||||||
|
|
||||||
path.LoadPaths();
|
|
||||||
|
|
||||||
LogInfo("Client Files Import Utility");
|
LogInfo("Client Files Import Utility");
|
||||||
if(!EQEmuConfig::LoadConfig()) {
|
if(!EQEmuConfig::LoadConfig()) {
|
||||||
LogError("Unable to load configuration file.");
|
LogError("Unable to load configuration file.");
|
||||||
@@ -89,13 +77,11 @@ int main(int argc, char **argv) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
content_db.SetMySQL(database);
|
content_db.SetMysql(database.getMySQL());
|
||||||
}
|
}
|
||||||
|
|
||||||
LogSys.SetDatabase(&database)
|
database.LoadLogSettings(LogSys.log_settings);
|
||||||
->SetLogPath(path.GetLogPath())
|
LogSys.StartFileLogs();
|
||||||
->LoadLogDatabaseSettings()
|
|
||||||
->StartFileLogs();
|
|
||||||
|
|
||||||
ImportSpells(&content_db);
|
ImportSpells(&content_db);
|
||||||
ImportSkillCaps(&content_db);
|
ImportSkillCaps(&content_db);
|
||||||
@@ -138,10 +124,9 @@ bool IsStringField(int i) {
|
|||||||
|
|
||||||
void ImportSpells(SharedDatabase *db) {
|
void ImportSpells(SharedDatabase *db) {
|
||||||
LogInfo("Importing Spells");
|
LogInfo("Importing Spells");
|
||||||
std::string file = fmt::format("{}/import/spells_us.txt", path.GetServerPath());
|
FILE *f = fopen("import/spells_us.txt", "r");
|
||||||
FILE *f = fopen(file.c_str(), "r");
|
|
||||||
if(!f) {
|
if(!f) {
|
||||||
LogError("Unable to open {} to read, skipping.", file);
|
LogError("Unable to open import/spells_us.txt to read, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -160,8 +145,8 @@ void ImportSpells(SharedDatabase *db) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string escaped = ::Strings::Escape(buffer);
|
std::string escaped = ::EscapeString(buffer);
|
||||||
auto split = Strings::Split(escaped, '^');
|
auto split = SplitString(escaped, '^');
|
||||||
int line_columns = (int)split.size();
|
int line_columns = (int)split.size();
|
||||||
|
|
||||||
std::string sql;
|
std::string sql;
|
||||||
@@ -228,10 +213,9 @@ void ImportSpells(SharedDatabase *db) {
|
|||||||
void ImportSkillCaps(SharedDatabase *db) {
|
void ImportSkillCaps(SharedDatabase *db) {
|
||||||
LogInfo("Importing Skill Caps");
|
LogInfo("Importing Skill Caps");
|
||||||
|
|
||||||
std::string file = fmt::format("{}/import/SkillCaps.txt", path.GetServerPath());
|
FILE *f = fopen("import/SkillCaps.txt", "r");
|
||||||
FILE *f = fopen(file.c_str(), "r");
|
|
||||||
if(!f) {
|
if(!f) {
|
||||||
LogError("Unable to open {} to read, skipping.", file);
|
LogError("Unable to open import/SkillCaps.txt to read, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -240,17 +224,17 @@ void ImportSkillCaps(SharedDatabase *db) {
|
|||||||
|
|
||||||
char buffer[2048];
|
char buffer[2048];
|
||||||
while(fgets(buffer, 2048, f)) {
|
while(fgets(buffer, 2048, f)) {
|
||||||
auto split = Strings::Split(buffer, '^');
|
auto split = SplitString(buffer, '^');
|
||||||
|
|
||||||
if(split.size() < 4) {
|
if(split.size() < 4) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
int class_id, skill_id, level, cap;
|
int class_id, skill_id, level, cap;
|
||||||
class_id = Strings::ToInt(split[0].c_str());
|
class_id = atoi(split[0].c_str());
|
||||||
skill_id = Strings::ToInt(split[1].c_str());
|
skill_id = atoi(split[1].c_str());
|
||||||
level = Strings::ToInt(split[2].c_str());
|
level = atoi(split[2].c_str());
|
||||||
cap = Strings::ToInt(split[3].c_str());
|
cap = atoi(split[3].c_str());
|
||||||
|
|
||||||
std::string sql = StringFormat("INSERT INTO skill_caps(class, skillID, level, cap) VALUES(%d, %d, %d, %d)",
|
std::string sql = StringFormat("INSERT INTO skill_caps(class, skillID, level, cap) VALUES(%d, %d, %d, %d)",
|
||||||
class_id, skill_id, level, cap);
|
class_id, skill_id, level, cap);
|
||||||
@@ -261,54 +245,57 @@ void ImportSkillCaps(SharedDatabase *db) {
|
|||||||
fclose(f);
|
fclose(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImportBaseData(SharedDatabase *db)
|
void ImportBaseData(SharedDatabase *db) {
|
||||||
{
|
|
||||||
LogInfo("Importing Base Data");
|
LogInfo("Importing Base Data");
|
||||||
|
|
||||||
const std::string& file_name = fmt::format("{}/import/BaseData.txt", path.GetServerPath());
|
FILE *f = fopen("import/BaseData.txt", "r");
|
||||||
|
if(!f) {
|
||||||
const auto& file_contents = File::GetContents(file_name);
|
LogError("Unable to open import/BaseData.txt to read, skipping.");
|
||||||
if (!file_contents.error.empty()) {
|
return;
|
||||||
LogError("{}", file_contents.error);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
db->QueryDatabase("DELETE FROM base_data");
|
std::string delete_sql = "DELETE FROM base_data";
|
||||||
|
db->QueryDatabase(delete_sql);
|
||||||
|
|
||||||
std::vector<BaseDataRepository::BaseData> v;
|
char buffer[2048];
|
||||||
|
while(fgets(buffer, 2048, f)) {
|
||||||
|
auto split = SplitString(buffer, '^');
|
||||||
|
|
||||||
auto e = BaseDataRepository::NewEntity();
|
if(split.size() < 10) {
|
||||||
|
|
||||||
for (const auto& line: Strings::Split(file_contents.contents, "\n")) {
|
|
||||||
const auto& line_data = Strings::Split(line, '^');
|
|
||||||
|
|
||||||
if (line_data.size() < 10) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
e.level = static_cast<uint8_t>(Strings::ToUnsignedInt(line_data[0]));
|
std::string sql;
|
||||||
e.class_ = static_cast<uint8_t>(Strings::ToUnsignedInt(line_data[1]));
|
int level, class_id;
|
||||||
e.hp = Strings::ToFloat(line_data[2]);
|
double hp, mana, end, unk1, unk2, hp_fac, mana_fac, end_fac;
|
||||||
e.mana = Strings::ToFloat(line_data[3]);
|
|
||||||
e.end = Strings::ToFloat(line_data[4]);
|
|
||||||
e.hp_regen = Strings::ToFloat(line_data[5]);
|
|
||||||
e.end_regen = Strings::ToFloat(line_data[6]);
|
|
||||||
e.hp_fac = Strings::ToFloat(line_data[7]);
|
|
||||||
e.mana_fac = Strings::ToFloat(line_data[8]);
|
|
||||||
e.end_fac = Strings::ToFloat(line_data[9]);
|
|
||||||
|
|
||||||
v.emplace_back(e);
|
level = atoi(split[0].c_str());
|
||||||
|
class_id = atoi(split[1].c_str());
|
||||||
|
hp = atof(split[2].c_str());
|
||||||
|
mana = atof(split[3].c_str());
|
||||||
|
end = atof(split[4].c_str());
|
||||||
|
unk1 = atof(split[5].c_str());
|
||||||
|
unk2 = atof(split[6].c_str());
|
||||||
|
hp_fac = atof(split[7].c_str());
|
||||||
|
mana_fac = atof(split[8].c_str());
|
||||||
|
end_fac = atof(split[9].c_str());
|
||||||
|
|
||||||
|
sql = StringFormat("INSERT INTO base_data(level, class, hp, mana, end, unk1, unk2, hp_fac, "
|
||||||
|
"mana_fac, end_fac) VALUES(%d, %d, %f, %f, %f, %f, %f, %f, %f, %f)",
|
||||||
|
level, class_id, hp, mana, end, unk1, unk2, hp_fac, mana_fac, end_fac);
|
||||||
|
|
||||||
|
db->QueryDatabase(sql);
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseDataRepository::InsertMany(*db, v);
|
fclose(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImportDBStrings(SharedDatabase *db) {
|
void ImportDBStrings(SharedDatabase *db) {
|
||||||
LogInfo("Importing DB Strings");
|
LogInfo("Importing DB Strings");
|
||||||
|
|
||||||
std::string file = fmt::format("{}/import/dbstr_us.txt", path.GetServerPath());
|
FILE *f = fopen("import/dbstr_us.txt", "r");
|
||||||
FILE *f = fopen(file.c_str(), "r");
|
|
||||||
if(!f) {
|
if(!f) {
|
||||||
LogError("Unable to open {} to read, skipping.", file);
|
LogError("Unable to open import/dbstr_us.txt to read, skipping.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -330,7 +317,7 @@ void ImportDBStrings(SharedDatabase *db) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto split = Strings::Split(buffer, '^');
|
auto split = SplitString(buffer, '^');
|
||||||
|
|
||||||
if(split.size() < 2) {
|
if(split.size() < 2) {
|
||||||
continue;
|
continue;
|
||||||
@@ -340,11 +327,11 @@ void ImportDBStrings(SharedDatabase *db) {
|
|||||||
int id, type;
|
int id, type;
|
||||||
std::string value;
|
std::string value;
|
||||||
|
|
||||||
id = Strings::ToInt(split[0].c_str());
|
id = atoi(split[0].c_str());
|
||||||
type = Strings::ToInt(split[1].c_str());
|
type = atoi(split[1].c_str());
|
||||||
|
|
||||||
if(split.size() >= 3) {
|
if(split.size() >= 3) {
|
||||||
value = ::Strings::Escape(split[2]);
|
value = ::EscapeString(split[2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
sql = StringFormat("INSERT INTO db_str(id, type, value) VALUES(%u, %u, '%s')",
|
sql = StringFormat("INSERT INTO db_str(id, type, value) VALUES(%u, %u, '%s')",
|
||||||
|
|||||||
+702
-782
File diff suppressed because it is too large
Load Diff
@@ -1,147 +0,0 @@
|
|||||||
/* EQEMu: Everquest Server Emulator
|
|
||||||
Copyright (C) 2001-2021 EQEMu Development Team (http://eqemulator.net)
|
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation; version 2 of the License.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY except by those people which sell it, which
|
|
||||||
are required to give you total support for your newly bought product;
|
|
||||||
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
||||||
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program; if not, write to the Free Software
|
|
||||||
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
#include <limits>
|
|
||||||
#include <type_traits>
|
|
||||||
#include <algorithm>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This is an additive lagged fibonacci generator as seen in The Art of Computer Programming, Vol. 2
|
|
||||||
* This should roughly match the implementation that EQ's client uses and be compatible with our Random class
|
|
||||||
*
|
|
||||||
* EQ's rand looks like it was from an example implementation that as posted on pscode.com
|
|
||||||
*
|
|
||||||
* You might also want to consider defining BIASED_INT_DIST as well to more closely match EQ
|
|
||||||
*/
|
|
||||||
|
|
||||||
namespace EQ {
|
|
||||||
template<typename UIntType, size_t w, size_t j, size_t k>
|
|
||||||
class additive_lagged_fibonacci_engine {
|
|
||||||
static_assert(std::is_unsigned<UIntType>::value, "result_type must be an unsigned integral type");
|
|
||||||
static_assert(0u < j && j < k, "0 < j < k");
|
|
||||||
static_assert(0u < w && w <= std::numeric_limits<UIntType>::digits,
|
|
||||||
"template argument substituting w out of bounds");
|
|
||||||
|
|
||||||
public:
|
|
||||||
using result_type = UIntType;
|
|
||||||
static constexpr size_t word_size = w;
|
|
||||||
static constexpr size_t short_lag = j;
|
|
||||||
static constexpr size_t long_lag = k;
|
|
||||||
static constexpr result_type default_seed = 19780503u; // default for subtract_with_carry_engine
|
|
||||||
|
|
||||||
additive_lagged_fibonacci_engine() : additive_lagged_fibonacci_engine(default_seed) {}
|
|
||||||
|
|
||||||
explicit additive_lagged_fibonacci_engine(result_type sd) { seed(sd); }
|
|
||||||
|
|
||||||
void seed(result_type seed = default_seed)
|
|
||||||
{
|
|
||||||
state1 = long_lag - long_lag;
|
|
||||||
state2 = long_lag - short_lag;
|
|
||||||
state[0] = static_cast<int>(seed) & ((1u << word_size) - 1);
|
|
||||||
state[1] = 1;
|
|
||||||
for (int i = 2; i < long_lag; ++i)
|
|
||||||
state[i] = (state[i - 1] + state[i - 2]) & ((1u << word_size) - 1);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// TODO: seed via seed_seq
|
|
||||||
|
|
||||||
static constexpr result_type min() { return 0; }
|
|
||||||
static constexpr result_type max() { return ((1u << word_size) - 1) >> 6; }
|
|
||||||
|
|
||||||
void discard(unsigned long long z) {
|
|
||||||
for (; z != 0ULL; --z)
|
|
||||||
(*this)();
|
|
||||||
}
|
|
||||||
|
|
||||||
result_type operator()() {
|
|
||||||
result_type rand = (state[state1] + state[state2]) & ((1u << word_size) - 1);
|
|
||||||
state[state1] = rand;
|
|
||||||
if (++state1 == long_lag)
|
|
||||||
state1 = 0;
|
|
||||||
if (++state2 == long_lag)
|
|
||||||
state2 = 0;
|
|
||||||
|
|
||||||
return rand >> 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
result_type state1;
|
|
||||||
result_type state2;
|
|
||||||
result_type state[long_lag];
|
|
||||||
|
|
||||||
public:
|
|
||||||
template<typename UInt, size_t W, size_t J, size_t K>
|
|
||||||
friend bool operator==(const additive_lagged_fibonacci_engine<UInt, W, J, K> &x,
|
|
||||||
const additive_lagged_fibonacci_engine<UInt, W, J, K> &y)
|
|
||||||
{
|
|
||||||
return std::equal(x.state, x.state + long_lag, y.state) && x.state1 == y.state1 &&
|
|
||||||
x.state2 == y.state2;
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename UInt, size_t W, size_t J, size_t K>
|
|
||||||
friend bool operator!=(const additive_lagged_fibonacci_engine<UInt, W, J, K> &x,
|
|
||||||
const additive_lagged_fibonacci_engine<UInt, W, J, K> &y)
|
|
||||||
{ return !(x == y); }
|
|
||||||
|
|
||||||
template<typename UInt, size_t W, size_t J, size_t K, typename CharT, typename Traits>
|
|
||||||
friend std::basic_ostream<CharT, Traits> &
|
|
||||||
operator<<(std::basic_istream<CharT, Traits> &os, additive_lagged_fibonacci_engine<UInt, W, J, K> &x)
|
|
||||||
{
|
|
||||||
using ios_base = typename std::basic_istream<CharT, Traits>::ios_base;
|
|
||||||
|
|
||||||
const typename ios_base::fmtflags flags = os.flags();
|
|
||||||
const CharT fill = os.fill();
|
|
||||||
const CharT space = os.widen(' ');
|
|
||||||
os.flags(ios_base::dec | ios_base::fixed | ios_base::left);
|
|
||||||
os.fill(space);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < long_lag; ++i)
|
|
||||||
os << x.state[i] << space;
|
|
||||||
os << x.state1 << space << x.state2;
|
|
||||||
|
|
||||||
os.flags(flags);
|
|
||||||
os.fill(fill);
|
|
||||||
return os;
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename UInt, size_t W, size_t J, size_t K, typename CharT, typename Traits>
|
|
||||||
friend std::basic_istream<CharT, Traits> &
|
|
||||||
operator>>(std::basic_istream<CharT, Traits> &is, additive_lagged_fibonacci_engine<UInt, W, J, K> &x)
|
|
||||||
{
|
|
||||||
using ios_base = typename std::basic_istream<CharT, Traits>::ios_base;
|
|
||||||
|
|
||||||
const typename ios_base::fmtflags flags = is.flags();
|
|
||||||
is.flags(ios_base::dec | ios_base::skipws);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < long_lag; ++i)
|
|
||||||
is >> x.state[i];
|
|
||||||
is >> x.state1;
|
|
||||||
is >> x.state2;
|
|
||||||
|
|
||||||
is.flags(flags);
|
|
||||||
return is;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using EQRand = additive_lagged_fibonacci_engine<uint32_t, 30, 24, 55>;
|
|
||||||
};
|
|
||||||
|
|
||||||
+190
@@ -0,0 +1,190 @@
|
|||||||
|
/*
|
||||||
|
* Boost Software License - Version 1.0 - August 17th, 2003
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person or organization
|
||||||
|
* obtaining a copy of the software and accompanying documentation covered by
|
||||||
|
* this license (the "Software") to use, reproduce, display, distribute,
|
||||||
|
* execute, and transmit the Software, and to prepare derivative works of the
|
||||||
|
* Software, and to permit third-parties to whom the Software is furnished to
|
||||||
|
* do so, all subject to the following:
|
||||||
|
*
|
||||||
|
* The copyright notices in the Software and this entire statement, including
|
||||||
|
* the above license grant, this restriction and the following disclaimer,
|
||||||
|
* must be included in all copies of the Software, in whole or in part, and
|
||||||
|
* all derivative works of the Software, unless such copies or derivative
|
||||||
|
* works are solely in the form of machine-executable object code generated by
|
||||||
|
* a source language processor.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||||
|
* SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||||
|
* FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||||
|
* DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// EQ::Any is a modified version of Boost::Any and as such retains the Boost licensing.
|
||||||
|
|
||||||
|
#ifndef EQEMU_COMMON_ANY_H
|
||||||
|
#define EQEMU_COMMON_ANY_H
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <typeinfo>
|
||||||
|
|
||||||
|
namespace EQ
|
||||||
|
{
|
||||||
|
class Any
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Any()
|
||||||
|
: content(nullptr)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
Any(const ValueType &value)
|
||||||
|
: content(new Holder<ValueType>(value))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Any(const Any &other)
|
||||||
|
: content(other.content ? other.content->clone() : 0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
~Any()
|
||||||
|
{
|
||||||
|
if(content)
|
||||||
|
delete content;
|
||||||
|
}
|
||||||
|
|
||||||
|
Any& swap(Any &rhs)
|
||||||
|
{
|
||||||
|
std::swap(content, rhs.content);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
Any& operator=(const ValueType &rhs)
|
||||||
|
{
|
||||||
|
Any(rhs).swap(*this);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Any& operator=(Any rhs)
|
||||||
|
{
|
||||||
|
rhs.swap(*this);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool empty() const
|
||||||
|
{
|
||||||
|
return !content;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::type_info& type() const
|
||||||
|
{
|
||||||
|
return content ? content->type() : typeid(void);
|
||||||
|
}
|
||||||
|
|
||||||
|
class Placeholder
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual ~Placeholder()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual const std::type_info& type() const = 0;
|
||||||
|
virtual Placeholder* clone() const = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
class Holder : public Placeholder
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Holder(const ValueType &value)
|
||||||
|
: held(value)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual const std::type_info& type() const
|
||||||
|
{
|
||||||
|
return typeid(ValueType);
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual Placeholder* clone() const
|
||||||
|
{
|
||||||
|
return new Holder(held);
|
||||||
|
}
|
||||||
|
|
||||||
|
ValueType held;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Holder& operator=(const Holder&);
|
||||||
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
|
template<typename ValueType>
|
||||||
|
friend ValueType* any_cast(Any*);
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
friend ValueType* unsafe_any_cast(Any*);
|
||||||
|
|
||||||
|
Placeholder* content;
|
||||||
|
};
|
||||||
|
|
||||||
|
class bad_any_cast : public std::bad_cast
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual const char * what() const throw()
|
||||||
|
{
|
||||||
|
return "DBI::bad_any_cast: failed conversion using DBI::any_cast";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
ValueType* any_cast(Any* operand)
|
||||||
|
{
|
||||||
|
return operand &&
|
||||||
|
operand->type() == typeid(ValueType) ? &static_cast<Any::Holder<ValueType>*>(operand->content)->held : nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
inline const ValueType* any_cast(const Any* operand)
|
||||||
|
{
|
||||||
|
return any_cast<ValueType>(const_cast<Any*>(operand));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
ValueType any_cast(Any& operand)
|
||||||
|
{
|
||||||
|
typedef typename std::remove_reference<ValueType>::type nonref;
|
||||||
|
nonref* result = any_cast<nonref>(&operand);
|
||||||
|
if(!result)
|
||||||
|
throw bad_any_cast();
|
||||||
|
return *result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
inline ValueType any_cast(const Any& operand)
|
||||||
|
{
|
||||||
|
typedef typename std::remove_reference<ValueType>::type nonref;
|
||||||
|
return any_cast<const nonref&>(const_cast<Any&>(operand));
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
inline ValueType* unsafe_any_cast(Any* operand)
|
||||||
|
{
|
||||||
|
return &static_cast<Any::Holder<ValueType>*>(operand->content)->held;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ValueType>
|
||||||
|
inline const ValueType* unsafe_any_cast(const Any* operand)
|
||||||
|
{
|
||||||
|
return unsafe_any_cast<ValueType>(const_cast<Any*>(operand));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
/* EQEMu: Everquest Server Emulator
|
||||||
|
Copyright (C) 2001-2013 EQEMu Development Team (http://eqemulator.net)
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; version 2 of the License.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY except by those people which sell it, which
|
||||||
|
are required to give you total support for your newly bought product;
|
||||||
|
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||||||
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __EQEMU_COMMON_BASE_DATA_H
|
||||||
|
#define __EQEMU_COMMON_BASE_DATA_H
|
||||||
|
|
||||||
|
struct BaseDataStruct
|
||||||
|
{
|
||||||
|
double base_hp;
|
||||||
|
double base_mana;
|
||||||
|
double base_end;
|
||||||
|
double hp_regen;
|
||||||
|
double end_regen;
|
||||||
|
double hp_factor;
|
||||||
|
double mana_factor;
|
||||||
|
double endurance_factor;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -23,18 +23,18 @@
|
|||||||
|
|
||||||
BasePacket::BasePacket(const unsigned char *buf, uint32 len)
|
BasePacket::BasePacket(const unsigned char *buf, uint32 len)
|
||||||
{
|
{
|
||||||
pBuffer=nullptr;
|
this->pBuffer=nullptr;
|
||||||
size=0;
|
this->size=0;
|
||||||
_wpos = 0;
|
this->_wpos = 0;
|
||||||
_rpos = 0;
|
this->_rpos = 0;
|
||||||
timestamp.tv_sec = 0;
|
this->timestamp.tv_sec = 0;
|
||||||
if (len>0) {
|
if (len>0) {
|
||||||
size=len;
|
this->size=len;
|
||||||
pBuffer= new unsigned char[len];
|
pBuffer= new unsigned char[len];
|
||||||
if (buf) {
|
if (buf) {
|
||||||
memcpy(pBuffer,buf,len);
|
memcpy(this->pBuffer,buf,len);
|
||||||
} else {
|
} else {
|
||||||
memset(pBuffer,0,len);
|
memset(this->pBuffer,0,len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,330 +0,0 @@
|
|||||||
#include "bazaar.h"
|
|
||||||
|
|
||||||
#include "../../common/item_instance.h"
|
|
||||||
#include "repositories/trader_repository.h"
|
|
||||||
#include <memory>
|
|
||||||
|
|
||||||
std::vector<BazaarSearchResultsFromDB_Struct>
|
|
||||||
Bazaar::GetSearchResults(
|
|
||||||
Database &db,
|
|
||||||
Database &content_db,
|
|
||||||
BazaarSearchCriteria_Struct search,
|
|
||||||
uint32 char_zone_id,
|
|
||||||
int32 char_zone_instance_id
|
|
||||||
)
|
|
||||||
{
|
|
||||||
LogTrading(
|
|
||||||
"Searching for items with search criteria - item_name [{}] min_cost [{}] max_cost [{}] min_level [{}] "
|
|
||||||
"max_level [{}] max_results [{}] prestige [{}] augment [{}] trader_entity_id [{}] trader_id [{}] "
|
|
||||||
"search_scope [{}] char_zone_id [{}], char_zone_instance_id [{}]",
|
|
||||||
search.item_name,
|
|
||||||
search.min_cost,
|
|
||||||
search.max_cost,
|
|
||||||
search.min_level,
|
|
||||||
search.max_level,
|
|
||||||
search.max_results,
|
|
||||||
search.prestige,
|
|
||||||
search.augment,
|
|
||||||
search.trader_entity_id,
|
|
||||||
search.trader_id,
|
|
||||||
search.search_scope,
|
|
||||||
char_zone_id,
|
|
||||||
char_zone_instance_id
|
|
||||||
);
|
|
||||||
|
|
||||||
static std::map<uint8, uint32> item_slot_searches_new = {
|
|
||||||
{EQ::invslot::slotCharm, 1},
|
|
||||||
{EQ::invslot::slotEar1, 2},
|
|
||||||
{EQ::invslot::slotHead, 4},
|
|
||||||
{EQ::invslot::slotFace, 8},
|
|
||||||
{EQ::invslot::slotEar2, 16},
|
|
||||||
{EQ::invslot::slotNeck, 32},
|
|
||||||
{EQ::invslot::slotShoulders, 64},
|
|
||||||
{EQ::invslot::slotArms, 128},
|
|
||||||
{EQ::invslot::slotBack, 256},
|
|
||||||
{EQ::invslot::slotWrist1, 512},
|
|
||||||
{EQ::invslot::slotWrist2, 1024},
|
|
||||||
{EQ::invslot::slotRange, 2048},
|
|
||||||
{EQ::invslot::slotHands, 4096},
|
|
||||||
{EQ::invslot::slotPrimary, 8192},
|
|
||||||
{EQ::invslot::slotSecondary, 16384},
|
|
||||||
{EQ::invslot::slotFinger1, 32768},
|
|
||||||
{EQ::invslot::slotFinger2, 65536},
|
|
||||||
{EQ::invslot::slotChest, 131072},
|
|
||||||
{EQ::invslot::slotLegs, 262144},
|
|
||||||
{EQ::invslot::slotFeet, 524288},
|
|
||||||
{EQ::invslot::slotWaist, 1048576},
|
|
||||||
{EQ::invslot::slotPowerSource, 2097152},
|
|
||||||
{EQ::invslot::slotAmmo, 4194304},
|
|
||||||
};
|
|
||||||
|
|
||||||
struct ItemSearchType {
|
|
||||||
EQ::item::ItemType type;
|
|
||||||
std::string condition;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::vector<ItemSearchType> item_search_types_new = {
|
|
||||||
{EQ::item::ItemType::ItemTypeBook, " AND (items.itemclass = 2 or items.itemclass = 31)"},
|
|
||||||
{EQ::item::ItemType::ItemTypeContainer, " AND (items.itemclass = 1 or items.itemclass = 67)"},
|
|
||||||
{EQ::item::ItemType::ItemTypeAllEffects, " AND (items.scrolleffect > 0 && items.scrolleffect < 65000)"},
|
|
||||||
{EQ::item::ItemType::ItemTypeUnknown9, " AND items.worneffect = 998"},
|
|
||||||
{EQ::item::ItemType::ItemTypeUnknown10, " AND (items.worneffect >= 1298 && items.worneffect <= 1307)"},
|
|
||||||
{EQ::item::ItemType::ItemTypeFocusEffect, " AND items.focuseffect > 0"},
|
|
||||||
{EQ::item::ItemType::ItemTypeArmor, " AND items.itemtype = 10"},
|
|
||||||
{EQ::item::ItemType::ItemType1HBlunt, " AND items.itemtype = 3"},
|
|
||||||
{EQ::item::ItemType::ItemType1HPiercing, " AND items.itemtype = 2"},
|
|
||||||
{EQ::item::ItemType::ItemType1HSlash, " AND items.itemtype = 0"},
|
|
||||||
{EQ::item::ItemType::ItemType2HBlunt, " AND items.itemtype = 4"},
|
|
||||||
{EQ::item::ItemType::ItemType2HSlash, " AND items.itemtype = 1"},
|
|
||||||
{EQ::item::ItemType::ItemTypeBow, " AND items.itemtype = 5"},
|
|
||||||
{EQ::item::ItemType::ItemTypeShield, " AND items.itemtype = 8"},
|
|
||||||
{EQ::item::ItemType::ItemTypeMisc, " AND items.itemtype = 11"},
|
|
||||||
{EQ::item::ItemType::ItemTypeFood, " AND items.itemtype = 14"},
|
|
||||||
{EQ::item::ItemType::ItemTypeDrink, " AND items.itemtype = 15"},
|
|
||||||
{EQ::item::ItemType::ItemTypeLight, " AND items.itemtype = 16"},
|
|
||||||
{EQ::item::ItemType::ItemTypeCombinable, " AND items.itemtype = 17"},
|
|
||||||
{EQ::item::ItemType::ItemTypeBandage, " AND items.itemtype = 18"},
|
|
||||||
{EQ::item::ItemType::ItemTypeSmallThrowing, " AND (items.itemtype = 19 OR items.itemtype = 7)"},
|
|
||||||
{EQ::item::ItemType::ItemTypeSpell, " AND items.itemtype = 20"},
|
|
||||||
{EQ::item::ItemType::ItemTypePotion, " AND items.itemtype = 21"},
|
|
||||||
{EQ::item::ItemType::ItemTypeBrassInstrument, " AND items.itemtype = 25"},
|
|
||||||
{EQ::item::ItemType::ItemTypeWindInstrument, " AND items.itemtype = 23"},
|
|
||||||
{EQ::item::ItemType::ItemTypeStringedInstrument, " AND items.itemtype = 24"},
|
|
||||||
{EQ::item::ItemType::ItemTypePercussionInstrument, " AND items.itemtype = 26"},
|
|
||||||
{EQ::item::ItemType::ItemTypeArrow, " AND items.itemtype = 27"},
|
|
||||||
{EQ::item::ItemType::ItemTypeJewelry, " AND items.itemtype = 29"},
|
|
||||||
{EQ::item::ItemType::ItemTypeNote, " AND items.itemtype = 32"},
|
|
||||||
{EQ::item::ItemType::ItemTypeKey, " AND items.itemtype = 33"},
|
|
||||||
{EQ::item::ItemType::ItemType2HPiercing, " AND items.itemtype = 35"},
|
|
||||||
{EQ::item::ItemType::ItemTypeAlcohol, " AND items.itemtype = 38"},
|
|
||||||
{EQ::item::ItemType::ItemTypeMartial, " AND items.itemtype = 45"},
|
|
||||||
{EQ::item::ItemType::ItemTypeAugmentation, " AND items.itemtype = 54"},
|
|
||||||
{EQ::item::ItemType::ItemTypeAlternateAbility, " AND items.itemtype = 57"},
|
|
||||||
{EQ::item::ItemType::ItemTypeCount, " AND items.itemtype = 65"},
|
|
||||||
{EQ::item::ItemType::ItemTypeCollectible, " AND items.itemtype = 66"}
|
|
||||||
};
|
|
||||||
|
|
||||||
// item stat searches
|
|
||||||
struct ItemStatSearch {
|
|
||||||
std::string query_string;
|
|
||||||
EQ::skills::SkillType skill_type;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::map<uint32, ItemStatSearch> item_stat_searches_new = {
|
|
||||||
{STAT_AC, {" items.ac" , static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_AGI, {" items.aagi", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_CHA, {" items.acha", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_DEX, {" items.adex", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_INT, {" items.aint", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_STA, {" items.asta", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_STR, {" items.astr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_WIS, {" items.awis", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_COLD, {" items.cr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_DISEASE, {" items.dr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_FIRE, {" items.fr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_MAGIC, {" items.mr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_POISON, {" items.pr", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HP, {" items.hp", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_MANA, {" items.mana", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_ENDURANCE, {" items.endur", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_ATTACK, {" items.attack", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HP_REGEN, {" items.regen", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_MANA_REGEN, {" items.manaregen", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HASTE, {" items.haste", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_DAMAGE_SHIELD, {" items.damageshield", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_DS_MITIGATION, {" items.dsmitigation", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEAL_AMOUNT, {" items.healamt", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_SPELL_DAMAGE, {" items.spelldmg", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_CLAIRVOYANCE, {" items.clairvoyance", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_AGILITY, {" items.heroic_agi", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_CHARISMA, {" items.heroic_cha", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_DEXTERITY, {" items.heroic_dex", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_INTELLIGENCE, {" items.heroic_int", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_STAMINA, {" items.heroic_sta", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_STRENGTH, {" items.heroic_str", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_HEROIC_WISDOM, {" items.heroic_wis", static_cast<EQ::skills::SkillType>(0)} },
|
|
||||||
{STAT_BASH, {" items.skillmodvalue", EQ::skills::SkillBash} },
|
|
||||||
{STAT_BACKSTAB, {" items.backstabdmg", EQ::skills::SkillBackstab} },
|
|
||||||
{STAT_DRAGON_PUNCH, {" items.skillmodvalue", EQ::skills::SkillDragonPunch} },
|
|
||||||
{STAT_EAGLE_STRIKE, {" items.skillmodvalue", EQ::skills::SkillEagleStrike} },
|
|
||||||
{STAT_FLYING_KICK, {" items.skillmodvalue", EQ::skills::SkillFlyingKick} },
|
|
||||||
{STAT_KICK, {" items.skillmodvalue", EQ::skills::SkillKick} },
|
|
||||||
{STAT_ROUND_KICK, {" items.skillmodvalue", EQ::skills::SkillRoundKick} },
|
|
||||||
{STAT_TIGER_CLAW, {" items.skillmodvalue", EQ::skills::SkillTigerClaw} },
|
|
||||||
{STAT_FRENZY, {" items.skillmodvalue", EQ::skills::SkillFrenzy} },
|
|
||||||
};
|
|
||||||
|
|
||||||
bool convert = false;
|
|
||||||
std::string search_criteria_trader("TRUE");
|
|
||||||
std::string field_criteria_items("FALSE");
|
|
||||||
std::string where_criteria_items(" TRUE ");
|
|
||||||
|
|
||||||
if (search.search_scope == NonRoFBazaarSearchScope) {
|
|
||||||
search_criteria_trader.append(
|
|
||||||
fmt::format(
|
|
||||||
" AND trader.char_entity_id = {} AND trader.char_zone_id = {} AND trader.char_zone_instance_id = {}",
|
|
||||||
search.trader_entity_id,
|
|
||||||
Zones::BAZAAR,
|
|
||||||
char_zone_instance_id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else if (search.search_scope == Local_Scope) {
|
|
||||||
search_criteria_trader.append(fmt::format(
|
|
||||||
" AND trader.char_zone_id = {} AND trader.char_zone_instance_id = {}",
|
|
||||||
char_zone_id,
|
|
||||||
char_zone_instance_id)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else if (search.trader_id > 0) {
|
|
||||||
if (RuleB(Bazaar, UseAlternateBazaarSearch)) {
|
|
||||||
if (search.trader_id >= TraderRepository::TRADER_CONVERT_ID) {
|
|
||||||
convert = true;
|
|
||||||
search_criteria_trader.append(fmt::format(
|
|
||||||
" AND trader.char_zone_id = {} AND trader.char_zone_instance_id = {}",
|
|
||||||
Zones::BAZAAR,
|
|
||||||
search.trader_id - TraderRepository::TRADER_CONVERT_ID)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
search_criteria_trader.append(fmt::format(" AND trader.char_id = {}", search.trader_id));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
search_criteria_trader.append(fmt::format(" AND trader.char_id = {}", search.trader_id));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.min_cost != 0) {
|
|
||||||
search_criteria_trader.append(fmt::format(" AND trader.item_cost >= {}", search.min_cost * 1000));
|
|
||||||
}
|
|
||||||
if (search.max_cost != 0) {
|
|
||||||
search_criteria_trader.append(fmt::format(" AND trader.item_cost <= {}", (uint64) search.max_cost * 1000));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.slot != std::numeric_limits<uint32>::max()) {
|
|
||||||
if (item_slot_searches_new.contains(search.slot)) {
|
|
||||||
where_criteria_items.append(
|
|
||||||
fmt::format(" AND items.slots & {0} = {0}", item_slot_searches_new[search.slot]));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.type != std::numeric_limits<uint32>::max()) {
|
|
||||||
for (auto const &[type, condition]: item_search_types_new) {
|
|
||||||
if (type == search.type) {
|
|
||||||
where_criteria_items.append(condition);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.race != std::numeric_limits<uint32>::max()) {
|
|
||||||
where_criteria_items.append(
|
|
||||||
fmt::format(" AND items.races & {0} = {0}", GetPlayerRaceBit(GetRaceIDFromPlayerRaceValue(search.race))));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search._class != std::numeric_limits<uint32>::max()) {
|
|
||||||
where_criteria_items.append(fmt::format(" AND items.classes & {0} = {0}", GetPlayerClassBit(search._class)));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.item_stat != std::numeric_limits<uint32>::max()) {
|
|
||||||
if (item_stat_searches_new.contains(search.item_stat)) {
|
|
||||||
field_criteria_items = fmt::format("{}", item_stat_searches_new[search.item_stat].query_string);
|
|
||||||
if (item_stat_searches_new[search.item_stat].skill_type) {
|
|
||||||
where_criteria_items.append(
|
|
||||||
fmt::format(" AND items.skillmodtype = {} ", item_stat_searches_new[search.item_stat].skill_type));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
where_criteria_items.append(
|
|
||||||
fmt::format(" AND {} > 0 ", item_stat_searches_new[search.item_stat].query_string));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.augment) {
|
|
||||||
where_criteria_items.append(fmt::format(
|
|
||||||
" AND (items.augslot1type = {0} OR "
|
|
||||||
"items.augslot2type = {0} OR "
|
|
||||||
"items.augslot3type = {0} OR "
|
|
||||||
"items.augslot4type = {0} OR "
|
|
||||||
"items.augslot5type = {0} OR "
|
|
||||||
"items.augslot6type = {0})",
|
|
||||||
search.augment)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.min_level != 1) {
|
|
||||||
where_criteria_items.append(fmt::format(" AND items.reclevel >= {}", search.min_level));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (search.max_level != 100) {
|
|
||||||
where_criteria_items.append(fmt::format(" AND items.reclevel <= {}", search.max_level));
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<BazaarSearchResultsFromDB_Struct> all_entries;
|
|
||||||
std::vector<std::string> trader_items_ids{};
|
|
||||||
|
|
||||||
auto const trader_results = TraderRepository::GetBazaarTraderDetails(db, search_criteria_trader);
|
|
||||||
if (trader_results.empty()) {
|
|
||||||
LogTradingDetail("Bazaar - No traders found in bazaar search.");
|
|
||||||
return all_entries;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto const &i: trader_results) {
|
|
||||||
trader_items_ids.push_back(std::to_string(i.trader.item_id));
|
|
||||||
}
|
|
||||||
|
|
||||||
auto const item_results = ItemsRepository::GetItemsForBazaarSearch(
|
|
||||||
content_db,
|
|
||||||
trader_items_ids,
|
|
||||||
std::string(search.item_name),
|
|
||||||
field_criteria_items,
|
|
||||||
where_criteria_items
|
|
||||||
);
|
|
||||||
|
|
||||||
if (item_results.empty()) {
|
|
||||||
LogError("Bazaar - No items found in bazaar search.");
|
|
||||||
return all_entries;
|
|
||||||
}
|
|
||||||
|
|
||||||
all_entries.reserve(trader_results.size());
|
|
||||||
|
|
||||||
for (auto const& t:trader_results) {
|
|
||||||
if (!item_results.contains(t.trader.item_id)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
BazaarSearchResultsFromDB_Struct r{};
|
|
||||||
r.count = 1;
|
|
||||||
r.trader_id = t.trader.char_id;
|
|
||||||
r.serial_number = t.trader.item_sn;
|
|
||||||
r.cost = t.trader.item_cost;
|
|
||||||
r.slot_id = t.trader.slot_id;
|
|
||||||
r.sum_charges = t.trader.item_charges;
|
|
||||||
r.stackable = item_results.at(t.trader.item_id).stackable;
|
|
||||||
r.icon_id = item_results.at(t.trader.item_id).icon;
|
|
||||||
r.trader_zone_id = t.trader.char_zone_id;
|
|
||||||
r.trader_zone_instance_id = t.trader.char_zone_instance_id;
|
|
||||||
r.trader_entity_id = t.trader.char_entity_id;
|
|
||||||
r.serial_number_RoF = fmt::format("{:016}\0", t.trader.item_sn);
|
|
||||||
r.item_name = fmt::format("{:.63}\0", item_results.at(t.trader.item_id).name);
|
|
||||||
r.trader_name = fmt::format("{:.63}\0", t.trader_name);
|
|
||||||
r.item_stat = item_results.at(t.trader.item_id).stats;
|
|
||||||
|
|
||||||
if (RuleB(Bazaar, UseAlternateBazaarSearch)) {
|
|
||||||
if (convert || (r.trader_zone_id == Zones::BAZAAR && r.trader_zone_instance_id != char_zone_instance_id)) {
|
|
||||||
r.trader_id = TraderRepository::TRADER_CONVERT_ID + r.trader_zone_instance_id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
all_entries.push_back(r);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (all_entries.size() > search.max_results) {
|
|
||||||
all_entries.resize(search.max_results);
|
|
||||||
}
|
|
||||||
|
|
||||||
LogTrading("Returning [{}] items from search results", all_entries.size());
|
|
||||||
|
|
||||||
return all_entries;
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
#ifndef EQEMU_BAZAAR_H
|
|
||||||
#define EQEMU_BAZAAR_H
|
|
||||||
|
|
||||||
#include <vector>
|
|
||||||
#include "shareddb.h"
|
|
||||||
#include "../../common/item_instance.h"
|
|
||||||
|
|
||||||
class Bazaar {
|
|
||||||
public:
|
|
||||||
static std::vector<BazaarSearchResultsFromDB_Struct>
|
|
||||||
GetSearchResults(Database &content_db, Database &db, BazaarSearchCriteria_Struct search, unsigned int char_zone_id, int char_zone_instance_id);
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#endif //EQEMU_BAZAAR_H
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
#include "../common/global_define.h"
|
|
||||||
#include "../common/bodytypes.h"
|
|
||||||
|
|
||||||
std::string BodyType::GetName(uint8 body_type_id)
|
|
||||||
{
|
|
||||||
return IsValid(body_type_id) ? body_type_names[body_type_id] : "UNKNOWN BODY TYPE";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool BodyType::IsValid(uint8 body_type_id)
|
|
||||||
{
|
|
||||||
return body_type_names.find(body_type_id) != body_type_names.end();
|
|
||||||
}
|
|
||||||
+43
-91
@@ -18,96 +18,48 @@
|
|||||||
#ifndef BODYTYPES_H
|
#ifndef BODYTYPES_H
|
||||||
#define BODYTYPES_H
|
#define BODYTYPES_H
|
||||||
|
|
||||||
#include "types.h"
|
typedef enum {
|
||||||
#include <map>
|
BT_Humanoid = 1,
|
||||||
#include <string>
|
BT_Lycanthrope = 2,
|
||||||
|
BT_Undead = 3,
|
||||||
// body types above 64 make the mob invisible
|
BT_Giant = 4,
|
||||||
namespace BodyType {
|
BT_Construct = 5,
|
||||||
constexpr uint8 Humanoid = 1;
|
BT_Extraplanar = 6,
|
||||||
constexpr uint8 Lycanthrope = 2;
|
BT_Magical = 7, //this name might be a bit off,
|
||||||
constexpr uint8 Undead = 3;
|
BT_SummonedUndead = 8,
|
||||||
constexpr uint8 Giant = 4;
|
BT_RaidGiant = 9,
|
||||||
constexpr uint8 Construct = 5;
|
// ...
|
||||||
constexpr uint8 Extraplanar = 6;
|
BT_NoTarget = 11, //no name, can't target this bodytype
|
||||||
constexpr uint8 Magical = 7; // this name might be a bit off,
|
BT_Vampire = 12,
|
||||||
constexpr uint8 SummonedUndead = 8;
|
BT_Atenha_Ra = 13,
|
||||||
constexpr uint8 RaidGiant = 9; // Velious era Raid Giant
|
BT_Greater_Akheva = 14,
|
||||||
constexpr uint8 RaidColdain = 10; // Velious era Raid Coldain
|
BT_Khati_Sha = 15,
|
||||||
constexpr uint8 NoTarget = 11; // no name, can't target this bodytype
|
BT_Seru = 16, //not confirmed....
|
||||||
constexpr uint8 Vampire = 12;
|
BT_Draz_Nurakk = 18,
|
||||||
constexpr uint8 AtenHaRa = 13;
|
BT_Zek = 19,
|
||||||
constexpr uint8 GreaterAkheva = 14;
|
BT_Luggald = 20,
|
||||||
constexpr uint8 KhatiSha = 15;
|
BT_Animal = 21,
|
||||||
constexpr uint8 Seru = 16;
|
BT_Insect = 22,
|
||||||
constexpr uint8 GriegVeneficus = 17;
|
BT_Monster = 23,
|
||||||
constexpr uint8 DrazNurakk = 18;
|
BT_Summoned = 24, //Elemental?
|
||||||
constexpr uint8 Zek = 19; //"creatures from the Plane of War."
|
BT_Plant = 25,
|
||||||
constexpr uint8 Luggald = 20;
|
BT_Dragon = 26,
|
||||||
constexpr uint8 Animal = 21;
|
BT_Summoned2 = 27,
|
||||||
constexpr uint8 Insect = 22;
|
BT_Summoned3 = 28,
|
||||||
constexpr uint8 Monster = 23;
|
//29
|
||||||
constexpr uint8 Summoned = 24; // Elemental?
|
BT_VeliousDragon = 30, //might not be a tight set
|
||||||
constexpr uint8 Plant = 25;
|
// ...
|
||||||
constexpr uint8 Dragon = 26;
|
BT_Dragon3 = 32,
|
||||||
constexpr uint8 Summoned2 = 27;
|
BT_Boxes = 33,
|
||||||
constexpr uint8 Summoned3 = 28;
|
BT_Muramite = 34, //tribal dudes
|
||||||
constexpr uint8 Dragon2 = 29; // database data indicates this is a dragon type (Kunark and DoN?)
|
// ...
|
||||||
constexpr uint8 VeliousDragon = 30; // might not be a tight set
|
BT_NoTarget2 = 60,
|
||||||
constexpr uint8 Familiar = 31;
|
// ...
|
||||||
constexpr uint8 Dragon3 = 32;
|
BT_SwarmPet = 63, //is this valid, or made up?
|
||||||
constexpr uint8 Boxes = 33;
|
// ...
|
||||||
constexpr uint8 Muramite = 34; // tribal dudes
|
BT_InvisMan = 66, //no name, seen on 'InvisMan', can be /targeted
|
||||||
constexpr uint8 NoTarget2 = 60;
|
BT_Special = 67
|
||||||
constexpr uint8 SwarmPet = 63; // Looks like weapon proc related temp pets and few misc pets, should not be used for checking swarm pets in general.
|
} bodyType;
|
||||||
constexpr uint8 MonsterSummon = 64;
|
/* bodytypes above 64 make the mob not show up */
|
||||||
constexpr uint8 InvisibleMan = 66; // no name, seen on 'InvisMan', can be /targeted
|
|
||||||
constexpr uint8 Special = 67;
|
|
||||||
|
|
||||||
std::string GetName(uint8 body_type_id);
|
|
||||||
bool IsValid(uint8 body_type_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint8, std::string> body_type_names = {
|
|
||||||
{ BodyType::Humanoid, "Humanoid" },
|
|
||||||
{ BodyType::Lycanthrope, "Lycanthrope" },
|
|
||||||
{ BodyType::Undead, "Undead" },
|
|
||||||
{ BodyType::Giant, "Giant" },
|
|
||||||
{ BodyType::Construct, "Construct" },
|
|
||||||
{ BodyType::Extraplanar, "Extraplanar" },
|
|
||||||
{ BodyType::Magical, "Magical" },
|
|
||||||
{ BodyType::SummonedUndead, "Summoned Undead" },
|
|
||||||
{ BodyType::RaidGiant, "Raid Giant" },
|
|
||||||
{ BodyType::RaidColdain, "Raid Coldain" },
|
|
||||||
{ BodyType::NoTarget, "Untargetable" },
|
|
||||||
{ BodyType::Vampire, "Vampire" },
|
|
||||||
{ BodyType::AtenHaRa, "Aten Ha Ra" },
|
|
||||||
{ BodyType::GreaterAkheva, "Greater Akheva" },
|
|
||||||
{ BodyType::KhatiSha, "Khati Sha" },
|
|
||||||
{ BodyType::Seru, "Seru" },
|
|
||||||
{ BodyType::GriegVeneficus, "Grieg Veneficus" },
|
|
||||||
{ BodyType::DrazNurakk, "Draz Nurakk" },
|
|
||||||
{ BodyType::Zek, "Zek" },
|
|
||||||
{ BodyType::Luggald, "Luggald" },
|
|
||||||
{ BodyType::Animal, "Animal" },
|
|
||||||
{ BodyType::Insect, "Insect" },
|
|
||||||
{ BodyType::Monster, "Monster" },
|
|
||||||
{ BodyType::Summoned, "Summoned" },
|
|
||||||
{ BodyType::Plant, "Plant" },
|
|
||||||
{ BodyType::Dragon, "Dragon" },
|
|
||||||
{ BodyType::Summoned2, "Summoned 2" },
|
|
||||||
{ BodyType::Summoned3, "Summoned 3" },
|
|
||||||
{ BodyType::Dragon2, "Dragon 2" },
|
|
||||||
{ BodyType::VeliousDragon, "Velious Dragon" },
|
|
||||||
{ BodyType::Familiar, "Familiar" },
|
|
||||||
{ BodyType::Dragon3, "Dragon 3" },
|
|
||||||
{ BodyType::Boxes, "Boxes" },
|
|
||||||
{ BodyType::Muramite, "Muramite" },
|
|
||||||
{ BodyType::NoTarget2, "Untargetable 2" },
|
|
||||||
{ BodyType::SwarmPet, "Swarm Pet" },
|
|
||||||
{ BodyType::MonsterSummon, "Monster Summon" },
|
|
||||||
{ BodyType::InvisibleMan, "Invisible Man" },
|
|
||||||
{ BodyType::Special, "Special" },
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
+448
-349
File diff suppressed because it is too large
Load Diff
+93
-101
@@ -19,106 +19,98 @@
|
|||||||
#define CLASSES_CH
|
#define CLASSES_CH
|
||||||
|
|
||||||
#include "../common/types.h"
|
#include "../common/types.h"
|
||||||
#include "../common/rulesys.h"
|
|
||||||
#include <string>
|
|
||||||
#include <map>
|
|
||||||
|
|
||||||
namespace Class {
|
#define WARRIOR 1
|
||||||
constexpr uint8 None = 0;
|
#define CLERIC 2
|
||||||
constexpr uint8 Warrior = 1;
|
#define PALADIN 3
|
||||||
constexpr uint8 Cleric = 2;
|
#define RANGER 4
|
||||||
constexpr uint8 Paladin = 3;
|
#define SHADOWKNIGHT 5
|
||||||
constexpr uint8 Ranger = 4;
|
#define DRUID 6
|
||||||
constexpr uint8 ShadowKnight = 5;
|
#define MONK 7
|
||||||
constexpr uint8 Druid = 6;
|
#define BARD 8
|
||||||
constexpr uint8 Monk = 7;
|
#define ROGUE 9
|
||||||
constexpr uint8 Bard = 8;
|
#define SHAMAN 10
|
||||||
constexpr uint8 Rogue = 9;
|
#define NECROMANCER 11
|
||||||
constexpr uint8 Shaman = 10;
|
#define WIZARD 12
|
||||||
constexpr uint8 Necromancer = 11;
|
#define MAGICIAN 13
|
||||||
constexpr uint8 Wizard = 12;
|
#define ENCHANTER 14
|
||||||
constexpr uint8 Magician = 13;
|
#define BEASTLORD 15
|
||||||
constexpr uint8 Enchanter = 14;
|
#define BERSERKER 16
|
||||||
constexpr uint8 Beastlord = 15;
|
#define WARRIORGM 20
|
||||||
constexpr uint8 Berserker = 16;
|
#define CLERICGM 21
|
||||||
constexpr uint8 WarriorGM = 20;
|
#define PALADINGM 22
|
||||||
constexpr uint8 ClericGM = 21;
|
#define RANGERGM 23
|
||||||
constexpr uint8 PaladinGM = 22;
|
#define SHADOWKNIGHTGM 24
|
||||||
constexpr uint8 RangerGM = 23;
|
#define DRUIDGM 25
|
||||||
constexpr uint8 ShadowKnightGM = 24;
|
#define MONKGM 26
|
||||||
constexpr uint8 DruidGM = 25;
|
#define BARDGM 27
|
||||||
constexpr uint8 MonkGM = 26;
|
#define ROGUEGM 28
|
||||||
constexpr uint8 BardGM = 27;
|
#define SHAMANGM 29
|
||||||
constexpr uint8 RogueGM = 28;
|
#define NECROMANCERGM 30
|
||||||
constexpr uint8 ShamanGM = 29;
|
#define WIZARDGM 31
|
||||||
constexpr uint8 NecromancerGM = 30;
|
#define MAGICIANGM 32
|
||||||
constexpr uint8 WizardGM = 31;
|
#define ENCHANTERGM 33
|
||||||
constexpr uint8 MagicianGM = 32;
|
#define BEASTLORDGM 34
|
||||||
constexpr uint8 EnchanterGM = 33;
|
#define BERSERKERGM 35
|
||||||
constexpr uint8 BeastlordGM = 34;
|
#define BANKER 40
|
||||||
constexpr uint8 BerserkerGM = 35;
|
#define MERCHANT 41
|
||||||
constexpr uint8 Banker = 40;
|
#define DISCORD_MERCHANT 59
|
||||||
constexpr uint8 Merchant = 41;
|
#define ADVENTURERECRUITER 60
|
||||||
constexpr uint8 DiscordMerchant = 59;
|
#define ADVENTUREMERCHANT 61
|
||||||
constexpr uint8 AdventureRecruiter = 60;
|
#define LDON_TREASURE 62 // objects you can use /open on first seen in LDONs
|
||||||
constexpr uint8 AdventureMerchant = 61;
|
#define CORPSE_CLASS 62 // only seen on Danvi's Corpse in Akheva so far..
|
||||||
constexpr uint8 LDoNTreasure = 62;
|
#define TRIBUTE_MASTER 63
|
||||||
constexpr uint8 TributeMaster = 63;
|
#define GUILD_TRIBUTE_MASTER 64 // not sure
|
||||||
constexpr uint8 GuildTributeMaster = 64;
|
#define GUILD_BANKER 66
|
||||||
constexpr uint8 GuildBanker = 66;
|
#define NORRATHS_KEEPERS_MERCHANT 67
|
||||||
constexpr uint8 NorrathsKeepersMerchant = 67;
|
#define DARK_REIGN_MERCHANT 68
|
||||||
constexpr uint8 DarkReignMerchant = 68;
|
#define FELLOWSHIP_MASTER 69
|
||||||
constexpr uint8 FellowshipMaster = 69;
|
#define ALT_CURRENCY_MERCHANT 70
|
||||||
constexpr uint8 AlternateCurrencyMerchant = 70;
|
#define MERCERNARY_MASTER 71
|
||||||
constexpr uint8 MercenaryLiaison = 71;
|
|
||||||
|
|
||||||
constexpr uint8 PLAYER_CLASS_COUNT = 16;
|
|
||||||
constexpr uint16 ALL_CLASSES_BITMASK = 65535;
|
|
||||||
};
|
|
||||||
|
|
||||||
static std::map<uint8, uint16> player_class_bitmasks = {
|
// player class values
|
||||||
{Class::Warrior, 1},
|
#define PLAYER_CLASS_UNKNOWN 0
|
||||||
{Class::Cleric, 2},
|
#define PLAYER_CLASS_WARRIOR 1
|
||||||
{Class::Paladin, 4},
|
#define PLAYER_CLASS_CLERIC 2
|
||||||
{Class::Ranger, 8},
|
#define PLAYER_CLASS_PALADIN 3
|
||||||
{Class::ShadowKnight, 16},
|
#define PLAYER_CLASS_RANGER 4
|
||||||
{Class::Druid, 32},
|
#define PLAYER_CLASS_SHADOWKNIGHT 5
|
||||||
{Class::Monk, 64},
|
#define PLAYER_CLASS_DRUID 6
|
||||||
{Class::Bard, 128},
|
#define PLAYER_CLASS_MONK 7
|
||||||
{Class::Rogue, 256},
|
#define PLAYER_CLASS_BARD 8
|
||||||
{Class::Shaman, 512},
|
#define PLAYER_CLASS_ROGUE 9
|
||||||
{Class::Necromancer, 1024},
|
#define PLAYER_CLASS_SHAMAN 10
|
||||||
{Class::Wizard, 2048},
|
#define PLAYER_CLASS_NECROMANCER 11
|
||||||
{Class::Magician, 4096},
|
#define PLAYER_CLASS_WIZARD 12
|
||||||
{Class::Enchanter, 8192},
|
#define PLAYER_CLASS_MAGICIAN 13
|
||||||
{Class::Beastlord, 16384},
|
#define PLAYER_CLASS_ENCHANTER 14
|
||||||
{Class::Berserker, 32768},
|
#define PLAYER_CLASS_BEASTLORD 15
|
||||||
};
|
#define PLAYER_CLASS_BERSERKER 16
|
||||||
|
|
||||||
static std::string shadow_knight_class_name = (
|
#define PLAYER_CLASS_COUNT 16
|
||||||
RuleB(World, UseOldShadowKnightClassExport) ?
|
|
||||||
"Shadowknight" :
|
|
||||||
"Shadow Knight"
|
|
||||||
);
|
|
||||||
|
|
||||||
static std::map<uint8, std::string> class_names = {
|
|
||||||
{Class::Warrior, "Warrior"},
|
// player class bits
|
||||||
{Class::Cleric, "Cleric"},
|
#define PLAYER_CLASS_UNKNOWN_BIT 0
|
||||||
{Class::Paladin, "Paladin"},
|
#define PLAYER_CLASS_WARRIOR_BIT 1
|
||||||
{Class::Ranger, "Ranger"},
|
#define PLAYER_CLASS_CLERIC_BIT 2
|
||||||
{Class::ShadowKnight, shadow_knight_class_name},
|
#define PLAYER_CLASS_PALADIN_BIT 4
|
||||||
{Class::Druid, "Druid"},
|
#define PLAYER_CLASS_RANGER_BIT 8
|
||||||
{Class::Monk, "Monk"},
|
#define PLAYER_CLASS_SHADOWKNIGHT_BIT 16
|
||||||
{Class::Bard, "Bard"},
|
#define PLAYER_CLASS_DRUID_BIT 32
|
||||||
{Class::Rogue, "Rogue"},
|
#define PLAYER_CLASS_MONK_BIT 64
|
||||||
{Class::Shaman, "Shaman"},
|
#define PLAYER_CLASS_BARD_BIT 128
|
||||||
{Class::Necromancer, "Necromancer"},
|
#define PLAYER_CLASS_ROGUE_BIT 256
|
||||||
{Class::Wizard, "Wizard"},
|
#define PLAYER_CLASS_SHAMAN_BIT 512
|
||||||
{Class::Magician, "Magician"},
|
#define PLAYER_CLASS_NECROMANCER_BIT 1024
|
||||||
{Class::Enchanter, "Enchanter"},
|
#define PLAYER_CLASS_WIZARD_BIT 2048
|
||||||
{Class::Beastlord, "Beastlord"},
|
#define PLAYER_CLASS_MAGICIAN_BIT 4096
|
||||||
{Class::Berserker, "Berserker"},
|
#define PLAYER_CLASS_ENCHANTER_BIT 8192
|
||||||
};
|
#define PLAYER_CLASS_BEASTLORD_BIT 16384
|
||||||
|
#define PLAYER_CLASS_BERSERKER_BIT 32768
|
||||||
|
|
||||||
|
#define PLAYER_CLASS_ALL_MASK 65535 // was 65536
|
||||||
|
|
||||||
|
|
||||||
#define ARMOR_TYPE_UNKNOWN 0
|
#define ARMOR_TYPE_UNKNOWN 0
|
||||||
@@ -133,12 +125,13 @@ static std::map<uint8, std::string> class_names = {
|
|||||||
|
|
||||||
|
|
||||||
const char* GetClassIDName(uint8 class_id, uint8 level = 0);
|
const char* GetClassIDName(uint8 class_id, uint8 level = 0);
|
||||||
|
const char* GetPlayerClassName(uint32 player_class_value, uint8 level = 0);
|
||||||
|
|
||||||
bool IsPlayerClass(uint8 class_id);
|
uint32 GetPlayerClassValue(uint8 class_id);
|
||||||
const std::string GetPlayerClassAbbreviation(uint8 class_id);
|
uint32 GetPlayerClassBit(uint8 class_id);
|
||||||
|
|
||||||
uint8 GetPlayerClassValue(uint8 class_id);
|
uint8 GetClassIDFromPlayerClassValue(uint32 player_class_value);
|
||||||
uint16 GetPlayerClassBit(uint8 class_id);
|
uint8 GetClassIDFromPlayerClassBit(uint32 player_class_bit);
|
||||||
|
|
||||||
bool IsFighterClass(uint8 class_id);
|
bool IsFighterClass(uint8 class_id);
|
||||||
bool IsSpellFighterClass(uint8 class_id);
|
bool IsSpellFighterClass(uint8 class_id);
|
||||||
@@ -147,8 +140,7 @@ bool IsHybridClass(uint8 class_id);
|
|||||||
bool IsCasterClass(uint8 class_id);
|
bool IsCasterClass(uint8 class_id);
|
||||||
bool IsINTCasterClass(uint8 class_id);
|
bool IsINTCasterClass(uint8 class_id);
|
||||||
bool IsWISCasterClass(uint8 class_id);
|
bool IsWISCasterClass(uint8 class_id);
|
||||||
bool IsHeroicINTCasterClass(uint8 class_id);
|
|
||||||
bool IsHeroicWISCasterClass(uint8 class_id);
|
|
||||||
bool IsPlateClass(uint8 class_id);
|
bool IsPlateClass(uint8 class_id);
|
||||||
bool IsChainClass(uint8 class_id);
|
bool IsChainClass(uint8 class_id);
|
||||||
bool IsLeatherClass(uint8 class_id);
|
bool IsLeatherClass(uint8 class_id);
|
||||||
|
|||||||
@@ -39,15 +39,15 @@ namespace EQEmuCommand {
|
|||||||
{
|
{
|
||||||
if (cmd[{"-d", "--debug"}]) {
|
if (cmd[{"-d", "--debug"}]) {
|
||||||
std::cout << "Positional args:\n";
|
std::cout << "Positional args:\n";
|
||||||
for (auto &pos_arg: cmd.pos_args())
|
for (auto &pos_arg : cmd.pos_args())
|
||||||
std::cout << '\t' << pos_arg << std::endl;
|
std::cout << '\t' << pos_arg << std::endl;
|
||||||
|
|
||||||
std::cout << "\nFlags:\n";
|
std::cout << "\nFlags:\n";
|
||||||
for (auto &flag: cmd.flags())
|
for (auto &flag : cmd.flags())
|
||||||
std::cout << '\t' << flag << std::endl;
|
std::cout << '\t' << flag << std::endl;
|
||||||
|
|
||||||
std::cout << "\nParameters:\n";
|
std::cout << "\nParameters:\n";
|
||||||
for (auto ¶m: cmd.params())
|
for (auto ¶m : cmd.params())
|
||||||
std::cout << '\t' << param.first << " : " << param.second << std::endl;
|
std::cout << '\t' << param.first << " : " << param.second << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -69,22 +69,22 @@ namespace EQEmuCommand {
|
|||||||
{
|
{
|
||||||
bool arguments_filled = true;
|
bool arguments_filled = true;
|
||||||
|
|
||||||
int index = 2;
|
int index = 2;
|
||||||
for (auto &arg: arguments) {
|
for (auto &arg : arguments) {
|
||||||
if (cmd(arg).str().empty() && cmd(index).str().empty()) {
|
if (cmd(arg).str().empty() && cmd(index).str().empty()) {
|
||||||
arguments_filled = false;
|
arguments_filled = false;
|
||||||
}
|
}
|
||||||
index++;
|
index++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!arguments_filled || (argc == 2 && !cmd[{"-h", "--help"}]) || (argc == 3 && cmd[{"-h", "--help"}])) {
|
if (!arguments_filled || argc == 2 || cmd[{"-h", "--help"}]) {
|
||||||
std::string arguments_string;
|
std::string arguments_string;
|
||||||
for (auto &arg: arguments) {
|
for (auto &arg : arguments) {
|
||||||
arguments_string += " " + arg;
|
arguments_string += " " + arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string options_string;
|
std::string options_string;
|
||||||
for (auto &opt: options) {
|
for (auto &opt : options) {
|
||||||
options_string += " " + opt + "\n";
|
options_string += " " + opt + "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -124,6 +124,14 @@ namespace EQEmuCommand {
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
std::string description;
|
std::string description;
|
||||||
|
bool ran_command = false;
|
||||||
|
for (auto &it: in_function_map) {
|
||||||
|
if (it.first == argv[1]) {
|
||||||
|
(it.second)(argc, argv, cmd, description);
|
||||||
|
ran_command = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (cmd[{"-h", "--help"}]) {
|
if (cmd[{"-h", "--help"}]) {
|
||||||
std::cout << std::endl;
|
std::cout << std::endl;
|
||||||
std::cout <<
|
std::cout <<
|
||||||
@@ -134,7 +142,9 @@ namespace EQEmuCommand {
|
|||||||
<< std::endl
|
<< std::endl
|
||||||
<< std::endl;
|
<< std::endl;
|
||||||
|
|
||||||
// Get max command length for padding length
|
/**
|
||||||
|
* Get max command length for padding length
|
||||||
|
*/
|
||||||
int max_command_length = 0;
|
int max_command_length = 0;
|
||||||
|
|
||||||
for (auto &it: in_function_map) {
|
for (auto &it: in_function_map) {
|
||||||
@@ -145,14 +155,18 @@ namespace EQEmuCommand {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display command menu
|
/**
|
||||||
|
* Display command menu
|
||||||
|
*/
|
||||||
std::string command_section;
|
std::string command_section;
|
||||||
for (auto &it: in_function_map) {
|
for (auto &it: in_function_map) {
|
||||||
description.clear();
|
description = "";
|
||||||
|
|
||||||
(it.second)(argc, argv, cmd, description);
|
(it.second)(argc, argv, cmd, description);
|
||||||
|
|
||||||
// Print section header
|
/**
|
||||||
|
* Print section header
|
||||||
|
*/
|
||||||
std::string command_prefix = it.first.substr(0, it.first.find(":"));
|
std::string command_prefix = it.first.substr(0, it.first.find(":"));
|
||||||
|
|
||||||
if (command_prefix.find("test") != std::string::npos) {
|
if (command_prefix.find("test") != std::string::npos) {
|
||||||
@@ -164,7 +178,9 @@ namespace EQEmuCommand {
|
|||||||
std::cout << termcolor::reset << command_prefix << std::endl;
|
std::cout << termcolor::reset << command_prefix << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print commands
|
/**
|
||||||
|
* Print commands
|
||||||
|
*/
|
||||||
std::stringstream command;
|
std::stringstream command;
|
||||||
command << termcolor::colorize << termcolor::yellow << it.first << termcolor::reset;
|
command << termcolor::colorize << termcolor::yellow << it.first << termcolor::reset;
|
||||||
printf(" %-*s %s\n", max_command_length, command.str().c_str(), description.c_str());
|
printf(" %-*s %s\n", max_command_length, command.str().c_str(), description.c_str());
|
||||||
@@ -175,15 +191,6 @@ namespace EQEmuCommand {
|
|||||||
std::exit(0);
|
std::exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ran_command = false;
|
|
||||||
|
|
||||||
for (auto &it: in_function_map) {
|
|
||||||
if (it.first == argv[1]) {
|
|
||||||
(it.second)(argc, argv, cmd, description);
|
|
||||||
ran_command = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ran_command) {
|
if (ran_command) {
|
||||||
std::exit(0);
|
std::exit(0);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,28 @@
|
|||||||
#include "world_content_service.h"
|
/**
|
||||||
|
* EQEmulator: Everquest Server Emulator
|
||||||
|
* Copyright (C) 2001-2019 EQEmulator Development Team (https://github.com/EQEmu/Server)
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; version 2 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY except by those people which sell it, which
|
||||||
|
* are required to give you total support for your newly bought product;
|
||||||
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||||||
|
* A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
#include <utility>
|
#include "world_content_service.h"
|
||||||
#include <glm/vec3.hpp>
|
|
||||||
#include "../database.h"
|
#include "../database.h"
|
||||||
#include "../rulesys.h"
|
#include "../rulesys.h"
|
||||||
#include "../eqemu_logsys.h"
|
#include "../eqemu_logsys.h"
|
||||||
#include "../repositories/instance_list_repository.h"
|
#include "../repositories/content_flags_repository.h"
|
||||||
#include "../zone_store.h"
|
|
||||||
|
|
||||||
|
|
||||||
WorldContentService::WorldContentService()
|
WorldContentService::WorldContentService()
|
||||||
@@ -19,12 +35,8 @@ int WorldContentService::GetCurrentExpansion() const
|
|||||||
return current_expansion;
|
return current_expansion;
|
||||||
}
|
}
|
||||||
|
|
||||||
WorldContentService *WorldContentService::SetExpansionContext()
|
void WorldContentService::SetExpansionContext()
|
||||||
{
|
{
|
||||||
// do a rule manager reload until where we store expansion is changed to somewhere else
|
|
||||||
RuleManager::Instance()->LoadRules(GetDatabase(), "default", true);
|
|
||||||
|
|
||||||
// pull expansion from rules
|
|
||||||
int expansion = RuleI(Expansion, CurrentExpansion);
|
int expansion = RuleI(Expansion, CurrentExpansion);
|
||||||
if (expansion >= Expansion::Classic && expansion <= Expansion::MaxId) {
|
if (expansion >= Expansion::Classic && expansion <= Expansion::MaxId) {
|
||||||
content_service.SetCurrentExpansion(expansion);
|
content_service.SetCurrentExpansion(expansion);
|
||||||
@@ -35,8 +47,6 @@ WorldContentService *WorldContentService::SetExpansionContext()
|
|||||||
GetCurrentExpansion(),
|
GetCurrentExpansion(),
|
||||||
GetCurrentExpansionName()
|
GetCurrentExpansionName()
|
||||||
);
|
);
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string WorldContentService::GetCurrentExpansionName()
|
std::string WorldContentService::GetCurrentExpansionName()
|
||||||
@@ -63,47 +73,15 @@ void WorldContentService::SetCurrentExpansion(int current_expansion)
|
|||||||
/**
|
/**
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
const std::vector<ContentFlagsRepository::ContentFlags> &WorldContentService::GetContentFlags() const
|
const std::vector<std::string> &WorldContentService::GetContentFlags() const
|
||||||
{
|
{
|
||||||
return content_flags;
|
return content_flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
std::vector<std::string> WorldContentService::GetContentFlagsEnabled()
|
|
||||||
{
|
|
||||||
std::vector<std::string> enabled_flags;
|
|
||||||
|
|
||||||
for (auto &f: GetContentFlags()) {
|
|
||||||
if (f.enabled) {
|
|
||||||
enabled_flags.emplace_back(f.flag_name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return enabled_flags;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
std::vector<std::string> WorldContentService::GetContentFlagsDisabled()
|
|
||||||
{
|
|
||||||
std::vector<std::string> disabled_flags;
|
|
||||||
|
|
||||||
for (auto &f: GetContentFlags()) {
|
|
||||||
if (!f.enabled) {
|
|
||||||
disabled_flags.emplace_back(f.flag_name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return disabled_flags;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param content_flags
|
* @param content_flags
|
||||||
*/
|
*/
|
||||||
void WorldContentService::SetContentFlags(const std::vector<ContentFlagsRepository::ContentFlags> &content_flags)
|
void WorldContentService::SetContentFlags(std::vector<std::string> content_flags)
|
||||||
{
|
{
|
||||||
WorldContentService::content_flags = content_flags;
|
WorldContentService::content_flags = content_flags;
|
||||||
}
|
}
|
||||||
@@ -112,10 +90,10 @@ void WorldContentService::SetContentFlags(const std::vector<ContentFlagsReposito
|
|||||||
* @param content_flag
|
* @param content_flag
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
bool WorldContentService::IsContentFlagEnabled(const std::string &content_flag)
|
bool WorldContentService::IsContentFlagEnabled(const std::string& content_flag)
|
||||||
{
|
{
|
||||||
for (auto &f: GetContentFlags()) {
|
for (auto &flag : GetContentFlags()) {
|
||||||
if (f.flag_name == content_flag && f.enabled == true) {
|
if (flag == content_flag) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -123,227 +101,20 @@ bool WorldContentService::IsContentFlagEnabled(const std::string &content_flag)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
void WorldContentService::ReloadContentFlags(Database &db)
|
||||||
* @param content_flag
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
bool WorldContentService::IsContentFlagDisabled(const std::string &content_flag)
|
|
||||||
{
|
{
|
||||||
for (auto &f: GetContentFlags()) {
|
std::vector<std::string> set_content_flags;
|
||||||
if (f.flag_name == content_flag && f.enabled == false) {
|
auto content_flags = ContentFlagsRepository::GetWhere(db, "enabled = 1");
|
||||||
return true;
|
|
||||||
}
|
set_content_flags.reserve(content_flags.size());
|
||||||
|
for (auto &flags: content_flags) {
|
||||||
|
set_content_flags.push_back(flags.flag_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
LogInfo(
|
||||||
}
|
"Enabled content flags [{}]",
|
||||||
|
implode(", ", set_content_flags)
|
||||||
bool WorldContentService::DoesPassContentFiltering(const ContentFlags &f)
|
);
|
||||||
{
|
|
||||||
// if we're not set to (-1 All) then fail when we aren't within minimum expansion
|
|
||||||
if (f.min_expansion > Expansion::EXPANSION_ALL && current_expansion < f.min_expansion && current_expansion != -1) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if we're not set to (-1 All) then fail when we aren't within max expansion
|
|
||||||
if (f.max_expansion > Expansion::EXPANSION_ALL && current_expansion > f.max_expansion && current_expansion != -1) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if we don't have any enabled flag in enabled flags, we fail
|
|
||||||
for (const auto &flag: Strings::Split(f.content_flags)) {
|
|
||||||
if (!Strings::Contains(GetContentFlagsEnabled(), flag)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// if we don't have any disabled flag in disabled flags, we fail
|
|
||||||
for (const auto &flag: Strings::Split(f.content_flags_disabled)) {
|
|
||||||
if (!Strings::Contains(GetContentFlagsDisabled(), flag)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WorldContentService::ReloadContentFlags()
|
|
||||||
{
|
|
||||||
std::vector<ContentFlagsRepository::ContentFlags> set_content_flags;
|
|
||||||
auto flags = ContentFlagsRepository::All(*GetDatabase());
|
|
||||||
|
|
||||||
set_content_flags.reserve(flags.size());
|
|
||||||
for (auto &f: flags) {
|
|
||||||
set_content_flags.push_back(f);
|
|
||||||
|
|
||||||
LogInfo(
|
|
||||||
"Loaded content flag [{}] [{}]",
|
|
||||||
f.flag_name,
|
|
||||||
(f.enabled ? "enabled" : "disabled")
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
SetContentFlags(set_content_flags);
|
SetContentFlags(set_content_flags);
|
||||||
LoadStaticGlobalZoneInstances();
|
|
||||||
zone_store.LoadZones(*m_content_database);
|
|
||||||
}
|
|
||||||
|
|
||||||
Database *WorldContentService::GetDatabase() const
|
|
||||||
{
|
|
||||||
return m_database;
|
|
||||||
}
|
|
||||||
|
|
||||||
WorldContentService *WorldContentService::SetDatabase(Database *database)
|
|
||||||
{
|
|
||||||
WorldContentService::m_database = database;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
Database *WorldContentService::GetContentDatabase() const
|
|
||||||
{
|
|
||||||
return m_content_database;
|
|
||||||
}
|
|
||||||
|
|
||||||
WorldContentService *WorldContentService::SetContentDatabase(Database *database)
|
|
||||||
{
|
|
||||||
WorldContentService::m_content_database = database;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WorldContentService::SetContentFlag(const std::string &content_flag_name, bool enabled)
|
|
||||||
{
|
|
||||||
auto flags = ContentFlagsRepository::GetWhere(
|
|
||||||
*GetDatabase(),
|
|
||||||
fmt::format("flag_name = '{}'", content_flag_name)
|
|
||||||
);
|
|
||||||
|
|
||||||
auto f = ContentFlagsRepository::NewEntity();
|
|
||||||
if (!flags.empty()) {
|
|
||||||
f = flags.front();
|
|
||||||
}
|
|
||||||
|
|
||||||
f.enabled = enabled ? 1 : 0;
|
|
||||||
f.flag_name = content_flag_name;
|
|
||||||
|
|
||||||
if (!flags.empty()) {
|
|
||||||
ContentFlagsRepository::UpdateOne(*GetDatabase(), f);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
ContentFlagsRepository::InsertOne(*GetDatabase(), f);
|
|
||||||
}
|
|
||||||
|
|
||||||
ReloadContentFlags();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WorldContentService::HandleZoneRoutingMiddleware(ZoneChange_Struct *zc)
|
|
||||||
{
|
|
||||||
auto r = FindZone(zc->zoneID, zc->instanceID);
|
|
||||||
if (r.zone_id == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
zc->instanceID = r.instance.id;
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadStaticGlobalZoneInstances loads all static global zone instances
|
|
||||||
// these are zones that are never set to expire and are global
|
|
||||||
// these are used commonly in v1/v2/v3 versions of the same zone for expansion routing
|
|
||||||
WorldContentService *WorldContentService::LoadStaticGlobalZoneInstances()
|
|
||||||
{
|
|
||||||
m_zone_static_instances = InstanceListRepository::GetWhere(
|
|
||||||
*GetDatabase(),
|
|
||||||
fmt::format("never_expires = 1 AND is_global = 1")
|
|
||||||
);
|
|
||||||
|
|
||||||
LogInfo("Loaded [{}] zone_instances", m_zone_static_instances.size());
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindZone handles content and context aware zone routing (middleware)
|
|
||||||
//
|
|
||||||
// this is a middleware function that is meant to be used in the zone change process
|
|
||||||
// this hooks all core zone changes within the server and routes the player to the correct zone
|
|
||||||
// returning a zone_id of non-zero means the middleware will route the player
|
|
||||||
// returning a zone_id of 0 means the middleware will not route the player
|
|
||||||
// this is useful for handling multiple versions of the same zone
|
|
||||||
//
|
|
||||||
// implementation >
|
|
||||||
// the zoning and process spawning logic already is handled by two keys "zone_id" and "instance_id"
|
|
||||||
// we leverage static, never expires instances to handle this and client still sees it as a normal zone
|
|
||||||
//
|
|
||||||
// content awareness >
|
|
||||||
// simply use the zone_id, server content settings and the middleware will handle the rest
|
|
||||||
// you don't have to think about instances in any data tables (use instance_id 0)
|
|
||||||
// you don't have to keep track of instance ids in scripts (use instance_id 0)
|
|
||||||
// the versions of zones are represented by two zone entries that have potentially different min/max expansion and/or different content flags
|
|
||||||
// we decide to route the client to the correct version of the zone based on the current server side expansion
|
|
||||||
//
|
|
||||||
// example >
|
|
||||||
// we want to route players to the correct version of lavastorm based on the current server side expansion (DoesZonePassContentFiltering)
|
|
||||||
// lavastorm (pre-don) version 0 (classic)
|
|
||||||
// zone table entry for version = 0, min_expansion = 0, max_expansion = 8
|
|
||||||
// instance_list table entry for lavastorm has version = 0, is_global = 1, never_expires = 1
|
|
||||||
// lavastorm (don) version 1
|
|
||||||
// zone table entry for version = 1, min_expansion = 9, max_expansion = 99
|
|
||||||
// instance_list table entry for lavastorm has version = 1, is_global = 1, never_expires = 1
|
|
||||||
WorldContentService::FindZoneResult WorldContentService::FindZone(uint32 zone_id, uint32 instance_id)
|
|
||||||
{
|
|
||||||
for (const auto &z: zone_store.GetZones()) {
|
|
||||||
for (auto &i: m_zone_static_instances) {
|
|
||||||
if (
|
|
||||||
z.zoneidnumber == zone_id &&
|
|
||||||
DoesZonePassContentFiltering(z) &&
|
|
||||||
i.zone == zone_id &&
|
|
||||||
i.version == z.version) {
|
|
||||||
|
|
||||||
if (instance_id > 0 && i.id != instance_id) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
LogInfo(
|
|
||||||
"Routed player to public static instance [{}] of zone [{}] ({}) version [{}] long_name [{}] notes [{}]",
|
|
||||||
i.id,
|
|
||||||
z.short_name,
|
|
||||||
z.zoneidnumber,
|
|
||||||
z.version,
|
|
||||||
z.long_name,
|
|
||||||
i.notes
|
|
||||||
);
|
|
||||||
|
|
||||||
return WorldContentService::FindZoneResult{
|
|
||||||
.zone_id = static_cast<uint32>(z.zoneidnumber),
|
|
||||||
.instance = i,
|
|
||||||
.zone = z
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return WorldContentService::FindZoneResult{.zone_id = 0};
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WorldContentService::IsInPublicStaticInstance(uint32 instance_id)
|
|
||||||
{
|
|
||||||
for (auto &i: m_zone_static_instances) {
|
|
||||||
if (i.id == instance_id) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WorldContentService::DoesZonePassContentFiltering(const ZoneRepository::Zone &z)
|
|
||||||
{
|
|
||||||
auto f = ContentFlags{
|
|
||||||
.min_expansion = z.min_expansion,
|
|
||||||
.max_expansion = z.max_expansion,
|
|
||||||
.content_flags = z.content_flags,
|
|
||||||
.content_flags_disabled = z.content_flags_disabled
|
|
||||||
};
|
|
||||||
|
|
||||||
return DoesPassContentFiltering(f);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,21 +1,31 @@
|
|||||||
|
/**
|
||||||
|
* EQEmulator: Everquest Server Emulator
|
||||||
|
* Copyright (C) 2001-2019 EQEmulator Development Team (https://github.com/EQEmu/Server)
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; version 2 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY except by those people which sell it, which
|
||||||
|
* are required to give you total support for your newly bought product;
|
||||||
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||||||
|
* A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef EQEMU_WORLD_CONTENT_SERVICE_H
|
#ifndef EQEMU_WORLD_CONTENT_SERVICE_H
|
||||||
#define EQEMU_WORLD_CONTENT_SERVICE_H
|
#define EQEMU_WORLD_CONTENT_SERVICE_H
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "../repositories/content_flags_repository.h"
|
|
||||||
#include "../repositories/zone_repository.h"
|
|
||||||
#include "../repositories/instance_list_repository.h"
|
|
||||||
|
|
||||||
class Database;
|
class Database;
|
||||||
|
|
||||||
struct ContentFlags {
|
|
||||||
int16 min_expansion;
|
|
||||||
int16 max_expansion;
|
|
||||||
std::string content_flags;
|
|
||||||
std::string content_flags_disabled;
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace Expansion {
|
namespace Expansion {
|
||||||
static const int EXPANSION_ALL = -1;
|
static const int EXPANSION_ALL = -1;
|
||||||
static const int EXPANSION_FILTER_MAX = 99;
|
static const int EXPANSION_FILTER_MAX = 99;
|
||||||
@@ -42,7 +52,7 @@ namespace Expansion {
|
|||||||
VeilOfAlaris,
|
VeilOfAlaris,
|
||||||
RainOfFear,
|
RainOfFear,
|
||||||
CallOfTheForsaken,
|
CallOfTheForsaken,
|
||||||
TheDarkenedSea,
|
TheDarkendSea,
|
||||||
TheBrokenMirror,
|
TheBrokenMirror,
|
||||||
EmpiresOfKunark,
|
EmpiresOfKunark,
|
||||||
RingOfScale,
|
RingOfScale,
|
||||||
@@ -115,7 +125,7 @@ public:
|
|||||||
bool IsVeilOfAlarisEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::VeilOfAlaris || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsVeilOfAlarisEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::VeilOfAlaris || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsRainOfFearEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::RainOfFear || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsRainOfFearEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::RainOfFear || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsCallOfTheForsakenEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::CallOfTheForsaken || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsCallOfTheForsakenEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::CallOfTheForsaken || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsTheDarkenedSeaEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::TheDarkenedSea || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsTheDarkendSeaEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::TheDarkendSea || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsTheBrokenMirrorEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::TheBrokenMirror || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsTheBrokenMirrorEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::TheBrokenMirror || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsEmpiresOfKunarkEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::EmpiresOfKunark || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsEmpiresOfKunarkEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::EmpiresOfKunark || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
bool IsRingOfScaleEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::RingOfScale || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
bool IsRingOfScaleEnabled() { return GetCurrentExpansion() >= Expansion::ExpansionNumber::RingOfScale || GetCurrentExpansion() == Expansion::EXPANSION_ALL; }
|
||||||
@@ -143,55 +153,22 @@ public:
|
|||||||
bool IsCurrentExpansionVeilOfAlaris() { return current_expansion == Expansion::ExpansionNumber::VeilOfAlaris; }
|
bool IsCurrentExpansionVeilOfAlaris() { return current_expansion == Expansion::ExpansionNumber::VeilOfAlaris; }
|
||||||
bool IsCurrentExpansionRainOfFear() { return current_expansion == Expansion::ExpansionNumber::RainOfFear; }
|
bool IsCurrentExpansionRainOfFear() { return current_expansion == Expansion::ExpansionNumber::RainOfFear; }
|
||||||
bool IsCurrentExpansionCallOfTheForsaken() { return current_expansion == Expansion::ExpansionNumber::CallOfTheForsaken; }
|
bool IsCurrentExpansionCallOfTheForsaken() { return current_expansion == Expansion::ExpansionNumber::CallOfTheForsaken; }
|
||||||
bool IsCurrentExpansionTheDarkenedSea() { return current_expansion == Expansion::ExpansionNumber::TheDarkenedSea; }
|
bool IsCurrentExpansionTheDarkendSea() { return current_expansion == Expansion::ExpansionNumber::TheDarkendSea; }
|
||||||
bool IsCurrentExpansionTheBrokenMirror() { return current_expansion == Expansion::ExpansionNumber::TheBrokenMirror; }
|
bool IsCurrentExpansionTheBrokenMirror() { return current_expansion == Expansion::ExpansionNumber::TheBrokenMirror; }
|
||||||
bool IsCurrentExpansionEmpiresOfKunark() { return current_expansion == Expansion::ExpansionNumber::EmpiresOfKunark; }
|
bool IsCurrentExpansionEmpiresOfKunark() { return current_expansion == Expansion::ExpansionNumber::EmpiresOfKunark; }
|
||||||
bool IsCurrentExpansionRingOfScale() { return current_expansion == Expansion::ExpansionNumber::RingOfScale; }
|
bool IsCurrentExpansionRingOfScale() { return current_expansion == Expansion::ExpansionNumber::RingOfScale; }
|
||||||
bool IsCurrentExpansionTheBurningLands() { return current_expansion == Expansion::ExpansionNumber::TheBurningLands; }
|
bool IsCurrentExpansionTheBurningLands() { return current_expansion == Expansion::ExpansionNumber::TheBurningLands; }
|
||||||
bool IsCurrentExpansionTormentOfVelious() { return current_expansion == Expansion::ExpansionNumber::TormentOfVelious; }
|
bool IsCurrentExpansionTormentOfVelious() { return current_expansion == Expansion::ExpansionNumber::TormentOfVelious; }
|
||||||
|
|
||||||
const std::vector<ContentFlagsRepository::ContentFlags> &GetContentFlags() const;
|
|
||||||
std::vector<std::string> GetContentFlagsEnabled();
|
|
||||||
std::vector<std::string> GetContentFlagsDisabled();
|
|
||||||
bool IsContentFlagEnabled(const std::string& content_flag);
|
|
||||||
bool IsContentFlagDisabled(const std::string& content_flag);
|
|
||||||
void SetContentFlags(const std::vector<ContentFlagsRepository::ContentFlags>& content_flags);
|
|
||||||
void ReloadContentFlags();
|
|
||||||
WorldContentService * SetExpansionContext();
|
|
||||||
|
|
||||||
bool DoesPassContentFiltering(const ContentFlags& f);
|
|
||||||
bool DoesZonePassContentFiltering(const ZoneRepository::Zone& z);
|
|
||||||
|
|
||||||
WorldContentService * SetDatabase(Database *database);
|
|
||||||
Database *GetDatabase() const;
|
|
||||||
|
|
||||||
WorldContentService * SetContentDatabase(Database *database);
|
|
||||||
Database *GetContentDatabase() const;
|
|
||||||
|
|
||||||
void SetContentFlag(const std::string &content_flag_name, bool enabled);
|
|
||||||
|
|
||||||
void HandleZoneRoutingMiddleware(ZoneChange_Struct *zc);
|
|
||||||
|
|
||||||
struct FindZoneResult {
|
|
||||||
uint32 zone_id = 0;
|
|
||||||
InstanceListRepository::InstanceList instance;
|
|
||||||
ZoneRepository::Zone zone;
|
|
||||||
};
|
|
||||||
|
|
||||||
FindZoneResult FindZone(uint32 zone_id, uint32 instance_id);
|
|
||||||
bool IsInPublicStaticInstance(uint32 instance_id);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int current_expansion{};
|
int current_expansion{};
|
||||||
std::vector<ContentFlagsRepository::ContentFlags> content_flags;
|
std::vector<std::string> content_flags;
|
||||||
|
public:
|
||||||
// reference to database
|
const std::vector<std::string> &GetContentFlags() const;
|
||||||
Database *m_database;
|
bool IsContentFlagEnabled(const std::string& content_flag);
|
||||||
Database *m_content_database;
|
void SetContentFlags(std::vector<std::string> content_flags);
|
||||||
|
void ReloadContentFlags(Database &db);
|
||||||
// holds a record of the zone table from the database
|
void SetExpansionContext();
|
||||||
WorldContentService *LoadStaticGlobalZoneInstances();
|
|
||||||
std::vector<InstanceListRepository::InstanceList> m_zone_static_instances;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
extern WorldContentService content_service;
|
extern WorldContentService content_service;
|
||||||
|
|||||||
+9
-144
@@ -1,95 +1,6 @@
|
|||||||
#include "global_define.h"
|
#include "global_define.h"
|
||||||
#include "eqemu_logsys.h"
|
#include "eqemu_logsys.h"
|
||||||
#include "crash.h"
|
#include "crash.h"
|
||||||
#include "strings.h"
|
|
||||||
#include "process/process.h"
|
|
||||||
#include "http/httplib.h"
|
|
||||||
#include "http/uri.h"
|
|
||||||
#include "json/json.h"
|
|
||||||
#include "version.h"
|
|
||||||
#include "eqemu_config.h"
|
|
||||||
#include "serverinfo.h"
|
|
||||||
#include "rulesys.h"
|
|
||||||
#include "platform.h"
|
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#ifdef _WINDOWS
|
|
||||||
#define popen _popen
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void SendCrashReport(const std::string &crash_report)
|
|
||||||
{
|
|
||||||
// can configure multiple endpoints if need be
|
|
||||||
std::vector<std::string> endpoints = {
|
|
||||||
"https://spire.akkadius.com/api/v1/analytics/server-crash-report",
|
|
||||||
// "http://localhost:3010/api/v1/analytics/server-crash-report", // development
|
|
||||||
};
|
|
||||||
|
|
||||||
auto config = EQEmuConfig::get();
|
|
||||||
for (auto &e: endpoints) {
|
|
||||||
uri u(e);
|
|
||||||
|
|
||||||
std::string base_url = fmt::format("{}://{}", u.get_scheme(), u.get_host());
|
|
||||||
if (u.get_port()) {
|
|
||||||
base_url += fmt::format(":{}", u.get_port());
|
|
||||||
}
|
|
||||||
|
|
||||||
// client
|
|
||||||
httplib::Client r(base_url);
|
|
||||||
r.set_connection_timeout(1, 0);
|
|
||||||
r.set_read_timeout(1, 0);
|
|
||||||
r.set_write_timeout(1, 0);
|
|
||||||
|
|
||||||
// os info
|
|
||||||
auto os = EQ::GetOS();
|
|
||||||
auto cpus = EQ::GetCPUs();
|
|
||||||
auto process_id = EQ::GetPID();
|
|
||||||
auto rss = EQ::GetRSS() / 1048576.0;
|
|
||||||
auto uptime = static_cast<uint32>(EQ::GetUptime());
|
|
||||||
|
|
||||||
// payload
|
|
||||||
Json::Value p;
|
|
||||||
p["platform_name"] = GetPlatformName();
|
|
||||||
p["crash_report"] = crash_report;
|
|
||||||
p["server_version"] = CURRENT_VERSION;
|
|
||||||
p["compile_date"] = COMPILE_DATE;
|
|
||||||
p["compile_time"] = COMPILE_TIME;
|
|
||||||
p["server_name"] = config->LongName;
|
|
||||||
p["server_short_name"] = config->ShortName;
|
|
||||||
p["uptime"] = uptime;
|
|
||||||
p["os_machine"] = os.machine;
|
|
||||||
p["os_release"] = os.release;
|
|
||||||
p["os_version"] = os.version;
|
|
||||||
p["os_sysname"] = os.sysname;
|
|
||||||
p["process_id"] = process_id;
|
|
||||||
p["rss_memory"] = rss;
|
|
||||||
p["cpus"] = cpus.size();
|
|
||||||
p["origination_info"] = "";
|
|
||||||
|
|
||||||
if (!LogSys.origination_info.zone_short_name.empty()) {
|
|
||||||
p["origination_info"] = fmt::format(
|
|
||||||
"{} ({}) instance_id [{}]",
|
|
||||||
LogSys.origination_info.zone_short_name,
|
|
||||||
LogSys.origination_info.zone_long_name,
|
|
||||||
LogSys.origination_info.instance_id
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::stringstream payload;
|
|
||||||
payload << p;
|
|
||||||
|
|
||||||
if (auto res = r.Post(e, payload.str(), "application/json")) {
|
|
||||||
if (res->status == 200) {
|
|
||||||
LogInfo("Sent crash report");
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
LogError("Failed to send crash report to [{}]", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if defined(_WINDOWS) && defined(CRASH_LOGGING)
|
#if defined(_WINDOWS) && defined(CRASH_LOGGING)
|
||||||
#include "StackWalker.h"
|
#include "StackWalker.h"
|
||||||
@@ -101,30 +12,22 @@ public:
|
|||||||
EQEmuStackWalker(DWORD dwProcessId, HANDLE hProcess) : StackWalker(dwProcessId, hProcess) { }
|
EQEmuStackWalker(DWORD dwProcessId, HANDLE hProcess) : StackWalker(dwProcessId, hProcess) { }
|
||||||
virtual void OnOutput(LPCSTR szText) {
|
virtual void OnOutput(LPCSTR szText) {
|
||||||
char buffer[4096];
|
char buffer[4096];
|
||||||
for (int i = 0; i < 4096; ++i) {
|
for(int i = 0; i < 4096; ++i) {
|
||||||
if (szText[i] == 0) {
|
if(szText[i] == 0) {
|
||||||
buffer[i] = '\0';
|
buffer[i] = '\0';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (szText[i] == '\n' || szText[i] == '\r') {
|
if(szText[i] == '\n' || szText[i] == '\r') {
|
||||||
buffer[i] = ' ';
|
buffer[i] = ' ';
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
buffer[i] = szText[i];
|
buffer[i] = szText[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string line = buffer;
|
|
||||||
_lines.push_back(line);
|
|
||||||
|
|
||||||
Log(Logs::General, Logs::Crash, buffer);
|
Log(Logs::General, Logs::Crash, buffer);
|
||||||
StackWalker::OnOutput(szText);
|
StackWalker::OnOutput(szText);
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::vector<std::string>& GetLines() { return _lines; }
|
|
||||||
private:
|
|
||||||
std::vector<std::string> _lines;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
LONG WINAPI windows_exception_handler(EXCEPTION_POINTERS *ExceptionInfo)
|
LONG WINAPI windows_exception_handler(EXCEPTION_POINTERS *ExceptionInfo)
|
||||||
@@ -198,20 +101,7 @@ LONG WINAPI windows_exception_handler(EXCEPTION_POINTERS *ExceptionInfo)
|
|||||||
|
|
||||||
if(EXCEPTION_STACK_OVERFLOW != ExceptionInfo->ExceptionRecord->ExceptionCode)
|
if(EXCEPTION_STACK_OVERFLOW != ExceptionInfo->ExceptionRecord->ExceptionCode)
|
||||||
{
|
{
|
||||||
EQEmuStackWalker sw;
|
EQEmuStackWalker sw; sw.ShowCallstack(GetCurrentThread(), ExceptionInfo->ContextRecord);
|
||||||
sw.ShowCallstack(GetCurrentThread(), ExceptionInfo->ContextRecord);
|
|
||||||
|
|
||||||
if (RuleB(Analytics, CrashReporting)) {
|
|
||||||
std::string crash_report;
|
|
||||||
auto& lines = sw.GetLines();
|
|
||||||
|
|
||||||
for (auto& line : lines) {
|
|
||||||
crash_report += line;
|
|
||||||
crash_report += "\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
SendCrashReport(crash_report);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return EXCEPTION_EXECUTE_HANDLER;
|
return EXCEPTION_EXECUTE_HANDLER;
|
||||||
@@ -235,27 +125,9 @@ void set_exception_handler() {
|
|||||||
|
|
||||||
void print_trace()
|
void print_trace()
|
||||||
{
|
{
|
||||||
bool does_gdb_exist = Strings::Contains(Process::execute("gdb -v"), "GNU");
|
auto uid = geteuid();
|
||||||
if (!does_gdb_exist) {
|
|
||||||
LogCrash(
|
|
||||||
"[Error] GDB is not installed, if you want crash dumps on Linux to work properly you will need GDB installed"
|
|
||||||
);
|
|
||||||
std::exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto uid = geteuid();
|
std::string temp_output_file = "/tmp/dump-output";
|
||||||
std::string temp_output_file = fmt::format("/tmp/dump-output-{}", Strings::Random(10));
|
|
||||||
|
|
||||||
// check for passwordless sudo if not root
|
|
||||||
if (uid != 0) {
|
|
||||||
bool sudo_password_required = Strings::Contains(Process::execute("sudo -n true"), "a password is required");
|
|
||||||
if (sudo_password_required) {
|
|
||||||
LogCrash(
|
|
||||||
"[Error] Current user does not have passwordless sudo installed. It is required to automatically process crash dumps with GDB as non-root."
|
|
||||||
);
|
|
||||||
std::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
char pid_buf[30];
|
char pid_buf[30];
|
||||||
sprintf(pid_buf, "%d", getpid());
|
sprintf(pid_buf, "%d", getpid());
|
||||||
@@ -264,6 +136,7 @@ void print_trace()
|
|||||||
int child_pid = fork();
|
int child_pid = fork();
|
||||||
if (!child_pid) {
|
if (!child_pid) {
|
||||||
int fd = open(temp_output_file.c_str(), O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
|
int fd = open(temp_output_file.c_str(), O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
|
||||||
|
|
||||||
dup2(fd, 1); // redirect output to stderr
|
dup2(fd, 1); // redirect output to stderr
|
||||||
fprintf(stdout, "stack trace for %s pid=%s\n", name_buf, pid_buf);
|
fprintf(stdout, "stack trace for %s pid=%s\n", name_buf, pid_buf);
|
||||||
if (uid == 0) {
|
if (uid == 0) {
|
||||||
@@ -278,24 +151,16 @@ void print_trace()
|
|||||||
abort(); /* If gdb failed to start */
|
abort(); /* If gdb failed to start */
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
waitpid(child_pid, nullptr, 0);
|
waitpid(child_pid, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::ifstream input(temp_output_file);
|
std::ifstream input(temp_output_file);
|
||||||
std::string crash_report;
|
|
||||||
for (std::string line; getline(input, line);) {
|
for (std::string line; getline(input, line);) {
|
||||||
LogCrash("{}", line);
|
LogCrash("{}", line);
|
||||||
crash_report += fmt::format("{}\n", line);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::remove(temp_output_file.c_str());
|
std::remove(temp_output_file.c_str());
|
||||||
|
|
||||||
if (RuleB(Analytics, CrashReporting)) {
|
|
||||||
SendCrashReport(crash_report);
|
|
||||||
}
|
|
||||||
|
|
||||||
LogSys.CloseFileLogs();
|
|
||||||
|
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -313,7 +313,7 @@ namespace cron
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return static_cast<cron_int>(Strings::ToUnsignedInt(text.data()));
|
return static_cast<cron_int>(std::stoul(text.data()));
|
||||||
}
|
}
|
||||||
catch (std::exception const & ex)
|
catch (std::exception const & ex)
|
||||||
{
|
{
|
||||||
|
|||||||
+1907
-1612
File diff suppressed because it is too large
Load Diff
+159
-162
@@ -18,8 +18,8 @@
|
|||||||
#ifndef EQEMU_DATABASE_H
|
#ifndef EQEMU_DATABASE_H
|
||||||
#define EQEMU_DATABASE_H
|
#define EQEMU_DATABASE_H
|
||||||
|
|
||||||
#define AUTHENTICATION_TIMEOUT 60
|
#define AUTHENTICATION_TIMEOUT 60
|
||||||
#define INVALID_ID 0xFFFFFFFF
|
#define INVALID_ID 0xFFFFFFFF
|
||||||
|
|
||||||
#include "global_define.h"
|
#include "global_define.h"
|
||||||
#include "eqemu_logsys.h"
|
#include "eqemu_logsys.h"
|
||||||
@@ -34,11 +34,14 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
|
//atoi is not uint32 or uint32 safe!!!!
|
||||||
|
#define atoul(str) strtoul(str, nullptr, 10)
|
||||||
|
|
||||||
class MySQLRequestResult;
|
class MySQLRequestResult;
|
||||||
class Client;
|
class Client;
|
||||||
|
|
||||||
namespace EQ {
|
namespace EQ
|
||||||
|
{
|
||||||
class InventoryProfile;
|
class InventoryProfile;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -51,11 +54,10 @@ struct npcDecayTimes_Struct {
|
|||||||
|
|
||||||
struct VarCache_Struct {
|
struct VarCache_Struct {
|
||||||
std::map<std::string, std::string> m_cache;
|
std::map<std::string, std::string> m_cache;
|
||||||
uint32 last_update;
|
uint32 last_update;
|
||||||
VarCache_Struct() : last_update(0) { }
|
VarCache_Struct() : last_update(0) { }
|
||||||
void Add(const std::string& key, const std::string& value) { m_cache[key] = value; }
|
void Add(const std::string &key, const std::string &value) { m_cache[key] = value; }
|
||||||
const std::string* Get(const std::string& key)
|
const std::string *Get(const std::string &key) {
|
||||||
{
|
|
||||||
auto it = m_cache.find(key);
|
auto it = m_cache.find(key);
|
||||||
return (it != m_cache.end() ? &it->second : nullptr);
|
return (it != m_cache.end() ? &it->second : nullptr);
|
||||||
}
|
}
|
||||||
@@ -79,208 +81,203 @@ class PTimerList;
|
|||||||
class Database : public DBcore {
|
class Database : public DBcore {
|
||||||
public:
|
public:
|
||||||
Database();
|
Database();
|
||||||
Database(
|
Database(const char* host, const char* user, const char* passwd, const char* database,uint32 port);
|
||||||
const std::string& host,
|
bool Connect(const char* host, const char* user, const char* passwd, const char* database, uint32 port, std::string connection_label = "default");
|
||||||
const std::string& user,
|
|
||||||
const std::string& password,
|
|
||||||
const std::string& database,
|
|
||||||
uint32 port
|
|
||||||
);
|
|
||||||
bool Connect(
|
|
||||||
const std::string& host,
|
|
||||||
const std::string& user,
|
|
||||||
const std::string& password,
|
|
||||||
const std::string& database,
|
|
||||||
uint32 port,
|
|
||||||
std::string connection_label = "default"
|
|
||||||
);
|
|
||||||
~Database();
|
~Database();
|
||||||
|
|
||||||
/* Character Creation */
|
/* Character Creation */
|
||||||
bool DeleteCharacter(const std::string& name);
|
|
||||||
bool MoveCharacterToZone(const std::string& name, uint32 zone_id);
|
bool AddToNameFilter(const char *name);
|
||||||
|
bool CreateCharacter(
|
||||||
|
uint32 account_id,
|
||||||
|
char *name,
|
||||||
|
uint16 gender,
|
||||||
|
uint16 race,
|
||||||
|
uint16 class_,
|
||||||
|
uint8 str,
|
||||||
|
uint8 sta,
|
||||||
|
uint8 cha,
|
||||||
|
uint8 dex,
|
||||||
|
uint8 int_,
|
||||||
|
uint8 agi,
|
||||||
|
uint8 wis,
|
||||||
|
uint8 face
|
||||||
|
);
|
||||||
|
bool DeleteCharacter(char *character_name);
|
||||||
|
bool MoveCharacterToZone(const char *charname, uint32 zone_id);
|
||||||
bool MoveCharacterToZone(uint32 character_id, uint32 zone_id);
|
bool MoveCharacterToZone(uint32 character_id, uint32 zone_id);
|
||||||
bool ReserveName(uint32 account_id, const std::string& name);
|
bool ReserveName(uint32 account_id, char *name);
|
||||||
bool SaveCharacterCreate(uint32 character_id, uint32 account_id, PlayerProfile_Struct* pp);
|
bool SaveCharacterCreate(uint32 character_id, uint32 account_id, PlayerProfile_Struct *pp);
|
||||||
bool UpdateName(const std::string& old_name, const std::string& new_name);
|
bool SetHackerFlag(const char *accountname, const char *charactername, const char *hacked);
|
||||||
|
bool SetMQDetectionFlag(const char *accountname, const char *charactername, const char *hacked, const char *zone);
|
||||||
|
bool SetMQDetectionFlag(const char *accountname, const char *charactername, const std::string &hacked, const char *zone);
|
||||||
|
bool UpdateName(const char *oldname, const char *newname);
|
||||||
bool CopyCharacter(
|
bool CopyCharacter(
|
||||||
const std::string& source_character_name,
|
std::string source_character_name,
|
||||||
const std::string& destination_character_name,
|
std::string destination_character_name,
|
||||||
const std::string& destination_account_name
|
std::string destination_account_name
|
||||||
);
|
);
|
||||||
|
|
||||||
/* General Information Queries */
|
/* General Information Queries */
|
||||||
bool AddBannedIP(const std::string& banned_ip, const std::string& notes); //Add IP address to the banned_ips table.
|
|
||||||
bool AddToNameFilter(const std::string& name);
|
|
||||||
bool CheckBannedIPs(const std::string& login_ip); //Check incoming connection against banned IP table.
|
|
||||||
bool CheckGMIPs(const std::string& login_ip, uint32 account_id);
|
|
||||||
bool CheckNameFilter(const std::string& name, bool surname = false);
|
|
||||||
bool IsNameUsed(const std::string& name);
|
|
||||||
|
|
||||||
uint32 GetAccountIDByChar(const std::string& name, uint32* character_id = 0);
|
bool AddBannedIP(char* bannedIP, const char* notes); //Add IP address to the banned_ips table.
|
||||||
uint32 GetAccountIDByChar(uint32 character_id);
|
bool AddGMIP(char* ip_address, char* name);
|
||||||
uint32 GetAccountIDByName(const std::string& account_name, const std::string& loginserver, int16* status = 0, uint32* lsid = 0);
|
bool CheckBannedIPs(const char* loginIP); //Check incoming connection against banned IP table.
|
||||||
uint32 GetCharacterID(const std::string& name);
|
bool CheckGMIPs(const char* loginIP, uint32 account_id);
|
||||||
uint32 GetGuildIDByCharID(uint32 character_id);
|
bool CheckNameFilter(const char* name, bool surname = false);
|
||||||
uint32 GetGroupIDByCharID(uint32 character_id);
|
bool CheckUsedName(const char* name);
|
||||||
uint32 GetRaidIDByCharID(uint32 character_id);
|
|
||||||
|
|
||||||
const std::string GetAccountName(uint32 account_id, uint32* lsaccount_id = 0);
|
uint32 GetAccountIDByChar(const char* charname, uint32* oCharID = 0);
|
||||||
const std::string GetCharName(uint32 character_id);
|
uint32 GetAccountIDByChar(uint32 char_id);
|
||||||
const std::string GetCharNameByID(uint32 character_id);
|
uint32 GetAccountIDByName(const char* accname, const char *loginserver, int16* status = 0, uint32* lsid = 0);
|
||||||
const std::string GetNPCNameByID(uint32 npc_id);
|
uint32 GetCharacterID(const char *name);
|
||||||
const std::string GetCleanNPCNameByID(uint32 npc_id);
|
uint32 GetCharacterInfo(const char* iName, uint32* oAccID = 0, uint32* oZoneID = 0, uint32* oInstanceID = 0, float* oX = 0, float* oY = 0, float* oZ = 0);
|
||||||
void LoginIP(uint32 account_id, const std::string& login_ip);
|
uint32 GetGuildIDByCharID(uint32 char_id);
|
||||||
|
uint32 GetGroupIDByCharID(uint32 char_id);
|
||||||
|
uint32 GetRaidIDByCharID(uint32 char_id);
|
||||||
|
|
||||||
|
void GetAccountName(uint32 accountid, char* name, uint32* oLSAccountID = 0);
|
||||||
|
void GetCharName(uint32 char_id, char* name);
|
||||||
|
std::string GetCharNameByID(uint32 char_id);
|
||||||
|
std::string GetNPCNameByID(uint32 npc_id);
|
||||||
|
void LoginIP(uint32 AccountID, const char* LoginIP);
|
||||||
|
|
||||||
/* Instancing */
|
/* Instancing */
|
||||||
bool AddClientToInstance(uint16 instance_id, uint32 character_id);
|
|
||||||
bool CheckInstanceByCharID(uint16 instance_id, uint32 character_id);
|
bool AddClientToInstance(uint16 instance_id, uint32 char_id);
|
||||||
|
bool CharacterInInstanceGroup(uint16 instance_id, uint32 char_id);
|
||||||
bool CheckInstanceExists(uint16 instance_id);
|
bool CheckInstanceExists(uint16 instance_id);
|
||||||
bool CheckInstanceExpired(uint16 instance_id);
|
bool CheckInstanceExpired(uint16 instance_id);
|
||||||
bool CreateInstance(uint16 instance_id, uint32 zone_id, uint32 version, uint32 duration);
|
bool CreateInstance(uint16 instance_id, uint32 zone_id, uint32 version, uint32 duration);
|
||||||
bool GetUnusedInstanceID(uint16& instance_id);
|
bool GetUnusedInstanceID(uint16 &instance_id);
|
||||||
bool IsGlobalInstance(uint16 instance_id);
|
bool GlobalInstance(uint16 instance_id);
|
||||||
bool RemoveClientFromInstance(uint16 instance_id, uint32 char_id);
|
bool RemoveClientFromInstance(uint16 instance_id, uint32 char_id);
|
||||||
bool RemoveClientsFromInstance(uint16 instance_id);
|
bool RemoveClientsFromInstance(uint16 instance_id);
|
||||||
bool VerifyInstanceAlive(uint16 instance_id, uint32 character_id);
|
bool VerifyInstanceAlive(uint16 instance_id, uint32 char_id);
|
||||||
bool VerifyZoneInstance(uint32 zone_id, uint16 instance_id);
|
bool VerifyZoneInstance(uint32 zone_id, uint16 instance_id);
|
||||||
|
|
||||||
uint16 GetInstanceID(uint32 zone, uint32 character_id, int16 version);
|
uint16 GetInstanceID(uint32 zone, uint32 charid, int16 version);
|
||||||
std::vector<uint16> GetInstanceIDs(uint32 zone_id, uint32 character_id);
|
uint16 GetInstanceVersion(uint16 instance_id);
|
||||||
uint8_t GetInstanceVersion(uint16 instance_id);
|
uint32 GetTimeRemainingInstance(uint16 instance_id, bool &is_perma);
|
||||||
uint32 GetTimeRemainingInstance(uint16 instance_id, bool& is_perma);
|
uint32 VersionFromInstanceID(uint16 instance_id);
|
||||||
uint32 GetInstanceZoneID(uint16 instance_id);
|
uint32 ZoneIDFromInstanceID(uint16 instance_id);
|
||||||
|
|
||||||
void AssignGroupToInstance(uint32 group_id, uint32 instance_id);
|
void AssignGroupToInstance(uint32 gid, uint32 instance_id);
|
||||||
void AssignRaidToInstance(uint32 raid_id, uint32 instance_id);
|
void AssignRaidToInstance(uint32 rid, uint32 instance_id);
|
||||||
|
void BuryCorpsesInInstance(uint16 instance_id);
|
||||||
void DeleteInstance(uint16 instance_id);
|
void DeleteInstance(uint16 instance_id);
|
||||||
void FlagInstanceByGroupLeader(uint32 zone_id, int16 version, uint32 character_id, uint32 group_id);
|
void FlagInstanceByGroupLeader(uint32 zone, int16 version, uint32 charid, uint32 gid);
|
||||||
void FlagInstanceByRaidLeader(uint32 zone_id, int16 version, uint32 character_id, uint32 raid_id);
|
void FlagInstanceByRaidLeader(uint32 zone, int16 version, uint32 charid, uint32 rid);
|
||||||
void GetCharactersInInstance(uint16 instance_id, std::list<uint32>& character_ids);
|
void GetCharactersInInstance(uint16 instance_id, std::list<uint32> &charid_list);
|
||||||
void PurgeExpiredInstances();
|
void PurgeExpiredInstances();
|
||||||
void SetInstanceDuration(uint16 instance_id, uint32 new_duration);
|
void SetInstanceDuration(uint16 instance_id, uint32 new_duration);
|
||||||
void CleanupInstanceCorpses();
|
|
||||||
|
|
||||||
/* Adventure related. */
|
/* Adventure related. */
|
||||||
void UpdateAdventureStatsEntry(uint32 character_id, uint8 theme_id, bool is_win = false, bool is_remove = false);
|
|
||||||
bool GetAdventureStats(uint32 character_id, AdventureStats_Struct* as);
|
void UpdateAdventureStatsEntry(uint32 char_id, uint8 theme, bool win, bool remove = false);
|
||||||
|
bool GetAdventureStats(uint32 char_id, AdventureStats_Struct *as);
|
||||||
|
|
||||||
/* Account Related */
|
/* Account Related */
|
||||||
const std::string GetLiveChar(uint32 account_id);
|
|
||||||
bool SetAccountStatus(const std::string& account_name, int16 status);
|
|
||||||
bool SetLocalPassword(uint32 account_id, const std::string& password);
|
|
||||||
bool UpdateLiveChar(const std::string& name, uint32 account_id);
|
|
||||||
int16 GetAccountStatus(uint32 account_id);
|
|
||||||
void SetAccountCRCField(uint32 account_id, const std::string& field_name, uint64 checksum);
|
|
||||||
uint32 CheckLogin(const std::string& name, const std::string& password, const std::string& loginserver, int16* status = 0);
|
|
||||||
uint32 CreateAccount(
|
|
||||||
const std::string& name,
|
|
||||||
const std::string& password,
|
|
||||||
int16 status,
|
|
||||||
const std::string& loginserver,
|
|
||||||
uint32 lsaccount_id
|
|
||||||
);
|
|
||||||
uint32 GetAccountIDFromLSID(
|
|
||||||
const std::string& in_loginserver_id,
|
|
||||||
uint32 in_loginserver_account_id,
|
|
||||||
char* in_account_name = 0,
|
|
||||||
int16* in_status = 0
|
|
||||||
);
|
|
||||||
|
|
||||||
uint8 GetAgreementFlag(uint32 account_id);
|
bool DeleteAccount(const char *name, const char* loginserver);
|
||||||
void SetAgreementFlag(uint32 account_id);
|
bool GetLiveChar(uint32 account_id, char* cname);
|
||||||
|
bool SetAccountStatus(const char* name, int16 status);
|
||||||
|
bool SetAccountStatus(const std::string& account_name, int16 status);
|
||||||
|
bool SetLocalPassword(uint32 accid, const char* password);
|
||||||
|
bool UpdateLiveChar(char* charname, uint32 account_id);
|
||||||
|
|
||||||
int GetIPExemption(const std::string& account_ip);
|
int16 CheckStatus(uint32 account_id);
|
||||||
void SetIPExemption(const std::string& account_ip, int exemption_amount);
|
|
||||||
|
|
||||||
int GetInstanceID(uint32 character_id, uint32 zone_id);
|
uint32 CheckLogin(const char* name, const char* password, const char *loginserver, int16* oStatus = 0);
|
||||||
|
uint32 CreateAccount(const char* name, const char* password, int16 status, const char* loginserver, uint32 lsaccount_id);
|
||||||
|
uint32 GetAccountIDFromLSID(const std::string& in_loginserver_id, uint32 in_loginserver_account_id, char* in_account_name = 0, int16* in_status = 0);
|
||||||
|
uint8 GetAgreementFlag(uint32 acctid);
|
||||||
|
|
||||||
|
void GetAccountFromID(uint32 id, char* oAccountName, int16* oStatus);
|
||||||
|
void SetAgreementFlag(uint32 acctid);
|
||||||
|
|
||||||
|
int GetIPExemption(std::string account_ip);
|
||||||
|
|
||||||
|
int GetInstanceID(uint32 char_id, uint32 zone_id);
|
||||||
|
|
||||||
|
|
||||||
/* Groups */
|
/* Groups */
|
||||||
std::string GetGroupLeaderForLogin(const std::string& character_name);
|
|
||||||
char* GetGroupLeadershipInfo(
|
char* GetGroupLeaderForLogin(const char* name,char* leaderbuf);
|
||||||
uint32 group_id,
|
char* GetGroupLeadershipInfo(uint32 gid, char* leaderbuf, char* maintank = nullptr, char* assist = nullptr, char* puller = nullptr, char *marknpc = nullptr, char *mentoree = nullptr, int *mentor_percent = nullptr, GroupLeadershipAA_Struct* GLAA = nullptr);
|
||||||
char* leaderbuf,
|
|
||||||
char* maintank = nullptr,
|
uint32 GetGroupID(const char* name);
|
||||||
char* assist = nullptr,
|
|
||||||
char* puller = nullptr,
|
void ClearGroup(uint32 gid = 0);
|
||||||
char* marknpc = nullptr,
|
void ClearGroupLeader(uint32 gid = 0);
|
||||||
char* mentoree = nullptr,
|
void SetGroupID(const char* name, uint32 id, uint32 charid, uint32 ismerc = false);
|
||||||
int* mentor_percent = nullptr,
|
void SetGroupLeaderName(uint32 gid, const char* name);
|
||||||
GroupLeadershipAA_Struct* GLAA = nullptr
|
|
||||||
);
|
|
||||||
std::string GetGroupLeaderName(uint32 group_id);
|
|
||||||
uint32 GetGroupID(const std::string& name);
|
|
||||||
void ClearGroup(uint32 group_id = 0);
|
|
||||||
void ClearGroupLeader(uint32 group_id = 0);
|
|
||||||
void SetGroupLeaderName(uint32 group_id, const std::string& name);
|
|
||||||
|
|
||||||
/* Raids */
|
/* Raids */
|
||||||
const std::string GetRaidLeaderName(uint32 raid_id);
|
|
||||||
uint32 GetRaidID(const std::string& name);
|
|
||||||
void ClearRaid(uint32 raid_id = 0);
|
|
||||||
void ClearRaidDetails(uint32 raid_id = 0);
|
|
||||||
void ClearRaidLeader(uint32 group_id = std::numeric_limits<uint32>::max(), uint32 raid_id = 0);
|
|
||||||
void GetGroupLeadershipInfo(
|
|
||||||
uint32 group_id,
|
|
||||||
uint32 raid_id,
|
|
||||||
char* maintank = nullptr,
|
|
||||||
char* assist = nullptr,
|
|
||||||
char* puller = nullptr,
|
|
||||||
char* marknpc = nullptr,
|
|
||||||
char* mentoree = nullptr,
|
|
||||||
int* mentor_percent = nullptr,
|
|
||||||
GroupLeadershipAA_Struct* GLAA = nullptr
|
|
||||||
);
|
|
||||||
void GetRaidLeadershipInfo(
|
|
||||||
uint32 raid_id,
|
|
||||||
char* maintank = nullptr,
|
|
||||||
char* assist = nullptr,
|
|
||||||
char* puller = nullptr,
|
|
||||||
char* marknpc = nullptr,
|
|
||||||
RaidLeadershipAA_Struct* RLAA = nullptr
|
|
||||||
);
|
|
||||||
void SetRaidGroupLeaderInfo(uint32 group_id, uint32 raid_id);
|
|
||||||
|
|
||||||
void PurgeAllDeletedDataBuckets();
|
const char *GetRaidLeaderName(uint32 rid);
|
||||||
void ClearGuildOnlineStatus();
|
|
||||||
void ClearTraderDetails();
|
|
||||||
void ClearBuyerDetails();
|
|
||||||
|
|
||||||
|
uint32 GetRaidID(const char* name);
|
||||||
|
|
||||||
|
void ClearRaid(uint32 rid = 0);
|
||||||
|
void ClearRaidDetails(uint32 rid = 0);
|
||||||
|
void ClearRaidLeader(uint32 gid = 0xFFFFFFFF, uint32 rid = 0);
|
||||||
|
void GetGroupLeadershipInfo(uint32 gid, uint32 rid, char* maintank = nullptr, char* assist = nullptr, char* puller = nullptr, char *marknpc = nullptr, char *mentoree = nullptr, int *mentor_percent = nullptr, GroupLeadershipAA_Struct* GLAA = nullptr);
|
||||||
|
void GetRaidLeadershipInfo(uint32 rid, char* maintank = nullptr, char* assist = nullptr, char* puller = nullptr, char *marknpc = nullptr, RaidLeadershipAA_Struct* RLAA = nullptr);
|
||||||
|
void SetRaidGroupLeaderInfo(uint32 gid, uint32 rid);
|
||||||
|
|
||||||
|
void PurgeAllDeletedDataBuckets();
|
||||||
|
|
||||||
|
/* Database Conversions 'database_conversions.cpp' */
|
||||||
|
|
||||||
|
bool CheckDatabaseConversions();
|
||||||
|
bool CheckDatabaseConvertCorpseDeblob();
|
||||||
|
bool CheckDatabaseConvertPPDeblob();
|
||||||
|
|
||||||
/* Database Variables */
|
/* Database Variables */
|
||||||
bool GetVariable(const std::string& name, std::string& value);
|
|
||||||
bool SetVariable(const std::string& name, const std::string& value);
|
|
||||||
bool LoadVariables();
|
|
||||||
|
|
||||||
uint8 GetPEQZone(uint32 zone_id, uint32 version);
|
bool GetVariable(std::string varname, std::string &varvalue);
|
||||||
uint32 GetServerType();
|
bool SetVariable(const std::string varname, const std::string &varvalue);
|
||||||
void AddReport(const std::string& who, const std::string& against, const std::string& lines);
|
bool LoadVariables();
|
||||||
struct TimeOfDay_Struct LoadTime(time_t& realtime);
|
|
||||||
bool SaveTime(int8 minute, int8 hour, int8 day, int8 month, int16 year);
|
|
||||||
void ClearMerchantTemp();
|
|
||||||
void ClearPTimers(uint32 character_id);
|
|
||||||
void SetFirstLogon(uint32 character_id, uint8 first_logon);
|
|
||||||
void SetLFG(uint32 character_id, bool is_lfg);
|
|
||||||
void SetLFP(uint32 character_id, bool is_lfp);
|
|
||||||
void SetLoginFlags(uint32 character_id, bool is_lfp, bool is_lfg, uint8 first_logon);
|
|
||||||
|
|
||||||
int64 CountInvSnapshots();
|
/* General Queries */
|
||||||
void ClearInvSnapshots(bool from_now = false);
|
|
||||||
|
|
||||||
void SourceDatabaseTableFromUrl(const std::string& table_name, const std::string& url);
|
bool GetSafePoints(const char* zone_short_name, uint32 instance_version, float* safe_x = 0, float* safe_y = 0, float* safe_z = 0, float* safe_heading = 0, int16* minstatus = 0, uint8* minlevel = 0, char *flag_needed = nullptr);
|
||||||
void SourceSqlFromUrl(const std::string& url);
|
bool GetZoneGraveyard(const uint32 graveyard_id, uint32* graveyard_zoneid = 0, float* graveyard_x = 0, float* graveyard_y = 0, float* graveyard_z = 0, float* graveyard_heading = 0);
|
||||||
void PurgeCharacterParcels();
|
bool GetZoneLongName(const char* short_name, char** long_name, char* file_name = 0, float* safe_x = 0, float* safe_y = 0, float* safe_z = 0, uint32* graveyard_id = 0, uint32* maxclients = 0);
|
||||||
void Encode(std::string &in);
|
bool LoadPTimers(uint32 charid, PTimerList &into);
|
||||||
void Decode(std::string &in);
|
|
||||||
|
uint32 GetZoneGraveyardID(uint32 zone_id, uint32 version);
|
||||||
|
|
||||||
|
uint8 GetPEQZone(uint32 zoneID, uint32 version);
|
||||||
|
uint8 GetRaceSkill(uint8 skillid, uint8 in_race);
|
||||||
|
uint8 GetServerType();
|
||||||
|
uint8 GetSkillCap(uint8 skillid, uint8 in_race, uint8 in_class, uint16 in_level);
|
||||||
|
|
||||||
|
void AddReport(std::string who, std::string against, std::string lines);
|
||||||
|
struct TimeOfDay_Struct LoadTime(time_t &realtime);
|
||||||
|
bool SaveTime(int8 minute, int8 hour, int8 day, int8 month, int16 year);
|
||||||
|
void ClearMerchantTemp();
|
||||||
|
void ClearPTimers(uint32 charid);
|
||||||
|
void SetFirstLogon(uint32 CharID, uint8 firstlogon);
|
||||||
|
void SetLFG(uint32 CharID, bool LFG);
|
||||||
|
void SetLFP(uint32 CharID, bool LFP);
|
||||||
|
void SetLoginFlags(uint32 CharID, bool LFP, bool LFG, uint8 firstlogon);
|
||||||
|
|
||||||
|
int CountInvSnapshots();
|
||||||
|
void ClearInvSnapshots(bool from_now = false);
|
||||||
|
|
||||||
|
/* EQEmuLogSys */
|
||||||
|
void LoadLogSettings(EQEmuLogSys::LogSettings* log_settings);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Mutex Mvarcache;
|
|
||||||
|
Mutex Mvarcache;
|
||||||
VarCache_Struct varcache;
|
VarCache_Struct varcache;
|
||||||
|
|
||||||
/* Groups, utility methods. */
|
/* Groups, utility methods. */
|
||||||
void ClearAllGroupLeaders();
|
void ClearAllGroupLeaders();
|
||||||
void ClearAllGroups();
|
void ClearAllGroups();
|
||||||
|
|
||||||
/* Raid, utility methods. */
|
/* Raid, utility methods. */
|
||||||
void ClearAllRaids();
|
void ClearAllRaids();
|
||||||
|
|||||||
@@ -23,12 +23,10 @@
|
|||||||
#include <iterator>
|
#include <iterator>
|
||||||
#include "database_dump_service.h"
|
#include "database_dump_service.h"
|
||||||
#include "../eqemu_logsys.h"
|
#include "../eqemu_logsys.h"
|
||||||
#include "../strings.h"
|
#include "../string_util.h"
|
||||||
#include "../eqemu_config.h"
|
#include "../eqemu_config.h"
|
||||||
#include "../database_schema.h"
|
#include "../database_schema.h"
|
||||||
#include "../file.h"
|
#include "../file_util.h"
|
||||||
#include "../process/process.h"
|
|
||||||
#include "../termcolor/rang.hpp"
|
|
||||||
|
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
|
|
||||||
@@ -37,12 +35,43 @@
|
|||||||
#else
|
#else
|
||||||
|
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define DATABASE_DUMP_PATH "backups/"
|
#define DATABASE_DUMP_PATH "backups/"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param cmd
|
||||||
|
* @param return_result
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
std::string DatabaseDumpService::execute(const std::string &cmd, bool return_result = true)
|
||||||
|
{
|
||||||
|
const char *file_name = "db-exec-result.txt";
|
||||||
|
|
||||||
|
if (return_result) {
|
||||||
|
#ifdef _WINDOWS
|
||||||
|
std::system((cmd + " > " + file_name + " 2>&1").c_str());
|
||||||
|
#else
|
||||||
|
std::system((cmd + " > " + file_name).c_str());
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
std::system((cmd).c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string result;
|
||||||
|
|
||||||
|
if (return_result) {
|
||||||
|
std::ifstream file(file_name);
|
||||||
|
result = {std::istreambuf_iterator<char>(file), std::istreambuf_iterator<char>()};
|
||||||
|
std::remove(file_name);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return bool
|
* @return bool
|
||||||
*/
|
*/
|
||||||
@@ -59,7 +88,7 @@ bool DatabaseDumpService::IsMySQLInstalled()
|
|||||||
*/
|
*/
|
||||||
bool DatabaseDumpService::IsTarAvailable()
|
bool DatabaseDumpService::IsTarAvailable()
|
||||||
{
|
{
|
||||||
std::string version_output = Process::execute("tar --version");
|
std::string version_output = execute("tar --version");
|
||||||
|
|
||||||
return version_output.find("GNU tar") != std::string::npos;
|
return version_output.find("GNU tar") != std::string::npos;
|
||||||
}
|
}
|
||||||
@@ -70,7 +99,7 @@ bool DatabaseDumpService::IsTarAvailable()
|
|||||||
*/
|
*/
|
||||||
bool DatabaseDumpService::Is7ZipAvailable()
|
bool DatabaseDumpService::Is7ZipAvailable()
|
||||||
{
|
{
|
||||||
std::string version_output = Process::execute("7z --help");
|
std::string version_output = execute("7z --help");
|
||||||
|
|
||||||
return version_output.find("7-Zip") != std::string::npos;
|
return version_output.find("7-Zip") != std::string::npos;
|
||||||
}
|
}
|
||||||
@@ -88,13 +117,11 @@ bool DatabaseDumpService::HasCompressionBinary()
|
|||||||
*/
|
*/
|
||||||
std::string DatabaseDumpService::GetMySQLVersion()
|
std::string DatabaseDumpService::GetMySQLVersion()
|
||||||
{
|
{
|
||||||
std::string version_output = Process::execute("mysql --version");
|
std::string version_output = execute("mysql --version");
|
||||||
|
|
||||||
return Strings::Trim(version_output);
|
return trim(version_output);
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string CREDENTIALS_FILE = "login.my.cnf";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
@@ -103,66 +130,114 @@ std::string DatabaseDumpService::GetBaseMySQLDumpCommand()
|
|||||||
auto config = EQEmuConfig::get();
|
auto config = EQEmuConfig::get();
|
||||||
if (IsDumpContentTables() && !config->ContentDbHost.empty()) {
|
if (IsDumpContentTables() && !config->ContentDbHost.empty()) {
|
||||||
return fmt::format(
|
return fmt::format(
|
||||||
"mysqldump --defaults-extra-file={} {}",
|
"mysqldump -u {} -p{} -h {} --port={} {}",
|
||||||
CREDENTIALS_FILE,
|
config->ContentDbUsername,
|
||||||
|
config->ContentDbPassword,
|
||||||
|
config->ContentDbHost,
|
||||||
|
config->ContentDbPort,
|
||||||
config->ContentDbName
|
config->ContentDbName
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return fmt::format(
|
return fmt::format(
|
||||||
"mysqldump --defaults-extra-file={} {}",
|
"mysqldump -u {} -p{} -h {} --port={} {}",
|
||||||
CREDENTIALS_FILE,
|
config->DatabaseUsername,
|
||||||
|
config->DatabasePassword,
|
||||||
|
config->DatabaseHost,
|
||||||
|
config->DatabasePort,
|
||||||
config->DatabaseDB
|
config->DatabaseDB
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetPlayerTablesList()
|
std::string DatabaseDumpService::GetPlayerTablesList()
|
||||||
{
|
{
|
||||||
return Strings::Join(DatabaseSchema::GetPlayerTables(), " ");
|
std::string tables_list;
|
||||||
}
|
std::vector<std::string> tables = DatabaseSchema::GetPlayerTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
std::string DatabaseDumpService::GetBotTablesList()
|
tables_list += table + " ";
|
||||||
{
|
}
|
||||||
return Strings::Join(DatabaseSchema::GetBotTables(), " ");
|
|
||||||
}
|
return trim(tables_list);
|
||||||
|
|
||||||
std::string DatabaseDumpService::GetMercTablesList()
|
|
||||||
{
|
|
||||||
return Strings::Join(DatabaseSchema::GetMercTables(), " ");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetLoginTableList()
|
std::string DatabaseDumpService::GetLoginTableList()
|
||||||
{
|
{
|
||||||
return Strings::Join(DatabaseSchema::GetLoginTables(), " ");
|
std::string tables_list;
|
||||||
|
std::vector<std::string> tables = DatabaseSchema::GetLoginTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
|
tables_list += table + " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(tables_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetQueryServTables()
|
std::string DatabaseDumpService::GetQueryServTables()
|
||||||
{
|
{
|
||||||
return Strings::Join(DatabaseSchema::GetQueryServerTables(), " ");
|
std::string tables_list;
|
||||||
|
std::vector<std::string> tables = DatabaseSchema::GetQueryServerTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
|
tables_list += table + " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(tables_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetSystemTablesList()
|
std::string DatabaseDumpService::GetSystemTablesList()
|
||||||
{
|
{
|
||||||
auto system_tables = DatabaseSchema::GetServerTables();
|
std::string tables_list;
|
||||||
auto version_tables = DatabaseSchema::GetVersionTables();
|
|
||||||
|
|
||||||
system_tables.insert(
|
std::vector<std::string> tables = DatabaseSchema::GetServerTables();
|
||||||
std::end(system_tables),
|
for (const auto &table : tables) {
|
||||||
std::begin(version_tables),
|
tables_list += table + " ";
|
||||||
std::end(version_tables)
|
}
|
||||||
);
|
|
||||||
|
|
||||||
return Strings::Join(system_tables, " ");
|
tables = DatabaseSchema::GetVersionTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
|
tables_list += table + " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(tables_list);
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetStateTablesList()
|
std::string DatabaseDumpService::GetStateTablesList()
|
||||||
{
|
{
|
||||||
return Strings::Join(DatabaseSchema::GetStateTables(), " ");
|
std::string tables_list;
|
||||||
|
|
||||||
|
std::vector<std::string> tables = DatabaseSchema::GetStateTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
|
tables_list += table + " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(tables_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
std::string DatabaseDumpService::GetContentTablesList()
|
std::string DatabaseDumpService::GetContentTablesList()
|
||||||
{
|
{
|
||||||
return Strings::Join(DatabaseSchema::GetContentTables(), " ");
|
std::string tables_list;
|
||||||
|
|
||||||
|
std::vector<std::string> tables = DatabaseSchema::GetContentTables();
|
||||||
|
for (const auto &table : tables) {
|
||||||
|
tables_list += table + " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(tables_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -197,7 +272,7 @@ std::string DatabaseDumpService::GetDumpFileNameWithPath()
|
|||||||
return GetSetDumpPath() + GetDumpFileName();
|
return GetSetDumpPath() + GetDumpFileName();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DatabaseDumpService::DatabaseDump()
|
void DatabaseDumpService::Dump()
|
||||||
{
|
{
|
||||||
if (!IsMySQLInstalled()) {
|
if (!IsMySQLInstalled()) {
|
||||||
LogError("MySQL is not installed; Please check your PATH for a valid MySQL installation");
|
LogError("MySQL is not installed; Please check your PATH for a valid MySQL installation");
|
||||||
@@ -243,16 +318,6 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
dump_descriptor += "-player";
|
dump_descriptor += "-player";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IsDumpBotTables()) {
|
|
||||||
tables_to_dump += GetBotTablesList() + " ";
|
|
||||||
dump_descriptor += "-bots";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (IsDumpMercTables()) {
|
|
||||||
tables_to_dump += GetMercTablesList() + " ";
|
|
||||||
dump_descriptor += "-mercs";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (IsDumpSystemTables()) {
|
if (IsDumpSystemTables()) {
|
||||||
tables_to_dump += GetSystemTablesList() + " ";
|
tables_to_dump += GetSystemTablesList() + " ";
|
||||||
dump_descriptor += "-system";
|
dump_descriptor += "-system";
|
||||||
@@ -279,11 +344,6 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IsDumpStaticInstanceData()) {
|
|
||||||
tables_to_dump += "instance_list";
|
|
||||||
options += " --no-create-info --where=\"instance_list.is_global > 0 and instance_list.never_expires > 0\"";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!dump_descriptor.empty()) {
|
if (!dump_descriptor.empty()) {
|
||||||
SetDumpFileName(GetDumpFileName() + dump_descriptor);
|
SetDumpFileName(GetDumpFileName() + dump_descriptor);
|
||||||
}
|
}
|
||||||
@@ -296,76 +356,48 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
pipe_file = fmt::format(" > {}.sql", GetDumpFileNameWithPath());
|
pipe_file = fmt::format(" > {}.sql", GetDumpFileNameWithPath());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!File::Exists(GetSetDumpPath()) && !IsDumpOutputToConsole()) {
|
std::string execute_command = fmt::format(
|
||||||
File::Makedir(GetSetDumpPath());
|
"{} {} {} {}",
|
||||||
|
GetBaseMySQLDumpCommand(),
|
||||||
|
options,
|
||||||
|
tables_to_dump,
|
||||||
|
pipe_file
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!FileUtil::exists(GetSetDumpPath()) && !IsDumpOutputToConsole()) {
|
||||||
|
FileUtil::mkdir(GetSetDumpPath());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IsDumpDropTableSyntaxOnly()) {
|
if (IsDumpDropTableSyntaxOnly()) {
|
||||||
std::vector<std::string> tables = Strings::Split(tables_to_dump, ' ');
|
std::vector<std::string> tables = SplitString(tables_to_dump, ' ');
|
||||||
|
|
||||||
for (auto &table: tables) {
|
for (auto &table : tables) {
|
||||||
std::cout << "DROP TABLE IF EXISTS `" << table << "`;" << std::endl;
|
std::cout << "DROP TABLE IF EXISTS `" << table << "`;" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tables_to_dump.empty()) {
|
if (tables_to_dump.empty()) {
|
||||||
std::cerr << "No tables were specified" << std::endl;
|
std::cerr << "No tables were specified" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
const auto execute_command = fmt::format(
|
std::string execution_result = execute(execute_command, IsDumpOutputToConsole());
|
||||||
"{} {} {} {}",
|
if (!execution_result.empty()) {
|
||||||
GetBaseMySQLDumpCommand(),
|
|
||||||
options,
|
|
||||||
tables_to_dump,
|
|
||||||
pipe_file
|
|
||||||
);
|
|
||||||
|
|
||||||
LogInfo("Backing up database [{}]", execute_command);
|
|
||||||
LogInfo("This can take a few minutes depending on the size of your database");
|
|
||||||
LogInfo("LOADING... PLEASE WAIT...");
|
|
||||||
|
|
||||||
BuildCredentialsFile();
|
|
||||||
std::string execution_result = Process::execute(execute_command);
|
|
||||||
if (!execution_result.empty() && IsDumpOutputToConsole()) {
|
|
||||||
std::cout << execution_result;
|
std::cout << execution_result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!IsDumpOutputToConsole()) {
|
|
||||||
LogSys.LoadLogSettingsDefaults();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!pipe_file.empty()) {
|
|
||||||
std::string file = fmt::format("{}.sql", GetDumpFileNameWithPath());
|
|
||||||
auto r = File::GetContents(file);
|
|
||||||
if (!r.error.empty()) {
|
|
||||||
LogError("{}", r.error);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &line: Strings::Split(r.contents, "\n")) {
|
|
||||||
if (Strings::Contains(line, "mysqldump:")) {
|
|
||||||
LogError("{}", line);
|
|
||||||
LogError("Database dump failed. Correct the error before continuing or trying again");
|
|
||||||
LogError("This is to prevent data loss on behalf of the server operator");
|
|
||||||
RemoveSqlBackup();
|
|
||||||
std::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!tables_to_dump.empty()) {
|
if (!tables_to_dump.empty()) {
|
||||||
LogInfo("Dumping Tables [{}]", Strings::Trim(tables_to_dump));
|
LogInfo("Dumping Tables [{}]", tables_to_dump);
|
||||||
}
|
}
|
||||||
|
|
||||||
LogInfo("Database dump created at [{}.sql]", GetDumpFileNameWithPath());
|
LogInfo("Database dump created at [{}.sql]", GetDumpFileNameWithPath());
|
||||||
|
|
||||||
if (IsDumpWithCompression() && !IsDumpOutputToConsole()) {
|
if (IsDumpWithCompression() && !IsDumpOutputToConsole()) {
|
||||||
if (HasCompressionBinary()) {
|
if (HasCompressionBinary()) {
|
||||||
LogInfo("Compression requested. Compressing dump [{}.sql]", GetDumpFileNameWithPath());
|
LogInfo("Compression requested... Compressing dump [{}.sql]", GetDumpFileNameWithPath());
|
||||||
|
|
||||||
if (IsTarAvailable()) {
|
if (IsTarAvailable()) {
|
||||||
Process::execute(
|
execute(
|
||||||
fmt::format(
|
fmt::format(
|
||||||
"tar -zcvf {}.tar.gz -C {} {}.sql",
|
"tar -zcvf {}.tar.gz -C {} {}.sql",
|
||||||
GetDumpFileNameWithPath(),
|
GetDumpFileNameWithPath(),
|
||||||
@@ -374,10 +406,9 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
LogInfo("Compressed dump created at [{}.tar.gz]", GetDumpFileNameWithPath());
|
LogInfo("Compressed dump created at [{}.tar.gz]", GetDumpFileNameWithPath());
|
||||||
RemoveSqlBackup();
|
|
||||||
}
|
}
|
||||||
else if (Is7ZipAvailable()) {
|
else if (Is7ZipAvailable()) {
|
||||||
Process::execute(
|
execute(
|
||||||
fmt::format(
|
fmt::format(
|
||||||
"7z a -t7z {}.zip {}.sql",
|
"7z a -t7z {}.zip {}.sql",
|
||||||
GetDumpFileNameWithPath(),
|
GetDumpFileNameWithPath(),
|
||||||
@@ -385,7 +416,6 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
LogInfo("Compressed dump created at [{}.zip]", GetDumpFileNameWithPath());
|
LogInfo("Compressed dump created at [{}.zip]", GetDumpFileNameWithPath());
|
||||||
RemoveSqlBackup();
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
LogInfo("Compression requested, but no available compression binary was found");
|
LogInfo("Compression requested, but no available compression binary was found");
|
||||||
@@ -396,8 +426,6 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
RemoveCredentialsFile();
|
|
||||||
|
|
||||||
// LogDebug("[{}] dump-to-console", IsDumpOutputToConsole());
|
// LogDebug("[{}] dump-to-console", IsDumpOutputToConsole());
|
||||||
// LogDebug("[{}] dump-path", GetSetDumpPath());
|
// LogDebug("[{}] dump-path", GetSetDumpPath());
|
||||||
// LogDebug("[{}] compression", (IsDumpWithCompression() ? "true" : "false"));
|
// LogDebug("[{}] compression", (IsDumpWithCompression() ? "true" : "false"));
|
||||||
@@ -408,7 +436,6 @@ void DatabaseDumpService::DatabaseDump()
|
|||||||
// LogDebug("[{}] login", (IsDumpLoginServerTables() ? "true" : "false"));
|
// LogDebug("[{}] login", (IsDumpLoginServerTables() ? "true" : "false"));
|
||||||
// LogDebug("[{}] player", (IsDumpPlayerTables() ? "true" : "false"));
|
// LogDebug("[{}] player", (IsDumpPlayerTables() ? "true" : "false"));
|
||||||
// LogDebug("[{}] system", (IsDumpSystemTables() ? "true" : "false"));
|
// LogDebug("[{}] system", (IsDumpSystemTables() ? "true" : "false"));
|
||||||
// LogDebug("[{}] bot", (IsDumpBotTables() ? "true" : "false"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DatabaseDumpService::IsDumpSystemTables() const
|
bool DatabaseDumpService::IsDumpSystemTables() const
|
||||||
@@ -550,83 +577,3 @@ void DatabaseDumpService::SetDumpStateTables(bool dump_state_tables)
|
|||||||
{
|
{
|
||||||
DatabaseDumpService::dump_state_tables = dump_state_tables;
|
DatabaseDumpService::dump_state_tables = dump_state_tables;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DatabaseDumpService::IsDumpBotTables() const
|
|
||||||
{
|
|
||||||
return dump_bot_tables;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::SetDumpBotTables(bool dump_bot_tables)
|
|
||||||
{
|
|
||||||
DatabaseDumpService::dump_bot_tables = dump_bot_tables;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DatabaseDumpService::IsDumpMercTables() const
|
|
||||||
{
|
|
||||||
return dump_merc_tables;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::SetDumpMercTables(bool dump_merc_tables)
|
|
||||||
{
|
|
||||||
DatabaseDumpService::dump_merc_tables = dump_merc_tables;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::RemoveSqlBackup()
|
|
||||||
{
|
|
||||||
std::string file = fmt::format("{}.sql", GetDumpFileNameWithPath());
|
|
||||||
if (File::Exists(file)) {
|
|
||||||
try {
|
|
||||||
std::filesystem::remove(file);
|
|
||||||
}
|
|
||||||
catch (std::exception &e) {
|
|
||||||
LogError("std::filesystem::remove err [{}]", e.what());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
RemoveCredentialsFile();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::BuildCredentialsFile()
|
|
||||||
{
|
|
||||||
auto config = EQEmuConfig::get();
|
|
||||||
std::ofstream out(CREDENTIALS_FILE);
|
|
||||||
if (out.is_open()) {
|
|
||||||
if (IsDumpContentTables() && !config->ContentDbHost.empty()) {
|
|
||||||
out << "[mysqldump]" << std::endl;
|
|
||||||
out << "user=" << config->ContentDbUsername << std::endl;
|
|
||||||
out << "password=" << config->ContentDbPassword << std::endl;
|
|
||||||
out << "host=" << config->ContentDbHost << std::endl;
|
|
||||||
out << "port=" << config->ContentDbPort << std::endl;
|
|
||||||
out << "default-character-set=utf8" << std::endl;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
out << "[mysqldump]" << std::endl;
|
|
||||||
out << "user=" << config->DatabaseUsername << std::endl;
|
|
||||||
out << "password=" << config->DatabasePassword << std::endl;
|
|
||||||
out << "host=" << config->DatabaseHost << std::endl;
|
|
||||||
out << "port=" << config->DatabasePort << std::endl;
|
|
||||||
out << "default-character-set=utf8" << std::endl;
|
|
||||||
}
|
|
||||||
out.close();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
LogError("Failed to open credentials file for writing");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::RemoveCredentialsFile()
|
|
||||||
{
|
|
||||||
if (File::Exists(CREDENTIALS_FILE)) {
|
|
||||||
std::filesystem::remove(CREDENTIALS_FILE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DatabaseDumpService::IsDumpStaticInstanceData()
|
|
||||||
{
|
|
||||||
return dump_static_instance_data;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseDumpService::SetDumpStaticInstanceData(bool b)
|
|
||||||
{
|
|
||||||
dump_static_instance_data = b;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -24,7 +24,7 @@
|
|||||||
|
|
||||||
class DatabaseDumpService {
|
class DatabaseDumpService {
|
||||||
public:
|
public:
|
||||||
void DatabaseDump();
|
void Dump();
|
||||||
bool IsDumpAllTables() const;
|
bool IsDumpAllTables() const;
|
||||||
void SetDumpAllTables(bool dump_all_tables);
|
void SetDumpAllTables(bool dump_all_tables);
|
||||||
bool IsDumpWithNoData() const;
|
bool IsDumpWithNoData() const;
|
||||||
@@ -53,13 +53,6 @@ public:
|
|||||||
void SetDumpDropTableSyntaxOnly(bool dump_drop_table_syntax_only);
|
void SetDumpDropTableSyntaxOnly(bool dump_drop_table_syntax_only);
|
||||||
bool IsDumpStateTables() const;
|
bool IsDumpStateTables() const;
|
||||||
void SetDumpStateTables(bool dump_state_tables);
|
void SetDumpStateTables(bool dump_state_tables);
|
||||||
bool IsDumpBotTables() const;
|
|
||||||
void SetDumpBotTables(bool dump_bot_tables);
|
|
||||||
bool IsDumpMercTables() const;
|
|
||||||
void SetDumpMercTables(bool dump_bot_tables);
|
|
||||||
|
|
||||||
void SetDumpStaticInstanceData(bool b);
|
|
||||||
bool IsDumpStaticInstanceData();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool dump_all_tables = false;
|
bool dump_all_tables = false;
|
||||||
@@ -74,19 +67,14 @@ private:
|
|||||||
bool dump_with_compression = false;
|
bool dump_with_compression = false;
|
||||||
bool dump_output_to_console = false;
|
bool dump_output_to_console = false;
|
||||||
bool dump_drop_table_syntax_only = false;
|
bool dump_drop_table_syntax_only = false;
|
||||||
bool dump_bot_tables = false;
|
|
||||||
bool dump_merc_tables = false;
|
|
||||||
bool dump_static_instance_data = false;
|
|
||||||
|
|
||||||
std::string dump_path;
|
std::string dump_path;
|
||||||
std::string dump_file_name;
|
std::string dump_file_name;
|
||||||
|
|
||||||
|
std::string execute(const std::string &cmd, bool return_result);
|
||||||
bool IsMySQLInstalled();
|
bool IsMySQLInstalled();
|
||||||
std::string GetMySQLVersion();
|
std::string GetMySQLVersion();
|
||||||
std::string GetBaseMySQLDumpCommand();
|
std::string GetBaseMySQLDumpCommand();
|
||||||
std::string GetPlayerTablesList();
|
std::string GetPlayerTablesList();
|
||||||
std::string GetBotTablesList();
|
|
||||||
std::string GetMercTablesList();
|
|
||||||
std::string GetSystemTablesList();
|
std::string GetSystemTablesList();
|
||||||
std::string GetStateTablesList();
|
std::string GetStateTablesList();
|
||||||
std::string GetContentTablesList();
|
std::string GetContentTablesList();
|
||||||
@@ -97,9 +85,6 @@ private:
|
|||||||
std::string GetDumpFileNameWithPath();
|
std::string GetDumpFileNameWithPath();
|
||||||
std::string GetSetDumpPath();
|
std::string GetSetDumpPath();
|
||||||
std::string GetQueryServTables();
|
std::string GetQueryServTables();
|
||||||
void RemoveSqlBackup();
|
|
||||||
void BuildCredentialsFile();
|
|
||||||
void RemoveCredentialsFile();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,334 +0,0 @@
|
|||||||
#include <filesystem>
|
|
||||||
#include "database_update.h"
|
|
||||||
#include "../eqemu_logsys.h"
|
|
||||||
#include "../database.h"
|
|
||||||
#include "../strings.h"
|
|
||||||
#include "../rulesys.h"
|
|
||||||
#include "../http/httplib.h"
|
|
||||||
|
|
||||||
#include "database_update_manifest.cpp"
|
|
||||||
#include "database_update_manifest_bots.cpp"
|
|
||||||
#include "database_dump_service.h"
|
|
||||||
|
|
||||||
constexpr int BREAK_LENGTH = 70;
|
|
||||||
|
|
||||||
DatabaseVersion DatabaseUpdate::GetDatabaseVersions()
|
|
||||||
{
|
|
||||||
auto results = m_database->QueryDatabase("SELECT `version`, `bots_version` FROM `db_version` LIMIT 1");
|
|
||||||
if (!results.Success() || !results.RowCount()) {
|
|
||||||
LogError("Failed to read from [db_version] table!");
|
|
||||||
return DatabaseVersion{};
|
|
||||||
}
|
|
||||||
|
|
||||||
auto r = results.begin();
|
|
||||||
|
|
||||||
return DatabaseVersion{
|
|
||||||
.server_database_version = Strings::ToInt(r[0]),
|
|
||||||
.bots_database_version = Strings::ToInt(r[1]),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
DatabaseVersion DatabaseUpdate::GetBinaryDatabaseVersions()
|
|
||||||
{
|
|
||||||
return DatabaseVersion{
|
|
||||||
.server_database_version = CURRENT_BINARY_DATABASE_VERSION,
|
|
||||||
.bots_database_version = (RuleB(Bots, Enabled) ? CURRENT_BINARY_BOTS_DATABASE_VERSION : 0),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// the amount of versions we look-back to ensure we have all migrations
|
|
||||||
// we may not want to force these, but just warn about the look-backs
|
|
||||||
constexpr int LOOK_BACK_AMOUNT = 10;
|
|
||||||
|
|
||||||
// this check will take action
|
|
||||||
void DatabaseUpdate::CheckDbUpdates()
|
|
||||||
{
|
|
||||||
InjectBotsVersionColumn();
|
|
||||||
auto v = GetDatabaseVersions();
|
|
||||||
auto b = GetBinaryDatabaseVersions();
|
|
||||||
if (CheckVersionsUpToDate(v, b)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (UpdateManifest(manifest_entries, v.server_database_version, b.server_database_version)) {
|
|
||||||
LogInfo(
|
|
||||||
"Updates ran successfully, setting database version to [{}] from [{}]",
|
|
||||||
b.server_database_version,
|
|
||||||
v.server_database_version
|
|
||||||
);
|
|
||||||
m_database->QueryDatabase(fmt::format("UPDATE `db_version` SET `version` = {}", b.server_database_version));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (b.bots_database_version > 0) {
|
|
||||||
if (UpdateManifest(bot_manifest_entries, v.bots_database_version, b.bots_database_version)) {
|
|
||||||
LogInfo(
|
|
||||||
"Updates ran successfully, setting database version to [{}] from [{}]",
|
|
||||||
b.bots_database_version,
|
|
||||||
v.bots_database_version
|
|
||||||
);
|
|
||||||
m_database->QueryDatabase(
|
|
||||||
fmt::format(
|
|
||||||
"UPDATE `db_version` SET `bots_version` = {}",
|
|
||||||
b.bots_database_version
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string DatabaseUpdate::GetQueryResult(const ManifestEntry& e)
|
|
||||||
{
|
|
||||||
auto results = (e.content_schema_update ? m_content_database : m_database)->QueryDatabase(e.check);
|
|
||||||
|
|
||||||
std::vector<std::string> result_lines = {};
|
|
||||||
|
|
||||||
for (auto row = results.begin(); row != results.end(); ++row) {
|
|
||||||
std::vector<std::string> cols;
|
|
||||||
|
|
||||||
int field_count = results.ColumnCount();
|
|
||||||
cols.reserve(field_count);
|
|
||||||
for (int i = 0; i < field_count; ++i) {
|
|
||||||
if (row[i] != nullptr) {
|
|
||||||
cols.emplace_back(row[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
result_lines.emplace_back(Strings::Join(cols, " "));
|
|
||||||
}
|
|
||||||
|
|
||||||
return Strings::Join(result_lines, "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DatabaseUpdate::ShouldRunMigration(ManifestEntry &e, std::string query_result)
|
|
||||||
{
|
|
||||||
std::string r = Strings::Trim(query_result);
|
|
||||||
if (e.condition == "contains") {
|
|
||||||
return Strings::Contains(r, e.match);
|
|
||||||
}
|
|
||||||
else if (e.condition == "match") {
|
|
||||||
return r == e.match;
|
|
||||||
}
|
|
||||||
else if (e.condition == "missing") {
|
|
||||||
return !Strings::Contains(r, e.match);
|
|
||||||
}
|
|
||||||
else if (e.condition == "empty") {
|
|
||||||
return r.empty();
|
|
||||||
}
|
|
||||||
else if (e.condition == "not_empty") {
|
|
||||||
return !r.empty();
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if we are running in a terminal
|
|
||||||
bool is_atty()
|
|
||||||
{
|
|
||||||
#ifdef _WINDOWS
|
|
||||||
return ::_isatty(_fileno(stdin));
|
|
||||||
#else
|
|
||||||
return isatty(fileno(stdin));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// return true if we ran updates
|
|
||||||
bool DatabaseUpdate::UpdateManifest(
|
|
||||||
std::vector<ManifestEntry> entries,
|
|
||||||
int version_low,
|
|
||||||
int version_high
|
|
||||||
)
|
|
||||||
{
|
|
||||||
std::vector<int> missing_migrations = {};
|
|
||||||
if (version_low != version_high) {
|
|
||||||
|
|
||||||
LogSys.DisableMySQLErrorLogs();
|
|
||||||
for (int version = version_low + 1; version <= version_high; ++version) {
|
|
||||||
for (auto &e: entries) {
|
|
||||||
if (e.version == version) {
|
|
||||||
bool has_migration = true;
|
|
||||||
std::string r = GetQueryResult(e);
|
|
||||||
if (ShouldRunMigration(e, r)) {
|
|
||||||
has_migration = false;
|
|
||||||
missing_migrations.emplace_back(e.version);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string prefix = fmt::format(
|
|
||||||
"[{}]",
|
|
||||||
has_migration ? "ok" : "missing"
|
|
||||||
);
|
|
||||||
|
|
||||||
LogInfo(
|
|
||||||
"[{}] {:>10} | [{}]",
|
|
||||||
e.version,
|
|
||||||
prefix,
|
|
||||||
e.description
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
LogSys.EnableMySQLErrorLogs();
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
|
|
||||||
if (!missing_migrations.empty() && m_skip_backup) {
|
|
||||||
LogInfo("Skipping database backup");
|
|
||||||
}
|
|
||||||
else if (!missing_migrations.empty()) {
|
|
||||||
LogInfo("Automatically backing up database before applying updates");
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
auto s = DatabaseDumpService();
|
|
||||||
s.SetDumpAllTables(true);
|
|
||||||
s.SetDumpWithCompression(true);
|
|
||||||
s.DatabaseDump();
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!missing_migrations.empty()) {
|
|
||||||
LogInfo("Running database migrations. Please wait...");
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &m: missing_migrations) {
|
|
||||||
for (auto &e: entries) {
|
|
||||||
if (e.version == m) {
|
|
||||||
bool errored_migration = false;
|
|
||||||
|
|
||||||
auto r = (e.content_schema_update ? m_content_database : m_database)->QueryDatabaseMulti(e.sql);
|
|
||||||
|
|
||||||
// ignore empty query result "errors"
|
|
||||||
if (r.ErrorNumber() != 1065 && !r.ErrorMessage().empty()) {
|
|
||||||
LogError("(#{}) [{}]", r.ErrorNumber(), r.ErrorMessage());
|
|
||||||
errored_migration = true;
|
|
||||||
|
|
||||||
LogInfo("Required database update failed. This could be a problem");
|
|
||||||
|
|
||||||
// if terminal attached then prompt for skip
|
|
||||||
if (is_atty()) {
|
|
||||||
LogInfo("Would you like to skip this update? [y/n] (Timeout 60s)");
|
|
||||||
|
|
||||||
// user input
|
|
||||||
std::string input;
|
|
||||||
bool gave_input = false;
|
|
||||||
time_t start_time = time(nullptr);
|
|
||||||
time_t wait_time_seconds = 60;
|
|
||||||
|
|
||||||
// spawn a concurrent thread that waits for input from std::cin
|
|
||||||
std::thread t1(
|
|
||||||
[&]() {
|
|
||||||
std::cin >> input;
|
|
||||||
gave_input = true;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
t1.detach();
|
|
||||||
|
|
||||||
// check the inputReceived flag once every 50ms for 10 seconds
|
|
||||||
while (time(nullptr) < start_time + wait_time_seconds && !gave_input) {
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(50));
|
|
||||||
}
|
|
||||||
|
|
||||||
// prompt for user skip
|
|
||||||
if (Strings::Trim(input) == "y") {
|
|
||||||
errored_migration = false;
|
|
||||||
LogInfo("Skipping update [{}] [{}]", e.version, e.description);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
errored_migration = true;
|
|
||||||
LogInfo("Skipping update [{}] [{}]", e.version, e.description);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LogInfo(
|
|
||||||
"[{}] [{}] [{}]",
|
|
||||||
e.version,
|
|
||||||
e.description,
|
|
||||||
(errored_migration ? "error" : "ok")
|
|
||||||
);
|
|
||||||
|
|
||||||
if (errored_migration) {
|
|
||||||
LogError("Fatal | Database migration [{}] failed to run", e.description);
|
|
||||||
LogError("Fatal | Shutting down");
|
|
||||||
std::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
DatabaseUpdate *DatabaseUpdate::SetDatabase(Database *db)
|
|
||||||
{
|
|
||||||
m_database = db;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
DatabaseUpdate *DatabaseUpdate::SetContentDatabase(Database *db)
|
|
||||||
{
|
|
||||||
m_content_database = db;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
DatabaseUpdate *DatabaseUpdate::SetSkipBackup(bool skip)
|
|
||||||
{
|
|
||||||
m_skip_backup = skip;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DatabaseUpdate::CheckVersionsUpToDate(DatabaseVersion v, DatabaseVersion b)
|
|
||||||
{
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
|
|
||||||
LogInfo(
|
|
||||||
"{:>8} | database [{}] binary [{}] {}",
|
|
||||||
"Server",
|
|
||||||
v.server_database_version,
|
|
||||||
b.server_database_version,
|
|
||||||
(v.server_database_version == b.server_database_version) ? "up to date" : "checking updates"
|
|
||||||
);
|
|
||||||
|
|
||||||
if (RuleB(Bots, Enabled) && b.bots_database_version > 0) {
|
|
||||||
LogInfo(
|
|
||||||
"{:>8} | database [{}] binary [{}] {}",
|
|
||||||
"Bots",
|
|
||||||
v.bots_database_version,
|
|
||||||
b.bots_database_version,
|
|
||||||
(v.bots_database_version == b.bots_database_version) ? "up to date" : "checking updates"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
LogInfo("{:>8} | [server.auto_database_updates] [<green>true]", "Config");
|
|
||||||
|
|
||||||
LogInfo("{}", Strings::Repeat("-", BREAK_LENGTH));
|
|
||||||
|
|
||||||
// server database version is required
|
|
||||||
bool server_up_to_date = v.server_database_version >= b.server_database_version;
|
|
||||||
// bots database version is optional, if not enabled then it is always up-to-date
|
|
||||||
bool bots_up_to_date = RuleB(Bots, Enabled) ? v.bots_database_version >= b.bots_database_version : true;
|
|
||||||
|
|
||||||
return server_up_to_date && bots_up_to_date;
|
|
||||||
}
|
|
||||||
|
|
||||||
// checks to see if there are pending updates
|
|
||||||
// used by zone to prevent launch or boot loop until updates are applied
|
|
||||||
bool DatabaseUpdate::HasPendingUpdates()
|
|
||||||
{
|
|
||||||
auto v = GetDatabaseVersions();
|
|
||||||
auto b = GetBinaryDatabaseVersions();
|
|
||||||
|
|
||||||
return !CheckVersionsUpToDate(v, b);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DatabaseUpdate::InjectBotsVersionColumn()
|
|
||||||
{
|
|
||||||
auto r = m_database->QueryDatabase("show columns from db_version where Field like '%bots_version%'");
|
|
||||||
if (r.RowCount() == 0) {
|
|
||||||
m_database->QueryDatabase("ALTER TABLE db_version ADD bots_version int(11) DEFAULT '0' AFTER version");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
#ifndef EQEMU_DATABASE_UPDATE_H
|
|
||||||
#define EQEMU_DATABASE_UPDATE_H
|
|
||||||
|
|
||||||
#include "../database.h"
|
|
||||||
|
|
||||||
struct ManifestEntry {
|
|
||||||
int version{}; // database version of the migration
|
|
||||||
std::string description{}; // description of the migration ex: "add_new_table" or "add_index_to_table"
|
|
||||||
std::string check{}; // query that checks against the condition
|
|
||||||
std::string condition{}; // condition or "match_type" - Possible values [contains|match|missing|empty|not_empty]
|
|
||||||
std::string match{}; // match field that is not always used, but works in conjunction with "condition" values [missing|match|contains]
|
|
||||||
std::string sql{}; // the SQL DDL that gets ran when the condition is true
|
|
||||||
bool content_schema_update{}; // if true, this migration is a content schema update and should be ran against the content database
|
|
||||||
};
|
|
||||||
|
|
||||||
struct DatabaseVersion {
|
|
||||||
int server_database_version;
|
|
||||||
int bots_database_version;
|
|
||||||
};
|
|
||||||
|
|
||||||
class DatabaseUpdate {
|
|
||||||
public:
|
|
||||||
DatabaseVersion GetDatabaseVersions();
|
|
||||||
DatabaseVersion GetBinaryDatabaseVersions();
|
|
||||||
void CheckDbUpdates();
|
|
||||||
std::string GetQueryResult(const ManifestEntry& e);
|
|
||||||
static bool ShouldRunMigration(ManifestEntry &e, std::string query_result);
|
|
||||||
bool UpdateManifest(std::vector<ManifestEntry> entries, int version_low, int version_high);
|
|
||||||
|
|
||||||
DatabaseUpdate *SetDatabase(Database *db);
|
|
||||||
DatabaseUpdate *SetContentDatabase(Database *db);
|
|
||||||
DatabaseUpdate *SetSkipBackup(bool skip);
|
|
||||||
bool HasPendingUpdates();
|
|
||||||
private:
|
|
||||||
bool m_skip_backup = false;
|
|
||||||
Database *m_database;
|
|
||||||
Database *m_content_database;
|
|
||||||
static bool CheckVersionsUpToDate(DatabaseVersion v, DatabaseVersion b);
|
|
||||||
void InjectBotsVersionColumn();
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif //EQEMU_DATABASE_UPDATE_H
|
|
||||||
File diff suppressed because one or more lines are too long
@@ -1,186 +0,0 @@
|
|||||||
#include "database_update.h"
|
|
||||||
|
|
||||||
std::vector<ManifestEntry> bot_manifest_entries = {
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9035,
|
|
||||||
.description = "2022_12_04_bot_archery.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_data` LIKE 'archery_setting'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_data`
|
|
||||||
ADD COLUMN `archery_setting` TINYINT(2) UNSIGNED NOT NULL DEFAULT '0' AFTER `enforce_spell_settings`;
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9036,
|
|
||||||
.description = "2023_01_19_drop_bot_views.sql",
|
|
||||||
.check = "SHOW TABLES LIKE 'vw_groups'",
|
|
||||||
.condition = "not_empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
DROP VIEW vw_bot_groups;
|
|
||||||
DROP VIEW vw_bot_character_mobs;
|
|
||||||
DROP VIEW vw_groups;
|
|
||||||
DROP VIEW vw_guild_members;
|
|
||||||
DROP TABLE bot_guild_members;
|
|
||||||
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9037,
|
|
||||||
.description = "2023_01_22_add_name_index.sql",
|
|
||||||
.check = "show index from bot_data WHERE key_name = 'name`",
|
|
||||||
.condition = "",
|
|
||||||
.match = "empty",
|
|
||||||
.sql = R"(
|
|
||||||
create index `name` on bot_data(`name`);
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9038,
|
|
||||||
.description = "2023_02_16_add_caster_range.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_data` LIKE 'caster_range'",
|
|
||||||
.condition = "",
|
|
||||||
.match = "empty",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_data`
|
|
||||||
ADD COLUMN `caster_range` INT(11) UNSIGNED NOT NULL DEFAULT '300' AFTER `archery_setting`;
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9039,
|
|
||||||
.description = "2023_03_31_remove_bot_groups.sql",
|
|
||||||
.check = "SHOW TABLES LIKE 'bot_groups'",
|
|
||||||
.condition = "",
|
|
||||||
.match = "not_empty",
|
|
||||||
.sql = R"(
|
|
||||||
SET FOREIGN_KEY_CHECKS = 0;
|
|
||||||
DROP TABLE IF EXISTS `bot_groups`;
|
|
||||||
DROP TABLE IF EXISTS `bot_group_members`;
|
|
||||||
SET FOREIGN_KEY_CHECKS = 1;
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9040,
|
|
||||||
.description = "2023_11_16_bot_starting_items.sql",
|
|
||||||
.check = "SHOW TABLES LIKE 'bot_starting_items'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
CREATE TABLE `bot_starting_items` (
|
|
||||||
`id` int(11) UNSIGNED NOT NULL AUTO_INCREMENT,
|
|
||||||
`races` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
|
||||||
`classes` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
|
||||||
`item_id` int(11) UNSIGNED NOT NULL DEFAULT 0,
|
|
||||||
`item_charges` tinyint(3) UNSIGNED NOT NULL DEFAULT 1,
|
|
||||||
`min_status` tinyint(3) UNSIGNED NOT NULL DEFAULT 0,
|
|
||||||
`slot_id` mediumint(9) NOT NULL DEFAULT -1,
|
|
||||||
`min_expansion` tinyint(4) NOT NULL DEFAULT -1,
|
|
||||||
`max_expansion` tinyint(4) NOT NULL DEFAULT -1,
|
|
||||||
`content_flags` varchar(100) CHARACTER SET latin1 COLLATE latin1_swedish_ci NULL DEFAULT NULL,
|
|
||||||
`content_flags_disabled` varchar(100) CHARACTER SET latin1 COLLATE latin1_swedish_ci NULL DEFAULT NULL,
|
|
||||||
PRIMARY KEY (`id`)
|
|
||||||
) ENGINE = InnoDB CHARACTER SET = latin1 COLLATE = latin1_swedish_ci;
|
|
||||||
)",
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9041,
|
|
||||||
.description = "2023_12_04_bot_timers.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_timers` LIKE 'recast_time'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_timers`
|
|
||||||
ADD COLUMN `recast_time` INT(11) UNSIGNED NOT NULL DEFAULT '0' AFTER `timer_value`,
|
|
||||||
ADD COLUMN `is_spell` TINYINT(2) UNSIGNED NOT NULL DEFAULT 0 AFTER `recast_time`,
|
|
||||||
ADD COLUMN `is_disc` TINYINT(2) UNSIGNED NOT NULL DEFAULT 0 AFTER `is_spell`,
|
|
||||||
ADD COLUMN `spell_id` INT(11) UNSIGNED NOT NULL DEFAULT '0' AFTER `is_disc`,
|
|
||||||
ADD COLUMN `is_item` TINYINT(2) UNSIGNED NOT NULL DEFAULT 0 AFTER `spell_id`,
|
|
||||||
ADD COLUMN `item_id` INT(11) UNSIGNED NOT NULL DEFAULT '0' AFTER `is_item`;
|
|
||||||
ALTER TABLE `bot_timers`
|
|
||||||
DROP FOREIGN KEY `FK_bot_timers_1`;
|
|
||||||
ALTER TABLE `bot_timers`
|
|
||||||
DROP PRIMARY KEY;
|
|
||||||
ALTER TABLE `bot_timers`
|
|
||||||
ADD PRIMARY KEY (`bot_id`, `timer_id`, `spell_id`, `item_id`);
|
|
||||||
)"
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9042,
|
|
||||||
.description = "2024_01_27_delete_bot_foreign_keys.sql",
|
|
||||||
.check = "SHOW CREATE TABLE `bot_stances`",
|
|
||||||
.condition = "contains",
|
|
||||||
.match = "FOREIGN",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_buffs` DROP FOREIGN KEY `FK_bot_buffs_1`;
|
|
||||||
ALTER TABLE `bot_heal_rotations` DROP FOREIGN KEY `FK_bot_heal_rotations`;
|
|
||||||
ALTER TABLE `bot_heal_rotation_members` DROP FOREIGN KEY `FK_bot_heal_rotation_members_1`;
|
|
||||||
ALTER TABLE `bot_heal_rotation_members` DROP FOREIGN KEY `FK_bot_heal_rotation_members_2`;
|
|
||||||
ALTER TABLE `bot_heal_rotation_targets` DROP FOREIGN KEY `FK_bot_heal_rotation_targets`;
|
|
||||||
ALTER TABLE `bot_inventories` DROP FOREIGN KEY `FK_bot_inventories_1`;
|
|
||||||
ALTER TABLE `bot_pets` DROP FOREIGN KEY `FK_bot_pets_1`;
|
|
||||||
ALTER TABLE `bot_pet_buffs` DROP FOREIGN KEY `FK_bot_pet_buffs_1`;
|
|
||||||
ALTER TABLE `bot_pet_inventories` DROP FOREIGN KEY `FK_bot_pet_inventories_1`;
|
|
||||||
ALTER TABLE `bot_stances` DROP FOREIGN KEY `FK_bot_stances_1`;
|
|
||||||
)"
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9043,
|
|
||||||
.description = "2024_02_18_bot_starting_items_augments.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_starting_items` LIKE 'augment_one'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_starting_items`
|
|
||||||
ADD COLUMN `augment_one` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `item_charges`,
|
|
||||||
ADD COLUMN `augment_two` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `augment_one`,
|
|
||||||
ADD COLUMN `augment_three` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `augment_two`,
|
|
||||||
ADD COLUMN `augment_four` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `augment_three`,
|
|
||||||
ADD COLUMN `augment_five` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `augment_four`,
|
|
||||||
ADD COLUMN `augment_six` int(11) UNSIGNED NOT NULL DEFAULT 0 AFTER `augment_five`;
|
|
||||||
)"
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9044,
|
|
||||||
.description = "2024_04_23_bot_extra_haste.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_data` LIKE 'extra_haste'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_data`
|
|
||||||
ADD COLUMN `extra_haste` mediumint(8) NOT NULL DEFAULT 0 AFTER `wis`;
|
|
||||||
)"
|
|
||||||
},
|
|
||||||
ManifestEntry{
|
|
||||||
.version = 9045,
|
|
||||||
.description = "2024_08_05_bot_spells_entries_unsigned_spell_id.sql",
|
|
||||||
.check = "SHOW COLUMNS FROM `bot_spells_entries` LIKE 'spell_id'",
|
|
||||||
.condition = "empty",
|
|
||||||
.match = "",
|
|
||||||
.sql = R"(
|
|
||||||
ALTER TABLE `bot_spells_entries`
|
|
||||||
CHANGE COLUMN `spellid` `spell_id` smallint(5) UNSIGNED NOT NULL DEFAULT 0 AFTER `npc_spells_id`;
|
|
||||||
)"
|
|
||||||
}
|
|
||||||
// -- template; copy/paste this when you need to create a new entry
|
|
||||||
// ManifestEntry{
|
|
||||||
// .version = 9228,
|
|
||||||
// .description = "some_new_migration.sql",
|
|
||||||
// .check = "SHOW COLUMNS FROM `table_name` LIKE 'column_name'",
|
|
||||||
// .condition = "empty",
|
|
||||||
// .match = "",
|
|
||||||
// .sql = R"(
|
|
||||||
//
|
|
||||||
//)"
|
|
||||||
};
|
|
||||||
|
|
||||||
// see struct definitions for what each field does
|
|
||||||
// struct ManifestEntry {
|
|
||||||
// int version{}; // database version of the migration
|
|
||||||
// std::string description{}; // description of the migration ex: "add_new_table" or "add_index_to_table"
|
|
||||||
// std::string check{}; // query that checks against the condition
|
|
||||||
// std::string condition{}; // condition or "match_type" - Possible values [contains|match|missing|empty|not_empty]
|
|
||||||
// std::string match{}; // match field that is not always used, but works in conjunction with "condition" values [missing|match|contains]
|
|
||||||
// std::string sql{}; // the SQL DDL that gets ran when the condition is true
|
|
||||||
// };
|
|
||||||
File diff suppressed because it is too large
Load Diff
+369
-367
@@ -18,19 +18,10 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|||||||
|
|
||||||
#include "../common/global_define.h"
|
#include "../common/global_define.h"
|
||||||
#include "../common/rulesys.h"
|
#include "../common/rulesys.h"
|
||||||
#include "../common/strings.h"
|
#include "../common/string_util.h"
|
||||||
#include "../common/timer.h"
|
#include "../common/timer.h"
|
||||||
#include "../common/repositories/character_corpses_repository.h"
|
|
||||||
#include "../common/repositories/dynamic_zone_members_repository.h"
|
#include "../common/repositories/dynamic_zone_members_repository.h"
|
||||||
#include "../common/repositories/dynamic_zones_repository.h"
|
#include "../common/repositories/dynamic_zones_repository.h"
|
||||||
#include "../common/repositories/group_id_repository.h"
|
|
||||||
#include "../common/repositories/instance_list_repository.h"
|
|
||||||
#include "../common/repositories/instance_list_player_repository.h"
|
|
||||||
#include "../common/repositories/raid_members_repository.h"
|
|
||||||
#include "../common/repositories/respawn_times_repository.h"
|
|
||||||
#include "../common/repositories/spawn_condition_values_repository.h"
|
|
||||||
#include "repositories/spawn2_disabled_repository.h"
|
|
||||||
|
|
||||||
|
|
||||||
#include "database.h"
|
#include "database.h"
|
||||||
|
|
||||||
@@ -50,244 +41,251 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
bool Database::AddClientToInstance(uint16 instance_id, uint32 character_id)
|
* @param instance_id
|
||||||
|
* @param char_id
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
bool Database::AddClientToInstance(uint16 instance_id, uint32 char_id)
|
||||||
{
|
{
|
||||||
auto e = InstanceListPlayerRepository::NewEntity();
|
std::string query = StringFormat(
|
||||||
|
"REPLACE INTO `instance_list_player` (id, charid) "
|
||||||
|
"VALUES "
|
||||||
|
"(%lu, %lu)",
|
||||||
|
(unsigned long) instance_id,
|
||||||
|
(unsigned long) char_id
|
||||||
|
);
|
||||||
|
|
||||||
e.id = instance_id;
|
auto results = QueryDatabase(query);
|
||||||
e.charid = character_id;
|
|
||||||
|
|
||||||
return InstanceListPlayerRepository::ReplaceOne(*this, e);
|
return results.Success();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::CheckInstanceByCharID(uint16 instance_id, uint32 character_id)
|
bool Database::CharacterInInstanceGroup(uint16 instance_id, uint32 char_id)
|
||||||
{
|
{
|
||||||
if (!instance_id) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto l = InstanceListPlayerRepository::GetWhere(
|
std::string query = StringFormat("SELECT charid FROM instance_list_player where id=%u AND charid=%u", instance_id, char_id);
|
||||||
*this,
|
auto results = QueryDatabase(query);
|
||||||
fmt::format(
|
|
||||||
"id = {} AND charid = {}",
|
if (!results.Success())
|
||||||
instance_id,
|
return false;
|
||||||
character_id
|
|
||||||
)
|
if (results.RowCount() != 1)
|
||||||
);
|
|
||||||
if (l.empty()) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::CheckInstanceExists(uint16 instance_id)
|
bool Database::CheckInstanceExists(uint16 instance_id) {
|
||||||
{
|
std::string query = StringFormat(
|
||||||
if (!instance_id) {
|
"SELECT "
|
||||||
return false;
|
"`id` "
|
||||||
}
|
"FROM "
|
||||||
|
"`instance_list` "
|
||||||
|
"WHERE "
|
||||||
|
"`id` = %u",
|
||||||
|
instance_id
|
||||||
|
);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
if (!results.Success())
|
||||||
if (!i.id) {
|
return false;
|
||||||
|
|
||||||
|
if (results.RowCount() == 0)
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::CheckInstanceExpired(uint16 instance_id)
|
bool Database::CheckInstanceExpired(uint16 instance_id)
|
||||||
{
|
{
|
||||||
if (!instance_id) {
|
|
||||||
|
int32 start_time = 0;
|
||||||
|
int32 duration = 0;
|
||||||
|
uint32 never_expires = 0;
|
||||||
|
|
||||||
|
std::string query = StringFormat(
|
||||||
|
"SELECT start_time, duration, never_expires FROM instance_list WHERE id=%u",
|
||||||
|
instance_id
|
||||||
|
);
|
||||||
|
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
if (results.RowCount() == 0) {
|
||||||
if (!i.id) {
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (i.never_expires) {
|
auto row = results.begin();
|
||||||
|
|
||||||
|
start_time = atoi(row[0]);
|
||||||
|
duration = atoi(row[1]);
|
||||||
|
never_expires = atoi(row[2]);
|
||||||
|
|
||||||
|
if (never_expires == 1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
timeval tv{};
|
timeval tv{};
|
||||||
gettimeofday(&tv, nullptr);
|
gettimeofday(&tv, nullptr);
|
||||||
|
|
||||||
// Use uint64_t for the addition to prevent overflow
|
return (start_time + duration) <= tv.tv_sec;
|
||||||
uint64_t expiration_time = static_cast<uint64_t>(i.start_time) + static_cast<uint64_t>(i.duration);
|
|
||||||
return expiration_time <= tv.tv_sec;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::CreateInstance(uint16 instance_id, uint32 zone_id, uint32 version, uint32 duration)
|
bool Database::CreateInstance(uint16 instance_id, uint32 zone_id, uint32 version, uint32 duration)
|
||||||
{
|
{
|
||||||
auto e = InstanceListRepository::NewEntity();
|
std::string query = StringFormat(
|
||||||
|
"INSERT INTO instance_list (id, zone, version, start_time, duration)"
|
||||||
|
" values (%u, %u, %u, UNIX_TIMESTAMP(), %u)",
|
||||||
|
instance_id,
|
||||||
|
zone_id,
|
||||||
|
version,
|
||||||
|
duration
|
||||||
|
);
|
||||||
|
|
||||||
e.id = instance_id;
|
auto results = QueryDatabase(query);
|
||||||
e.zone = zone_id;
|
|
||||||
e.version = version;
|
|
||||||
e.start_time = std::time(nullptr);
|
|
||||||
e.duration = duration;
|
|
||||||
|
|
||||||
return InstanceListRepository::InsertOne(*this, e).id;
|
return results.Success();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::GetUnusedInstanceID(uint16 &instance_id)
|
bool Database::GetUnusedInstanceID(uint16 &instance_id)
|
||||||
{
|
{
|
||||||
uint32 max_reserved_instance_id = RuleI(Instances, ReservedInstances);
|
uint32 max_reserved_instance_id = RuleI(Instances, ReservedInstances);
|
||||||
uint32 max_instance_id = 32000;
|
uint32 max = 32000;
|
||||||
|
|
||||||
// sanity check reserved
|
std::string query = StringFormat(
|
||||||
if (max_reserved_instance_id >= max_instance_id) {
|
"SELECT IFNULL(MAX(id),%u)+1 FROM instance_list WHERE id > %u",
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// recycle instances
|
|
||||||
if (RuleB(Instances, RecycleInstanceIds)) {
|
|
||||||
|
|
||||||
//query to get first unused id above reserved
|
|
||||||
auto query = fmt::format(
|
|
||||||
SQL(
|
|
||||||
SELECT id
|
|
||||||
FROM instance_list
|
|
||||||
WHERE id = {};
|
|
||||||
),
|
|
||||||
max_reserved_instance_id + 1
|
|
||||||
);
|
|
||||||
|
|
||||||
auto results = QueryDatabase(query);
|
|
||||||
|
|
||||||
// could not successfully query - bail out
|
|
||||||
if (!results.Success()) {
|
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// first id is available
|
|
||||||
if (results.RowCount() == 0) {
|
|
||||||
instance_id = max_reserved_instance_id + 1;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// now look for next available above reserved
|
|
||||||
query = fmt::format(
|
|
||||||
SQL(
|
|
||||||
SELECT MIN(i.id + 1) AS next_available
|
|
||||||
FROM instance_list i
|
|
||||||
LEFT JOIN instance_list i2 ON i.id + 1 = i2.id
|
|
||||||
WHERE i.id >= {}
|
|
||||||
AND i2.id IS NULL;
|
|
||||||
),
|
|
||||||
max_reserved_instance_id
|
|
||||||
);
|
|
||||||
|
|
||||||
results = QueryDatabase(query);
|
|
||||||
|
|
||||||
// could not successfully query - bail out
|
|
||||||
if (!results.Success()) {
|
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// did not retrieve any rows - bail out
|
|
||||||
if (results.RowCount() == 0) {
|
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto row = results.begin();
|
|
||||||
|
|
||||||
// check that id is within limits
|
|
||||||
if (row[0] && Strings::ToInt(row[0]) <= max_instance_id) {
|
|
||||||
instance_id = Strings::ToInt(row[0]);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// no available instance ids
|
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// get max unused id above reserved
|
|
||||||
auto query = fmt::format(
|
|
||||||
"SELECT IFNULL(MAX(id), {}) + 1 FROM instance_list WHERE id > {}",
|
|
||||||
max_reserved_instance_id,
|
max_reserved_instance_id,
|
||||||
max_reserved_instance_id
|
max_reserved_instance_id
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (RuleB(Instances, RecycleInstanceIds)) {
|
||||||
|
query = (
|
||||||
|
SQL(
|
||||||
|
SELECT i.id + 1 AS next_available
|
||||||
|
FROM instance_list i
|
||||||
|
LEFT JOIN instance_list i2 ON i2.id = i.id + 1
|
||||||
|
WHERE i2.id IS NULL
|
||||||
|
ORDER BY i.id
|
||||||
|
LIMIT 0, 1;
|
||||||
|
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
auto results = QueryDatabase(query);
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
// could not successfully query - bail out
|
|
||||||
if (!results.Success()) {
|
if (!results.Success()) {
|
||||||
instance_id = 0;
|
instance_id = 0;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// did not retrieve any rows - bail out
|
if (results.RowCount() == 0) {
|
||||||
|
instance_id = max_reserved_instance_id;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
|
||||||
|
if (atoi(row[0]) <= max) {
|
||||||
|
instance_id = atoi(row[0]);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (instance_id < max_reserved_instance_id) {
|
||||||
|
instance_id = max_reserved_instance_id;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
query = StringFormat("SELECT id FROM instance_list where id > %u ORDER BY id", max_reserved_instance_id);
|
||||||
|
results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success()) {
|
||||||
|
instance_id = 0;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (results.RowCount() == 0) {
|
if (results.RowCount() == 0) {
|
||||||
instance_id = 0;
|
instance_id = 0;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto row = results.begin();
|
max_reserved_instance_id++;
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
|
if (max_reserved_instance_id < atoi(row[0])) {
|
||||||
|
instance_id = max_reserved_instance_id;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// no instances currently used
|
if (max_reserved_instance_id > max) {
|
||||||
if (!row[0]) {
|
instance_id = 0;
|
||||||
instance_id = max_reserved_instance_id + 1;
|
return false;
|
||||||
return true;
|
}
|
||||||
|
|
||||||
|
max_reserved_instance_id++;
|
||||||
}
|
}
|
||||||
|
|
||||||
// check that id is within limits
|
instance_id = max_reserved_instance_id;
|
||||||
if (Strings::ToInt(row[0]) <= max_instance_id) {
|
|
||||||
instance_id = Strings::ToInt(row[0]);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// no available instance ids
|
return true;
|
||||||
instance_id = 0;
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::IsGlobalInstance(uint16 instance_id)
|
bool Database::GlobalInstance(uint16 instance_id)
|
||||||
{
|
{
|
||||||
if (!instance_id) {
|
std::string query = StringFormat(
|
||||||
return false;
|
"SELECT "
|
||||||
}
|
"is_global "
|
||||||
|
"FROM "
|
||||||
|
"instance_list "
|
||||||
|
"WHERE "
|
||||||
|
"id = %u "
|
||||||
|
"LIMIT 1 ",
|
||||||
|
instance_id
|
||||||
|
);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
if (!results.Success())
|
||||||
if (!i.id) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return i.is_global;
|
if (results.RowCount() == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
|
||||||
|
return (atoi(row[0]) == 1) ? true : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::RemoveClientFromInstance(uint16 instance_id, uint32 char_id)
|
bool Database::RemoveClientFromInstance(uint16 instance_id, uint32 char_id)
|
||||||
{
|
{
|
||||||
return InstanceListPlayerRepository::DeleteWhere(
|
std::string query = StringFormat("DELETE FROM instance_list_player WHERE id=%lu AND charid=%lu",
|
||||||
*this,
|
(unsigned long)instance_id, (unsigned long)char_id);
|
||||||
fmt::format(
|
auto results = QueryDatabase(query);
|
||||||
"id = {} AND charid = {}",
|
|
||||||
instance_id,
|
return results.Success();
|
||||||
char_id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool Database::RemoveClientsFromInstance(uint16 instance_id)
|
bool Database::RemoveClientsFromInstance(uint16 instance_id)
|
||||||
{
|
{
|
||||||
return InstanceListPlayerRepository::DeleteOne(*this, instance_id);
|
std::string query = StringFormat("DELETE FROM instance_list_player WHERE id=%lu", (unsigned long)instance_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
return results.Success();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Database::VerifyInstanceAlive(uint16 instance_id, uint32 character_id)
|
bool Database::VerifyInstanceAlive(uint16 instance_id, uint32 char_id)
|
||||||
{
|
{
|
||||||
//we are not saved to this instance so set our instance to 0
|
//we are not saved to this instance so set our instance to 0
|
||||||
if (!IsGlobalInstance(instance_id) && !CheckInstanceByCharID(instance_id, character_id)) {
|
if (!GlobalInstance(instance_id) && !CharacterInInstanceGroup(instance_id, char_id))
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
if (CheckInstanceExpired(instance_id)) {
|
if (CheckInstanceExpired(instance_id))
|
||||||
|
{
|
||||||
DeleteInstance(instance_id);
|
DeleteInstance(instance_id);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -297,102 +295,99 @@ bool Database::VerifyInstanceAlive(uint16 instance_id, uint32 character_id)
|
|||||||
|
|
||||||
bool Database::VerifyZoneInstance(uint32 zone_id, uint16 instance_id)
|
bool Database::VerifyZoneInstance(uint32 zone_id, uint16 instance_id)
|
||||||
{
|
{
|
||||||
auto l = InstanceListRepository::GetWhere(
|
|
||||||
*this,
|
std::string query = StringFormat("SELECT id FROM instance_list where id=%u AND zone=%u", instance_id, zone_id);
|
||||||
fmt::format(
|
auto results = QueryDatabase(query);
|
||||||
"id = {} AND zone = {}",
|
|
||||||
instance_id,
|
if (!results.Success())
|
||||||
zone_id
|
return false;
|
||||||
)
|
|
||||||
);
|
if (results.RowCount() == 0)
|
||||||
if (l.empty()) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16 Database::GetInstanceID(uint32 zone_id, uint32 character_id, int16 version)
|
uint16 Database::GetInstanceID(uint32 zone, uint32 character_id, int16 version)
|
||||||
{
|
{
|
||||||
if (!zone_id) {
|
if (!zone)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
std::string query = StringFormat(
|
||||||
|
"SELECT "
|
||||||
|
"instance_list.id "
|
||||||
|
"FROM "
|
||||||
|
"instance_list, "
|
||||||
|
"instance_list_player "
|
||||||
|
"WHERE "
|
||||||
|
"instance_list.zone = %u "
|
||||||
|
"AND instance_list.version = %u "
|
||||||
|
"AND instance_list.id = instance_list_player.id "
|
||||||
|
"AND instance_list_player.charid = %u "
|
||||||
|
"LIMIT 1; ",
|
||||||
|
zone,
|
||||||
|
version,
|
||||||
|
character_id
|
||||||
|
);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (results.RowCount() == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
|
||||||
|
return atoi(row[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16 Database::GetInstanceVersion(uint16 instance_id) {
|
||||||
|
if (instance_id == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
std::string query = StringFormat("SELECT version FROM instance_list where id=%u", instance_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (results.RowCount() == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
return atoi(row[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32 Database::GetTimeRemainingInstance(uint16 instance_id, bool &is_perma)
|
||||||
|
{
|
||||||
|
uint32 start_time = 0;
|
||||||
|
uint32 duration = 0;
|
||||||
|
uint32 never_expires = 0;
|
||||||
|
|
||||||
|
std::string query = StringFormat("SELECT start_time, duration, never_expires FROM instance_list WHERE id=%u", instance_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
|
{
|
||||||
|
is_perma = false;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto query = fmt::format(
|
if (results.RowCount() == 0)
|
||||||
"SELECT instance_list.id FROM "
|
{
|
||||||
"instance_list, instance_list_player WHERE "
|
is_perma = false;
|
||||||
"instance_list.zone = {} AND "
|
|
||||||
"instance_list.version = {} AND "
|
|
||||||
"instance_list.id = instance_list_player.id AND "
|
|
||||||
"instance_list_player.charid = {} "
|
|
||||||
"LIMIT 1;",
|
|
||||||
zone_id,
|
|
||||||
version,
|
|
||||||
character_id
|
|
||||||
);
|
|
||||||
auto results = QueryDatabase(query);
|
|
||||||
|
|
||||||
if (!results.Success() || !results.RowCount()) {
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto row = results.begin();
|
auto row = results.begin();
|
||||||
|
|
||||||
return static_cast<uint16>(Strings::ToUnsignedInt(row[0]));
|
start_time = atoi(row[0]);
|
||||||
}
|
duration = atoi(row[1]);
|
||||||
|
never_expires = atoi(row[2]);
|
||||||
|
|
||||||
std::vector<uint16> Database::GetInstanceIDs(uint32 zone_id, uint32 character_id)
|
if (never_expires == 1)
|
||||||
{
|
{
|
||||||
std::vector<uint16> l;
|
|
||||||
|
|
||||||
if (!zone_id) {
|
|
||||||
return l;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto query = fmt::format(
|
|
||||||
"SELECT instance_list.id FROM "
|
|
||||||
"instance_list, instance_list_player WHERE "
|
|
||||||
"instance_list.zone = {} AND "
|
|
||||||
"instance_list.id = instance_list_player.id AND "
|
|
||||||
"instance_list_player.charid = {}",
|
|
||||||
zone_id,
|
|
||||||
character_id
|
|
||||||
);
|
|
||||||
auto results = QueryDatabase(query);
|
|
||||||
|
|
||||||
if (!results.Success() || !results.RowCount()) {
|
|
||||||
return l;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto row : results) {
|
|
||||||
l.push_back(static_cast<uint16>(Strings::ToUnsignedInt(row[0])));
|
|
||||||
}
|
|
||||||
|
|
||||||
return l;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t Database::GetInstanceVersion(uint16 instance_id) {
|
|
||||||
if (!instance_id) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
|
||||||
if (!i.id) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return i.version;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 Database::GetTimeRemainingInstance(uint16 instance_id, bool &is_perma)
|
|
||||||
{
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
|
||||||
if (!i.id) {
|
|
||||||
is_perma = false;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (i.never_expires) {
|
|
||||||
is_perma = true;
|
is_perma = true;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -401,197 +396,204 @@ uint32 Database::GetTimeRemainingInstance(uint16 instance_id, bool &is_perma)
|
|||||||
|
|
||||||
timeval tv;
|
timeval tv;
|
||||||
gettimeofday(&tv, nullptr);
|
gettimeofday(&tv, nullptr);
|
||||||
return ((i.start_time + i.duration) - tv.tv_sec);
|
return ((start_time + duration) - tv.tv_sec);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 Database::GetInstanceZoneID(uint16 instance_id)
|
uint32 Database::VersionFromInstanceID(uint16 instance_id)
|
||||||
{
|
{
|
||||||
if (!instance_id) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
std::string query = StringFormat("SELECT version FROM instance_list where id=%u", instance_id);
|
||||||
if (!i.id) {
|
auto results = QueryDatabase(query);
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return i.zone;
|
if (!results.Success())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (results.RowCount() == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
|
||||||
|
return atoi(row[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32 Database::ZoneIDFromInstanceID(uint16 instance_id)
|
||||||
|
{
|
||||||
|
|
||||||
|
std::string query = StringFormat("SELECT zone FROM instance_list where id=%u", instance_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (results.RowCount() == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto row = results.begin();
|
||||||
|
|
||||||
|
return atoi(row[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::AssignGroupToInstance(uint32 group_id, uint32 instance_id)
|
void Database::AssignGroupToInstance(uint32 group_id, uint32 instance_id)
|
||||||
{
|
{
|
||||||
auto zone_id = GetInstanceZoneID(instance_id);
|
|
||||||
auto version = GetInstanceVersion(instance_id);
|
|
||||||
|
|
||||||
const auto& l = GroupIdRepository::GetWhere(
|
uint32 zone_id = ZoneIDFromInstanceID(instance_id);
|
||||||
*this,
|
uint16 version = VersionFromInstanceID(instance_id);
|
||||||
fmt::format(
|
|
||||||
"`group_id` = {}",
|
|
||||||
group_id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
if (l.empty()) {
|
std::string query = StringFormat("SELECT `charid` FROM `group_id` WHERE `groupid` = %u", group_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& e : l) {
|
for (auto row = results.begin(); row != results.end(); ++row)
|
||||||
if (!e.character_id) {
|
{
|
||||||
continue;
|
uint32 charid = atoi(row[0]);
|
||||||
}
|
if (GetInstanceID(zone_id, charid, version) == 0)
|
||||||
|
AddClientToInstance(instance_id, charid);
|
||||||
if (!GetInstanceID(zone_id, e.character_id, version)) {
|
|
||||||
AddClientToInstance(instance_id, e.character_id);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::AssignRaidToInstance(uint32 raid_id, uint32 instance_id)
|
void Database::AssignRaidToInstance(uint32 raid_id, uint32 instance_id)
|
||||||
{
|
{
|
||||||
auto zone_id = GetInstanceZoneID(instance_id);
|
|
||||||
auto version = GetInstanceVersion(instance_id);
|
|
||||||
|
|
||||||
auto l = RaidMembersRepository::GetWhere(
|
uint32 zone_id = ZoneIDFromInstanceID(instance_id);
|
||||||
*this,
|
uint16 version = VersionFromInstanceID(instance_id);
|
||||||
|
|
||||||
|
std::string query = StringFormat("SELECT `charid` FROM `raid_members` WHERE `raidid` = %u", raid_id);
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success())
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row)
|
||||||
|
{
|
||||||
|
uint32 charid = atoi(row[0]);
|
||||||
|
if (GetInstanceID(zone_id, charid, version) == 0)
|
||||||
|
AddClientToInstance(instance_id, charid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Database::BuryCorpsesInInstance(uint16 instance_id) {
|
||||||
|
QueryDatabase(
|
||||||
fmt::format(
|
fmt::format(
|
||||||
"raidid = {}",
|
"UPDATE character_corpses SET is_buried = 1, instance_id = 0 WHERE instance_id = {}",
|
||||||
raid_id
|
instance_id
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
if (l.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& e : l) {
|
|
||||||
if (!GetInstanceID(zone_id, e.charid, version)) {
|
|
||||||
AddClientToInstance(instance_id, e.charid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::DeleteInstance(uint16 instance_id)
|
void Database::DeleteInstance(uint16 instance_id)
|
||||||
{
|
{
|
||||||
// I'm not sure why this isn't in here but we should add it in a later change and make sure it's tested
|
std::string query;
|
||||||
// InstanceListRepository::DeleteWhere(*this, fmt::format("id = {}", instance_id));
|
|
||||||
InstanceListPlayerRepository::DeleteWhere(*this, fmt::format("id = {}", instance_id));
|
query = StringFormat("DELETE FROM instance_list_player WHERE id=%u", instance_id);
|
||||||
RespawnTimesRepository::DeleteWhere(*this, fmt::format("instance_id = {}", instance_id));
|
QueryDatabase(query);
|
||||||
SpawnConditionValuesRepository::DeleteWhere(*this, fmt::format("instance_id = {}", instance_id));
|
|
||||||
|
query = StringFormat("DELETE FROM respawn_times WHERE instance_id=%u", instance_id);
|
||||||
|
QueryDatabase(query);
|
||||||
|
|
||||||
|
query = StringFormat("DELETE FROM spawn_condition_values WHERE instance_id=%u", instance_id);
|
||||||
|
QueryDatabase(query);
|
||||||
|
|
||||||
DynamicZoneMembersRepository::DeleteByInstance(*this, instance_id);
|
DynamicZoneMembersRepository::DeleteByInstance(*this, instance_id);
|
||||||
DynamicZonesRepository::DeleteWhere(*this, fmt::format("instance_id = {}", instance_id));
|
DynamicZonesRepository::DeleteWhere(*this, fmt::format("instance_id = {}", instance_id));
|
||||||
CharacterCorpsesRepository::BuryInstance(*this, instance_id);
|
|
||||||
|
BuryCorpsesInInstance(instance_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::FlagInstanceByGroupLeader(uint32 zone_id, int16 version, uint32 character_id, uint32 group_id)
|
void Database::FlagInstanceByGroupLeader(uint32 zone, int16 version, uint32 charid, uint32 gid)
|
||||||
{
|
{
|
||||||
auto instance_id = GetInstanceID(zone_id, character_id, version);
|
uint16 id = GetInstanceID(zone, charid, version);
|
||||||
if (instance_id) {
|
if (id != 0)
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
char ln[128];
|
char ln[128];
|
||||||
memset(ln, 0, 128);
|
memset(ln, 0, 128);
|
||||||
GetGroupLeadershipInfo(group_id, ln);
|
GetGroupLeadershipInfo(gid, ln);
|
||||||
|
uint32 l_charid = GetCharacterID((const char*)ln);
|
||||||
|
uint16 l_id = GetInstanceID(zone, l_charid, version);
|
||||||
|
|
||||||
auto group_leader_id = GetCharacterID((const char*)ln);
|
if (l_id == 0)
|
||||||
auto group_leader_instance_id = GetInstanceID(zone_id, group_leader_id, version);
|
|
||||||
|
|
||||||
if (!group_leader_instance_id) {
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
AddClientToInstance(group_leader_instance_id, character_id);
|
AddClientToInstance(l_id, charid);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::FlagInstanceByRaidLeader(uint32 zone_id, int16 version, uint32 character_id, uint32 raid_id)
|
void Database::FlagInstanceByRaidLeader(uint32 zone, int16 version, uint32 charid, uint32 rid)
|
||||||
{
|
{
|
||||||
uint16 instance_id = GetInstanceID(zone_id, character_id, version);
|
uint16 id = GetInstanceID(zone, charid, version);
|
||||||
if (instance_id) {
|
if (id != 0)
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
auto raid_leader_id = GetCharacterID(GetRaidLeaderName(raid_id).c_str());
|
uint32 l_charid = GetCharacterID(GetRaidLeaderName(rid));
|
||||||
auto raid_leader_instance_id = GetInstanceID(zone_id, raid_leader_id, version);
|
uint16 l_id = GetInstanceID(zone, l_charid, version);
|
||||||
|
|
||||||
if (!raid_leader_instance_id) {
|
if (l_id == 0)
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
AddClientToInstance(raid_leader_instance_id, character_id);
|
AddClientToInstance(l_id, charid);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::GetCharactersInInstance(uint16 instance_id, std::list<uint32> &character_ids)
|
void Database::GetCharactersInInstance(uint16 instance_id, std::list<uint32> &charid_list) {
|
||||||
{
|
|
||||||
auto l = InstanceListPlayerRepository::GetWhere(*this, fmt::format("id = {}", instance_id));
|
|
||||||
if (l.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& e : l) {
|
std::string query = StringFormat("SELECT `charid` FROM `instance_list_player` WHERE `id` = %u", instance_id);
|
||||||
character_ids.push_back(e.charid);
|
auto results = QueryDatabase(query);
|
||||||
}
|
|
||||||
|
if (!results.Success())
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (auto row = results.begin(); row != results.end(); ++row)
|
||||||
|
charid_list.push_back(atoi(row[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::PurgeExpiredInstances()
|
void Database::PurgeExpiredInstances()
|
||||||
{
|
{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delay purging by a day so that we can continue using adjacent free instance id's
|
* Delay purging by a day so that we can continue using adjacent free instance id's
|
||||||
* from the table without risking the chance we immediately re-allocate a zone that freshly expired but
|
* from the table without risking the chance we immediately re-allocate a zone that freshly expired but
|
||||||
* has not been fully de-allocated
|
* has not been fully de-allocated
|
||||||
*/
|
*/
|
||||||
auto l = InstanceListRepository::GetWhere(
|
std::string query =
|
||||||
*this,
|
SQL(
|
||||||
"(start_time + duration) <= (UNIX_TIMESTAMP() - 86400) AND never_expires = 0"
|
SELECT
|
||||||
);
|
id
|
||||||
if (l.empty()) {
|
FROM
|
||||||
|
instance_list
|
||||||
|
where
|
||||||
|
(start_time + duration) <= (UNIX_TIMESTAMP() - 86400)
|
||||||
|
and never_expires = 0
|
||||||
|
);
|
||||||
|
|
||||||
|
auto results = QueryDatabase(query);
|
||||||
|
|
||||||
|
if (!results.Success()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (results.RowCount() == 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<std::string> instance_ids;
|
std::vector<std::string> instance_ids;
|
||||||
for (const auto& e : l) {
|
for (auto row = results.begin(); row != results.end(); ++row) {
|
||||||
instance_ids.emplace_back(std::to_string(e.id));
|
instance_ids.emplace_back(row[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto imploded_instance_ids = Strings::Implode(",", instance_ids);
|
std::string imploded_instance_ids = implode(",", instance_ids);
|
||||||
|
|
||||||
InstanceListRepository::DeleteWhere(*this, fmt::format("id IN ({})", imploded_instance_ids));
|
QueryDatabase(fmt::format("DELETE FROM instance_list WHERE id IN ({})", imploded_instance_ids));
|
||||||
InstanceListPlayerRepository::DeleteWhere(*this, fmt::format("id IN ({})", imploded_instance_ids));
|
QueryDatabase(fmt::format("DELETE FROM instance_list_player WHERE id IN ({})", imploded_instance_ids));
|
||||||
RespawnTimesRepository::DeleteWhere(*this, fmt::format("instance_id IN ({})", imploded_instance_ids));
|
QueryDatabase(fmt::format("DELETE FROM respawn_times WHERE instance_id IN ({})", imploded_instance_ids));
|
||||||
SpawnConditionValuesRepository::DeleteWhere(*this, fmt::format("instance_id IN ({})", imploded_instance_ids));
|
QueryDatabase(fmt::format("DELETE FROM spawn_condition_values WHERE instance_id IN ({})", imploded_instance_ids));
|
||||||
CharacterCorpsesRepository::BuryInstances(*this, imploded_instance_ids);
|
QueryDatabase(fmt::format("UPDATE character_corpses SET is_buried = 1, instance_id = 0 WHERE instance_id IN ({})", imploded_instance_ids));
|
||||||
DynamicZoneMembersRepository::DeleteByManyInstances(*this, imploded_instance_ids);
|
DynamicZoneMembersRepository::DeleteByManyInstances(*this, imploded_instance_ids);
|
||||||
DynamicZonesRepository::DeleteWhere(*this, fmt::format("instance_id IN ({})", imploded_instance_ids));
|
DynamicZonesRepository::DeleteWhere(*this, fmt::format("instance_id IN ({})", imploded_instance_ids));
|
||||||
Spawn2DisabledRepository::DeleteWhere(*this, fmt::format("instance_id IN ({})", imploded_instance_ids));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Database::SetInstanceDuration(uint16 instance_id, uint32 new_duration)
|
void Database::SetInstanceDuration(uint16 instance_id, uint32 new_duration)
|
||||||
{
|
{
|
||||||
auto i = InstanceListRepository::FindOne(*this, instance_id);
|
std::string query = StringFormat("UPDATE `instance_list` SET start_time=UNIX_TIMESTAMP(), "
|
||||||
if (!i.id) {
|
"duration=%u WHERE id=%u", new_duration, instance_id);
|
||||||
return;
|
auto results = QueryDatabase(query);
|
||||||
}
|
|
||||||
|
|
||||||
i.start_time = std::time(nullptr);
|
|
||||||
i.duration = new_duration;
|
|
||||||
|
|
||||||
InstanceListRepository::UpdateOne(*this, i);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Database::CleanupInstanceCorpses() {
|
|
||||||
auto l = InstanceListRepository::GetWhere(
|
|
||||||
*this,
|
|
||||||
"never_expires = 0"
|
|
||||||
);
|
|
||||||
|
|
||||||
if (l.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<std::string> instance_ids;
|
|
||||||
for (const auto& e : l) {
|
|
||||||
instance_ids.emplace_back(std::to_string(e.id));
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto imploded_instance_ids = Strings::Implode(",", instance_ids);
|
|
||||||
|
|
||||||
CharacterCorpsesRepository::BuryInstances(*this, imploded_instance_ids);
|
|
||||||
}
|
}
|
||||||
|
|||||||
+11
-96
@@ -36,6 +36,7 @@ namespace DatabaseSchema {
|
|||||||
{
|
{
|
||||||
return {
|
return {
|
||||||
{"adventure_stats", "player_id"},
|
{"adventure_stats", "player_id"},
|
||||||
|
{"buyer", "charid"},
|
||||||
{"char_recipe_list", "char_id"},
|
{"char_recipe_list", "char_id"},
|
||||||
{"character_activities", "charid"},
|
{"character_activities", "charid"},
|
||||||
{"character_alt_currency", "char_id"},
|
{"character_alt_currency", "char_id"},
|
||||||
@@ -51,30 +52,22 @@ namespace DatabaseSchema {
|
|||||||
{"character_enabledtasks", "charid"},
|
{"character_enabledtasks", "charid"},
|
||||||
{"character_expedition_lockouts", "character_id"},
|
{"character_expedition_lockouts", "character_id"},
|
||||||
{"character_exp_modifiers", "character_id"},
|
{"character_exp_modifiers", "character_id"},
|
||||||
{"character_evolving_items", "character_id"},
|
|
||||||
{"character_inspect_messages", "id"},
|
{"character_inspect_messages", "id"},
|
||||||
{"character_instance_safereturns", "character_id"},
|
|
||||||
{"character_item_recast", "id"},
|
{"character_item_recast", "id"},
|
||||||
{"character_languages", "id"},
|
{"character_languages", "id"},
|
||||||
{"character_leadership_abilities", "id"},
|
{"character_leadership_abilities", "id"},
|
||||||
{"character_material", "id"},
|
{"character_material", "id"},
|
||||||
{"character_memmed_spells", "id"},
|
{"character_memmed_spells", "id"},
|
||||||
{"character_parcels", "char_id"},
|
|
||||||
{"character_parcels_containers", "id"},
|
|
||||||
{"character_pet_buffs", "char_id"},
|
{"character_pet_buffs", "char_id"},
|
||||||
{"character_pet_info", "char_id"},
|
{"character_pet_info", "char_id"},
|
||||||
{"character_pet_inventory", "char_id"},
|
{"character_pet_inventory", "char_id"},
|
||||||
{"character_pet_name", "character_id"},
|
|
||||||
{"character_peqzone_flags", "id"},
|
|
||||||
{"character_potionbelt", "id"},
|
{"character_potionbelt", "id"},
|
||||||
{"character_skills", "id"},
|
{"character_skills", "id"},
|
||||||
{"character_spells", "id"},
|
{"character_spells", "id"},
|
||||||
{"character_stats_record", "character_id"},
|
|
||||||
{"character_task_timers", "character_id"},
|
|
||||||
{"character_tasks", "charid"},
|
{"character_tasks", "charid"},
|
||||||
{"character_tribute", "character_id"},
|
{"character_tribute", "id"},
|
||||||
{"completed_tasks", "charid"},
|
{"completed_tasks", "charid"},
|
||||||
{"data_buckets", "character_id"},
|
{"data_buckets", "id"},
|
||||||
{"faction_values", "char_id"},
|
{"faction_values", "char_id"},
|
||||||
{"friends", "charid"},
|
{"friends", "charid"},
|
||||||
{"guild_members", "char_id"},
|
{"guild_members", "char_id"},
|
||||||
@@ -87,6 +80,7 @@ namespace DatabaseSchema {
|
|||||||
{"player_titlesets", "char_id"},
|
{"player_titlesets", "char_id"},
|
||||||
{"quest_globals", "charid"},
|
{"quest_globals", "charid"},
|
||||||
{"timers", "char_id"},
|
{"timers", "char_id"},
|
||||||
|
{"titles", "char_id"},
|
||||||
{"trader", "char_id"},
|
{"trader", "char_id"},
|
||||||
{"zone_flags", "charID"}
|
{"zone_flags", "charID"}
|
||||||
};
|
};
|
||||||
@@ -108,8 +102,6 @@ namespace DatabaseSchema {
|
|||||||
"adventure_details",
|
"adventure_details",
|
||||||
"adventure_stats",
|
"adventure_stats",
|
||||||
"buyer",
|
"buyer",
|
||||||
"buyer_buy_lines",
|
|
||||||
"buyer_trade_items",
|
|
||||||
"char_recipe_list",
|
"char_recipe_list",
|
||||||
"character_activities",
|
"character_activities",
|
||||||
"character_alt_currency",
|
"character_alt_currency",
|
||||||
@@ -126,25 +118,18 @@ namespace DatabaseSchema {
|
|||||||
"character_enabledtasks",
|
"character_enabledtasks",
|
||||||
"character_expedition_lockouts",
|
"character_expedition_lockouts",
|
||||||
"character_exp_modifiers",
|
"character_exp_modifiers",
|
||||||
"character_evolving_items",
|
|
||||||
"character_inspect_messages",
|
"character_inspect_messages",
|
||||||
"character_instance_safereturns",
|
|
||||||
"character_item_recast",
|
"character_item_recast",
|
||||||
"character_languages",
|
"character_languages",
|
||||||
"character_leadership_abilities",
|
"character_leadership_abilities",
|
||||||
"character_material",
|
"character_material",
|
||||||
"character_memmed_spells",
|
"character_memmed_spells",
|
||||||
"character_parcels",
|
|
||||||
"character_parcels_containers",
|
|
||||||
"character_pet_buffs",
|
"character_pet_buffs",
|
||||||
"character_pet_info",
|
"character_pet_info",
|
||||||
"character_pet_inventory",
|
"character_pet_inventory",
|
||||||
"character_peqzone_flags",
|
|
||||||
"character_potionbelt",
|
"character_potionbelt",
|
||||||
"character_skills",
|
"character_skills",
|
||||||
"character_spells",
|
"character_spells",
|
||||||
"character_stats_record",
|
|
||||||
"character_task_timers",
|
|
||||||
"character_tasks",
|
"character_tasks",
|
||||||
"character_tribute",
|
"character_tribute",
|
||||||
"completed_tasks",
|
"completed_tasks",
|
||||||
@@ -154,10 +139,8 @@ namespace DatabaseSchema {
|
|||||||
"friends",
|
"friends",
|
||||||
"guild_bank",
|
"guild_bank",
|
||||||
"guild_members",
|
"guild_members",
|
||||||
"guild_permissions",
|
|
||||||
"guild_ranks",
|
"guild_ranks",
|
||||||
"guild_relations",
|
"guild_relations",
|
||||||
"guild_tributes",
|
|
||||||
"guilds",
|
"guilds",
|
||||||
"instance_list_player",
|
"instance_list_player",
|
||||||
"inventory",
|
"inventory",
|
||||||
@@ -171,6 +154,7 @@ namespace DatabaseSchema {
|
|||||||
"spell_buckets",
|
"spell_buckets",
|
||||||
"spell_globals",
|
"spell_globals",
|
||||||
"timers",
|
"timers",
|
||||||
|
"titles",
|
||||||
"trader",
|
"trader",
|
||||||
"trader_audit",
|
"trader_audit",
|
||||||
"zone_flags"
|
"zone_flags"
|
||||||
@@ -201,21 +185,19 @@ namespace DatabaseSchema {
|
|||||||
"char_create_point_allocations",
|
"char_create_point_allocations",
|
||||||
"damageshieldtypes",
|
"damageshieldtypes",
|
||||||
"doors",
|
"doors",
|
||||||
"dynamic_zone_templates",
|
|
||||||
"faction_association",
|
|
||||||
"faction_base_data",
|
"faction_base_data",
|
||||||
"faction_list",
|
"faction_list",
|
||||||
"faction_list_mod",
|
"faction_list_mod",
|
||||||
"fishing",
|
"fishing",
|
||||||
"forage",
|
"forage",
|
||||||
"global_loot",
|
"global_loot",
|
||||||
|
"goallists",
|
||||||
"graveyard",
|
"graveyard",
|
||||||
"grid",
|
"grid",
|
||||||
"grid_entries",
|
"grid_entries",
|
||||||
"ground_spawns",
|
"ground_spawns",
|
||||||
"horses",
|
"horses",
|
||||||
"items",
|
"items",
|
||||||
"items_evolving_details",
|
|
||||||
"ldon_trap_entries",
|
"ldon_trap_entries",
|
||||||
"ldon_trap_templates",
|
"ldon_trap_templates",
|
||||||
"lootdrop",
|
"lootdrop",
|
||||||
@@ -235,9 +217,9 @@ namespace DatabaseSchema {
|
|||||||
"npc_types_tint",
|
"npc_types_tint",
|
||||||
"object",
|
"object",
|
||||||
"pets",
|
"pets",
|
||||||
"pets_beastlord_data",
|
|
||||||
"pets_equipmentset",
|
"pets_equipmentset",
|
||||||
"pets_equipmentset_entries",
|
"pets_equipmentset_entries",
|
||||||
|
"proximities",
|
||||||
"skill_caps",
|
"skill_caps",
|
||||||
"spawn2",
|
"spawn2",
|
||||||
"spawn_conditions",
|
"spawn_conditions",
|
||||||
@@ -269,9 +251,7 @@ namespace DatabaseSchema {
|
|||||||
{
|
{
|
||||||
return {
|
return {
|
||||||
"chatchannels",
|
"chatchannels",
|
||||||
"chatchannel_reserved_names",
|
|
||||||
"command_settings",
|
"command_settings",
|
||||||
"command_subsettings",
|
|
||||||
"content_flags",
|
"content_flags",
|
||||||
"db_str",
|
"db_str",
|
||||||
"eqtime",
|
"eqtime",
|
||||||
@@ -285,7 +265,6 @@ namespace DatabaseSchema {
|
|||||||
"perl_event_export_settings",
|
"perl_event_export_settings",
|
||||||
"profanity_list",
|
"profanity_list",
|
||||||
"rule_sets",
|
"rule_sets",
|
||||||
"titles",
|
|
||||||
"rule_values",
|
"rule_values",
|
||||||
"variables",
|
"variables",
|
||||||
};
|
};
|
||||||
@@ -330,26 +309,20 @@ namespace DatabaseSchema {
|
|||||||
"banned_ips",
|
"banned_ips",
|
||||||
"bug_reports",
|
"bug_reports",
|
||||||
"bugs",
|
"bugs",
|
||||||
"buyer",
|
|
||||||
"buyer_buy_lines",
|
|
||||||
"buyer_trade_items",
|
|
||||||
"completed_shared_task_activity_state",
|
|
||||||
"completed_shared_task_members",
|
|
||||||
"completed_shared_tasks",
|
|
||||||
"discord_webhooks",
|
|
||||||
"dynamic_zone_members",
|
"dynamic_zone_members",
|
||||||
"dynamic_zones",
|
"dynamic_zones",
|
||||||
|
"eventlog",
|
||||||
"expedition_lockouts",
|
"expedition_lockouts",
|
||||||
"expeditions",
|
"expeditions",
|
||||||
"gm_ips",
|
"gm_ips",
|
||||||
"group_id",
|
"group_id",
|
||||||
"group_leaders",
|
"group_leaders",
|
||||||
"instance_list",
|
"hackers",
|
||||||
"ip_exemptions",
|
"ip_exemptions",
|
||||||
|
"instance_list",
|
||||||
|
"item_tick",
|
||||||
"lfguild",
|
"lfguild",
|
||||||
"merc_buffs",
|
|
||||||
"merchantlist_temp",
|
"merchantlist_temp",
|
||||||
"mercs",
|
|
||||||
"object_contents",
|
"object_contents",
|
||||||
"raid_details",
|
"raid_details",
|
||||||
"raid_leaders",
|
"raid_leaders",
|
||||||
@@ -358,13 +331,6 @@ namespace DatabaseSchema {
|
|||||||
"respawn_times",
|
"respawn_times",
|
||||||
"saylink",
|
"saylink",
|
||||||
"server_scheduled_events",
|
"server_scheduled_events",
|
||||||
"spawn2_disabled",
|
|
||||||
"player_event_log_settings",
|
|
||||||
"player_event_logs",
|
|
||||||
"shared_task_activity_state",
|
|
||||||
"shared_task_dynamic_zones",
|
|
||||||
"shared_task_members",
|
|
||||||
"shared_tasks",
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -397,57 +363,6 @@ namespace DatabaseSchema {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @description Gets all player bot tables
|
|
||||||
* @note These tables have no content in the PEQ daily dump
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
static std::vector<std::string> GetBotTables()
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
"bot_buffs",
|
|
||||||
"bot_command_settings",
|
|
||||||
"bot_create_combinations",
|
|
||||||
"bot_data",
|
|
||||||
"bot_heal_rotation_members",
|
|
||||||
"bot_heal_rotation_targets",
|
|
||||||
"bot_heal_rotations",
|
|
||||||
"bot_inspect_messages",
|
|
||||||
"bot_inventories",
|
|
||||||
"bot_owner_options",
|
|
||||||
"bot_pet_buffs",
|
|
||||||
"bot_pet_inventories",
|
|
||||||
"bot_pets",
|
|
||||||
"bot_spell_casting_chances",
|
|
||||||
"bot_spell_settings",
|
|
||||||
"bot_spells_entries",
|
|
||||||
"bot_stances",
|
|
||||||
"bot_timers"
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::vector<std::string> GetMercTables()
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
"merc_armorinfo",
|
|
||||||
"merc_inventory",
|
|
||||||
"merc_merchant_entries",
|
|
||||||
"merc_merchant_template_entries",
|
|
||||||
"merc_merchant_templates",
|
|
||||||
"merc_name_types",
|
|
||||||
"merc_npc_types",
|
|
||||||
"merc_spell_list_entries",
|
|
||||||
"merc_spell_lists",
|
|
||||||
"merc_stance_entries",
|
|
||||||
"merc_stats",
|
|
||||||
"merc_subtypes",
|
|
||||||
"merc_templates",
|
|
||||||
"merc_types",
|
|
||||||
"merc_weaponinfo"
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif //EQEMU_DATABASE_SCHEMA_H
|
#endif //EQEMU_DATABASE_SCHEMA_H
|
||||||
|
|||||||
+53
-196
@@ -7,13 +7,12 @@
|
|||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
|
||||||
#include "dbcore.h"
|
#include "dbcore.h"
|
||||||
#include "mysql_stmt.h"
|
|
||||||
|
|
||||||
|
#include <errmsg.h>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <mysqld_error.h>
|
#include <mysqld_error.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include "strings.h"
|
|
||||||
|
|
||||||
#ifdef _WINDOWS
|
#ifdef _WINDOWS
|
||||||
#define snprintf _snprintf
|
#define snprintf _snprintf
|
||||||
@@ -35,16 +34,14 @@
|
|||||||
|
|
||||||
DBcore::DBcore()
|
DBcore::DBcore()
|
||||||
{
|
{
|
||||||
mysql = mysql_init(nullptr);
|
mysql_init(&mysql);
|
||||||
mysqlOwner = true;
|
pHost = nullptr;
|
||||||
pHost = nullptr;
|
pUser = nullptr;
|
||||||
pUser = nullptr;
|
pPassword = nullptr;
|
||||||
pPassword = nullptr;
|
pDatabase = nullptr;
|
||||||
pDatabase = nullptr;
|
pCompress = false;
|
||||||
pCompress = false;
|
pSSL = false;
|
||||||
pSSL = false;
|
pStatus = Closed;
|
||||||
pStatus = Closed;
|
|
||||||
m_mutex = new Mutex;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DBcore::~DBcore()
|
DBcore::~DBcore()
|
||||||
@@ -54,10 +51,16 @@ DBcore::~DBcore()
|
|||||||
* are re-using the default database connection pointer when we dont have an
|
* are re-using the default database connection pointer when we dont have an
|
||||||
* external configuration setup ex: (content_database)
|
* external configuration setup ex: (content_database)
|
||||||
*/
|
*/
|
||||||
if (mysqlOwner) {
|
std::string mysql_connection_host;
|
||||||
mysql_close(mysql);
|
if (mysql.host) {
|
||||||
|
mysql_connection_host = mysql.host;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (GetOriginHost() != mysql_connection_host) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mysql_close(&mysql);
|
||||||
safe_delete_array(pHost);
|
safe_delete_array(pHost);
|
||||||
safe_delete_array(pUser);
|
safe_delete_array(pUser);
|
||||||
safe_delete_array(pPassword);
|
safe_delete_array(pPassword);
|
||||||
@@ -67,21 +70,20 @@ DBcore::~DBcore()
|
|||||||
// Sends the MySQL server a keepalive
|
// Sends the MySQL server a keepalive
|
||||||
void DBcore::ping()
|
void DBcore::ping()
|
||||||
{
|
{
|
||||||
if (!m_mutex->trylock()) {
|
if (!MDatabase.trylock()) {
|
||||||
// well, if's it's locked, someone's using it. If someone's using it, it doesnt need a keepalive
|
// well, if's it's locked, someone's using it. If someone's using it, it doesnt need a keepalive
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
mysql_ping(mysql);
|
mysql_ping(&mysql);
|
||||||
m_mutex->unlock();
|
MDatabase.unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
MySQLRequestResult DBcore::QueryDatabase(const std::string& query, bool retryOnFailureOnce)
|
MySQLRequestResult DBcore::QueryDatabase(std::string query, bool retryOnFailureOnce)
|
||||||
{
|
{
|
||||||
auto r = QueryDatabase(query.c_str(), query.length(), retryOnFailureOnce);
|
return QueryDatabase(query.c_str(), query.length(), retryOnFailureOnce);
|
||||||
return r;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DBcore::DoesTableExist(const std::string& table_name)
|
bool DBcore::DoesTableExist(std::string table_name)
|
||||||
{
|
{
|
||||||
auto results = QueryDatabase(fmt::format("SHOW TABLES LIKE '{}'", table_name));
|
auto results = QueryDatabase(fmt::format("SHOW TABLES LIKE '{}'", table_name));
|
||||||
|
|
||||||
@@ -93,16 +95,18 @@ MySQLRequestResult DBcore::QueryDatabase(const char *query, uint32 querylen, boo
|
|||||||
BenchTimer timer;
|
BenchTimer timer;
|
||||||
timer.reset();
|
timer.reset();
|
||||||
|
|
||||||
LockMutex lock(m_mutex);
|
LockMutex lock(&MDatabase);
|
||||||
|
|
||||||
// Reconnect if we are not connected before hand.
|
// Reconnect if we are not connected before hand.
|
||||||
if (pStatus != Connected) {
|
if (pStatus != Connected) {
|
||||||
Open();
|
Open();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// request query. != 0 indicates some kind of error.
|
// request query. != 0 indicates some kind of error.
|
||||||
if (mysql_real_query(mysql, query, querylen) != 0) {
|
if (mysql_real_query(&mysql, query, querylen) != 0) {
|
||||||
unsigned int errorNumber = mysql_errno(mysql);
|
unsigned int errorNumber = mysql_errno(&mysql);
|
||||||
|
|
||||||
if (errorNumber == CR_SERVER_GONE_ERROR) {
|
if (errorNumber == CR_SERVER_GONE_ERROR) {
|
||||||
pStatus = Error;
|
pStatus = Error;
|
||||||
@@ -126,26 +130,26 @@ MySQLRequestResult DBcore::QueryDatabase(const char *query, uint32 querylen, boo
|
|||||||
|
|
||||||
auto errorBuffer = new char[MYSQL_ERRMSG_SIZE];
|
auto errorBuffer = new char[MYSQL_ERRMSG_SIZE];
|
||||||
|
|
||||||
snprintf(errorBuffer, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(mysql), mysql_error(mysql));
|
snprintf(errorBuffer, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(&mysql), mysql_error(&mysql));
|
||||||
|
|
||||||
return MySQLRequestResult(nullptr, 0, 0, 0, 0, (uint32) mysql_errno(mysql), errorBuffer);
|
return MySQLRequestResult(nullptr, 0, 0, 0, 0, (uint32) mysql_errno(&mysql), errorBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto errorBuffer = new char[MYSQL_ERRMSG_SIZE];
|
auto errorBuffer = new char[MYSQL_ERRMSG_SIZE];
|
||||||
snprintf(errorBuffer, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(mysql), mysql_error(mysql));
|
snprintf(errorBuffer, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(&mysql), mysql_error(&mysql));
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error logging
|
* Error logging
|
||||||
*/
|
*/
|
||||||
if (mysql_errno(mysql) > 0 && query[0] != '\0') {
|
if (mysql_errno(&mysql) > 0 && strlen(query) > 0) {
|
||||||
LogMySQLError("MySQL Error ({}) [{}] Query [{}]", mysql_errno(mysql), mysql_error(mysql), query);
|
LogMySQLError("[{}] [{}]\n[{}]", mysql_errno(&mysql), mysql_error(&mysql), query);
|
||||||
}
|
}
|
||||||
|
|
||||||
return MySQLRequestResult(nullptr, 0, 0, 0, 0, mysql_errno(mysql), errorBuffer);
|
return MySQLRequestResult(nullptr, 0, 0, 0, 0, mysql_errno(&mysql), errorBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
// successful query. get results.
|
// successful query. get results.
|
||||||
MYSQL_RES *res = mysql_store_result(mysql);
|
MYSQL_RES *res = mysql_store_result(&mysql);
|
||||||
uint32 rowCount = 0;
|
uint32 rowCount = 0;
|
||||||
|
|
||||||
if (res != nullptr) {
|
if (res != nullptr) {
|
||||||
@@ -154,16 +158,16 @@ MySQLRequestResult DBcore::QueryDatabase(const char *query, uint32 querylen, boo
|
|||||||
|
|
||||||
MySQLRequestResult requestResult(
|
MySQLRequestResult requestResult(
|
||||||
res,
|
res,
|
||||||
(uint32) mysql_affected_rows(mysql),
|
(uint32) mysql_affected_rows(&mysql),
|
||||||
rowCount,
|
rowCount,
|
||||||
(uint32) mysql_field_count(mysql),
|
(uint32) mysql_field_count(&mysql),
|
||||||
(uint32) mysql_insert_id(mysql)
|
(uint32) mysql_insert_id(&mysql)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (LogSys.log_settings[Logs::MySQLQuery].is_category_enabled == 1) {
|
if (LogSys.log_settings[Logs::MySQLQuery].is_category_enabled == 1) {
|
||||||
if ((strncasecmp(query, "select", 6) == 0)) {
|
if ((strncasecmp(query, "select", 6) == 0)) {
|
||||||
LogMySQLQuery(
|
LogMySQLQuery(
|
||||||
"{0} -- ({1} row{2} returned) ({3}s)",
|
"{0} ({1} row{2} returned) ({3}s)",
|
||||||
query,
|
query,
|
||||||
requestResult.RowCount(),
|
requestResult.RowCount(),
|
||||||
requestResult.RowCount() == 1 ? "" : "s",
|
requestResult.RowCount() == 1 ? "" : "s",
|
||||||
@@ -172,7 +176,7 @@ MySQLRequestResult DBcore::QueryDatabase(const char *query, uint32 querylen, boo
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
LogMySQLQuery(
|
LogMySQLQuery(
|
||||||
"{0} -- ({1} row{2} affected) ({3}s)",
|
"{0} ({1} row{2} affected) ({3}s)",
|
||||||
query,
|
query,
|
||||||
requestResult.RowsAffected(),
|
requestResult.RowsAffected(),
|
||||||
requestResult.RowsAffected() == 1 ? "" : "s",
|
requestResult.RowsAffected() == 1 ? "" : "s",
|
||||||
@@ -203,7 +207,7 @@ uint32 DBcore::DoEscapeString(char *tobuf, const char *frombuf, uint32 fromlen)
|
|||||||
{
|
{
|
||||||
// No good reason to lock the DB, we only need it in the first place to check char encoding.
|
// No good reason to lock the DB, we only need it in the first place to check char encoding.
|
||||||
// LockMutex lock(&MDatabase);
|
// LockMutex lock(&MDatabase);
|
||||||
return mysql_real_escape_string(mysql, tobuf, frombuf, fromlen);
|
return mysql_real_escape_string(&mysql, tobuf, frombuf, fromlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DBcore::Open(
|
bool DBcore::Open(
|
||||||
@@ -218,7 +222,7 @@ bool DBcore::Open(
|
|||||||
bool iSSL
|
bool iSSL
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
LockMutex lock(m_mutex);
|
LockMutex lock(&MDatabase);
|
||||||
safe_delete_array(pHost);
|
safe_delete_array(pHost);
|
||||||
safe_delete_array(pUser);
|
safe_delete_array(pUser);
|
||||||
safe_delete_array(pPassword);
|
safe_delete_array(pPassword);
|
||||||
@@ -238,13 +242,13 @@ bool DBcore::Open(uint32 *errnum, char *errbuf)
|
|||||||
if (errbuf) {
|
if (errbuf) {
|
||||||
errbuf[0] = 0;
|
errbuf[0] = 0;
|
||||||
}
|
}
|
||||||
LockMutex lock(m_mutex);
|
LockMutex lock(&MDatabase);
|
||||||
if (GetStatus() == Connected) {
|
if (GetStatus() == Connected) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (GetStatus() == Error) {
|
if (GetStatus() == Error) {
|
||||||
mysql_close(mysql);
|
mysql_close(&mysql);
|
||||||
mysql_init(mysql); // Initialize structure again
|
mysql_init(&mysql); // Initialize structure again
|
||||||
}
|
}
|
||||||
if (!pHost) {
|
if (!pHost) {
|
||||||
return false;
|
return false;
|
||||||
@@ -261,7 +265,7 @@ bool DBcore::Open(uint32 *errnum, char *errbuf)
|
|||||||
if (pSSL) {
|
if (pSSL) {
|
||||||
flags |= CLIENT_SSL;
|
flags |= CLIENT_SSL;
|
||||||
}
|
}
|
||||||
if (mysql_real_connect(mysql, pHost, pUser, pPassword, pDatabase, pPort, 0, flags)) {
|
if (mysql_real_connect(&mysql, pHost, pUser, pPassword, pDatabase, pPort, 0, flags)) {
|
||||||
pStatus = Connected;
|
pStatus = Connected;
|
||||||
|
|
||||||
std::string connected_origin_host = pHost;
|
std::string connected_origin_host = pHost;
|
||||||
@@ -271,16 +275,21 @@ bool DBcore::Open(uint32 *errnum, char *errbuf)
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (errnum) {
|
if (errnum) {
|
||||||
*errnum = mysql_errno(mysql);
|
*errnum = mysql_errno(&mysql);
|
||||||
}
|
}
|
||||||
if (errbuf) {
|
if (errbuf) {
|
||||||
snprintf(errbuf, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(mysql), mysql_error(mysql));
|
snprintf(errbuf, MYSQL_ERRMSG_SIZE, "#%i: %s", mysql_errno(&mysql), mysql_error(&mysql));
|
||||||
}
|
}
|
||||||
pStatus = Error;
|
pStatus = Error;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DBcore::SetMysql(MYSQL *mysql)
|
||||||
|
{
|
||||||
|
DBcore::mysql = *mysql;
|
||||||
|
}
|
||||||
|
|
||||||
const std::string &DBcore::GetOriginHost() const
|
const std::string &DBcore::GetOriginHost() const
|
||||||
{
|
{
|
||||||
return origin_host;
|
return origin_host;
|
||||||
@@ -290,155 +299,3 @@ void DBcore::SetOriginHost(const std::string &origin_host)
|
|||||||
{
|
{
|
||||||
DBcore::origin_host = origin_host;
|
DBcore::origin_host = origin_host;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string DBcore::Escape(const std::string& s)
|
|
||||||
{
|
|
||||||
const std::size_t s_len = s.length();
|
|
||||||
std::vector<char> temp((s_len * 2) + 1, '\0');
|
|
||||||
mysql_real_escape_string(mysql, temp.data(), s.c_str(), s_len);
|
|
||||||
|
|
||||||
return temp.data();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DBcore::SetMutex(Mutex *mutex)
|
|
||||||
{
|
|
||||||
safe_delete(m_mutex);
|
|
||||||
|
|
||||||
DBcore::m_mutex = mutex;
|
|
||||||
}
|
|
||||||
|
|
||||||
// executes multiple statements in one query
|
|
||||||
// do not use this in application logic
|
|
||||||
// this was built and maintained for database migrations only
|
|
||||||
MySQLRequestResult DBcore::QueryDatabaseMulti(const std::string &query)
|
|
||||||
{
|
|
||||||
SetMultiStatementsOn();
|
|
||||||
|
|
||||||
BenchTimer timer;
|
|
||||||
timer.reset();
|
|
||||||
|
|
||||||
LockMutex lock(m_mutex);
|
|
||||||
|
|
||||||
// Reconnect if we are not connected before hand.
|
|
||||||
if (pStatus != Connected) {
|
|
||||||
Open();
|
|
||||||
}
|
|
||||||
auto r = MySQLRequestResult{};
|
|
||||||
|
|
||||||
int status = mysql_real_query(mysql, query.c_str(), query.length());
|
|
||||||
|
|
||||||
// process single result
|
|
||||||
if (status != 0) {
|
|
||||||
unsigned int error_number = mysql_errno(mysql);
|
|
||||||
|
|
||||||
if (error_number == CR_SERVER_GONE_ERROR) {
|
|
||||||
pStatus = Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
// error logging
|
|
||||||
if (mysql_errno(mysql) > 0 && query.length() > 0 && mysql_errno(mysql) != 1065) {
|
|
||||||
std::string error_raw = fmt::format("{}", mysql_error(mysql));
|
|
||||||
std::string mysql_err = Strings::Trim(error_raw);
|
|
||||||
std::string clean_query = Strings::Replace(query, "\n", "");
|
|
||||||
LogMySQLError("[{}] ({}) query [{}]", mysql_err, mysql_errno(mysql), clean_query);
|
|
||||||
|
|
||||||
MYSQL_RES *res = mysql_store_result(mysql);
|
|
||||||
|
|
||||||
uint32 row_count = 0;
|
|
||||||
if (res) {
|
|
||||||
row_count = (uint32) mysql_num_rows(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
r = MySQLRequestResult(
|
|
||||||
res,
|
|
||||||
(uint32) mysql_affected_rows(mysql),
|
|
||||||
row_count,
|
|
||||||
(uint32) mysql_field_count(mysql),
|
|
||||||
(uint32) mysql_insert_id(mysql)
|
|
||||||
);
|
|
||||||
|
|
||||||
std::string error_message = mysql_error(mysql);
|
|
||||||
r.SetErrorMessage(error_message);
|
|
||||||
r.SetErrorNumber(mysql_errno(mysql));
|
|
||||||
|
|
||||||
if (res) {
|
|
||||||
mysql_free_result(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
SetMultiStatementsOff();
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int index = 0;
|
|
||||||
|
|
||||||
// there could be a query with a semicolon in the actual data, this is best effort for
|
|
||||||
// logging / display purposes
|
|
||||||
// rare that we see this when this is only used in DDL statements
|
|
||||||
auto pieces = Strings::Split(query, ";");
|
|
||||||
|
|
||||||
// process each statement result
|
|
||||||
do {
|
|
||||||
uint32 row_count = 0;
|
|
||||||
MYSQL_RES *res = mysql_store_result(mysql);
|
|
||||||
|
|
||||||
r = MySQLRequestResult(
|
|
||||||
res,
|
|
||||||
(uint32) mysql_affected_rows(mysql),
|
|
||||||
row_count,
|
|
||||||
(uint32) mysql_field_count(mysql),
|
|
||||||
(uint32) mysql_insert_id(mysql)
|
|
||||||
);
|
|
||||||
|
|
||||||
if (pieces.size() >= index) {
|
|
||||||
auto piece = pieces[index];
|
|
||||||
LogMySQLQuery(
|
|
||||||
"{} -- ({} row{} affected) ({}s)",
|
|
||||||
piece,
|
|
||||||
r.RowsAffected(),
|
|
||||||
r.RowsAffected() == 1 ? "" : "s",
|
|
||||||
std::to_string(timer.elapsed())
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (res) {
|
|
||||||
row_count = (uint32) mysql_num_rows(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
// more results? -1 = no, >0 = error, 0 = yes (keep looping)
|
|
||||||
if ((status = mysql_next_result(mysql)) > 0) {
|
|
||||||
if (mysql_errno(mysql) > 0) {
|
|
||||||
LogMySQLError("[{}] [{}]", mysql_errno(mysql), mysql_error(mysql));
|
|
||||||
}
|
|
||||||
|
|
||||||
mysql_free_result(res);
|
|
||||||
|
|
||||||
// error logging
|
|
||||||
std::string error_message = mysql_error(mysql);
|
|
||||||
r.SetErrorMessage(error_message);
|
|
||||||
r.SetErrorNumber(mysql_errno(mysql));
|
|
||||||
|
|
||||||
SetMultiStatementsOff();
|
|
||||||
|
|
||||||
// we handle errors elsewhere
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (res) {
|
|
||||||
mysql_free_result(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
index++;
|
|
||||||
} while (status == 0);
|
|
||||||
|
|
||||||
SetMultiStatementsOff();
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
mysql::PreparedStmt DBcore::Prepare(std::string query)
|
|
||||||
{
|
|
||||||
return mysql::PreparedStmt(*mysql, std::move(query), m_mutex);
|
|
||||||
}
|
|
||||||
|
|||||||
+7
-40
@@ -12,12 +12,6 @@
|
|||||||
|
|
||||||
#include <mysql.h>
|
#include <mysql.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
#define CR_SERVER_GONE_ERROR 2006
|
|
||||||
#define CR_SERVER_LOST 2013
|
|
||||||
|
|
||||||
namespace mysql { class PreparedStmt; }
|
|
||||||
|
|
||||||
class DBcore {
|
class DBcore {
|
||||||
public:
|
public:
|
||||||
@@ -29,31 +23,19 @@ public:
|
|||||||
~DBcore();
|
~DBcore();
|
||||||
eStatus GetStatus() { return pStatus; }
|
eStatus GetStatus() { return pStatus; }
|
||||||
MySQLRequestResult QueryDatabase(const char *query, uint32 querylen, bool retryOnFailureOnce = true);
|
MySQLRequestResult QueryDatabase(const char *query, uint32 querylen, bool retryOnFailureOnce = true);
|
||||||
MySQLRequestResult QueryDatabase(const std::string& query, bool retryOnFailureOnce = true);
|
MySQLRequestResult QueryDatabase(std::string query, bool retryOnFailureOnce = true);
|
||||||
MySQLRequestResult QueryDatabaseMulti(const std::string &query);
|
|
||||||
void TransactionBegin();
|
void TransactionBegin();
|
||||||
void TransactionCommit();
|
void TransactionCommit();
|
||||||
void TransactionRollback();
|
void TransactionRollback();
|
||||||
std::string Escape(const std::string& s);
|
|
||||||
uint32 DoEscapeString(char *tobuf, const char *frombuf, uint32 fromlen);
|
uint32 DoEscapeString(char *tobuf, const char *frombuf, uint32 fromlen);
|
||||||
void ping();
|
void ping();
|
||||||
|
MYSQL *getMySQL() { return &mysql; }
|
||||||
|
void SetMysql(MYSQL *mysql);
|
||||||
|
|
||||||
const std::string &GetOriginHost() const;
|
const std::string &GetOriginHost() const;
|
||||||
void SetOriginHost(const std::string &origin_host);
|
void SetOriginHost(const std::string &origin_host);
|
||||||
|
|
||||||
bool DoesTableExist(const std::string& table_name);
|
bool DoesTableExist(std::string table_name);
|
||||||
|
|
||||||
void SetMySQL(const DBcore &o)
|
|
||||||
{
|
|
||||||
mysql = o.mysql;
|
|
||||||
mysqlOwner = false;
|
|
||||||
}
|
|
||||||
void SetMutex(Mutex *mutex);
|
|
||||||
|
|
||||||
// only safe on connections shared with other threads if results buffered
|
|
||||||
// unsafe to use off main thread due to internal server logging
|
|
||||||
// throws std::runtime_error on failure
|
|
||||||
mysql::PreparedStmt Prepare(std::string query);
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool Open(
|
bool Open(
|
||||||
@@ -71,13 +53,10 @@ protected:
|
|||||||
private:
|
private:
|
||||||
bool Open(uint32 *errnum = nullptr, char *errbuf = nullptr);
|
bool Open(uint32 *errnum = nullptr, char *errbuf = nullptr);
|
||||||
|
|
||||||
MYSQL* mysql;
|
MYSQL mysql;
|
||||||
bool mysqlOwner;
|
Mutex MDatabase;
|
||||||
Mutex *m_mutex;
|
|
||||||
eStatus pStatus;
|
eStatus pStatus;
|
||||||
|
|
||||||
std::mutex m_query_lock{};
|
|
||||||
|
|
||||||
std::string origin_host;
|
std::string origin_host;
|
||||||
|
|
||||||
char *pHost;
|
char *pHost;
|
||||||
@@ -88,20 +67,8 @@ private:
|
|||||||
uint32 pPort;
|
uint32 pPort;
|
||||||
bool pSSL;
|
bool pSSL;
|
||||||
|
|
||||||
// allows multiple queries to be executed within the same query
|
|
||||||
// do not use this under normal operation
|
|
||||||
// we use this during database migrations only currently
|
|
||||||
void SetMultiStatementsOn()
|
|
||||||
{
|
|
||||||
mysql_set_server_option(mysql, MYSQL_OPTION_MULTI_STATEMENTS_ON);
|
|
||||||
}
|
|
||||||
|
|
||||||
// disables multiple statements to be executed in one query
|
|
||||||
void SetMultiStatementsOff()
|
|
||||||
{
|
|
||||||
mysql_set_server_option(mysql, MYSQL_OPTION_MULTI_STATEMENTS_OFF);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
+121
-7
@@ -11,7 +11,7 @@
|
|||||||
are required to give you total support for your newly bought product;
|
are required to give you total support for your newly bought product;
|
||||||
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||||||
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
@@ -19,17 +19,131 @@
|
|||||||
|
|
||||||
#include "deity.h"
|
#include "deity.h"
|
||||||
|
|
||||||
uint32 Deity::GetBitmask(uint32 deity_id)
|
|
||||||
|
EQ::deity::DeityTypeBit EQ::deity::ConvertDeityTypeToDeityTypeBit(DeityType deity_type)
|
||||||
{
|
{
|
||||||
return IsValid(deity_id) ? deity_bitmasks[deity_id] : Deity::Bitmask::All;
|
switch (deity_type) {
|
||||||
|
case DeityBertoxxulous:
|
||||||
|
return bit_DeityBertoxxulous;
|
||||||
|
case DeityBrellSirilis:
|
||||||
|
return bit_DeityBrellSirilis;
|
||||||
|
case DeityCazicThule:
|
||||||
|
return bit_DeityCazicThule;
|
||||||
|
case DeityErollisiMarr:
|
||||||
|
return bit_DeityErollisiMarr;
|
||||||
|
case DeityBristlebane:
|
||||||
|
return bit_DeityBristlebane;
|
||||||
|
case DeityInnoruuk:
|
||||||
|
return bit_DeityInnoruuk;
|
||||||
|
case DeityKarana:
|
||||||
|
return bit_DeityKarana;
|
||||||
|
case DeityMithanielMarr:
|
||||||
|
return bit_DeityMithanielMarr;
|
||||||
|
case DeityPrexus:
|
||||||
|
return bit_DeityPrexus;
|
||||||
|
case DeityQuellious:
|
||||||
|
return bit_DeityQuellious;
|
||||||
|
case DeityRallosZek:
|
||||||
|
return bit_DeityRallosZek;
|
||||||
|
case DeityRodcetNife:
|
||||||
|
return bit_DeityRodcetNife;
|
||||||
|
case DeitySolusekRo:
|
||||||
|
return bit_DeitySolusekRo;
|
||||||
|
case DeityTheTribunal:
|
||||||
|
return bit_DeityTheTribunal;
|
||||||
|
case DeityTunare:
|
||||||
|
return bit_DeityTunare;
|
||||||
|
case DeityVeeshan:
|
||||||
|
return bit_DeityVeeshan;
|
||||||
|
case DeityAgnostic_LB:
|
||||||
|
case DeityAgnostic:
|
||||||
|
return bit_DeityAgnostic;
|
||||||
|
default:
|
||||||
|
return bit_DeityAll;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string Deity::GetName(uint32 deity_id)
|
EQ::deity::DeityType EQ::deity::ConvertDeityTypeBitToDeityType(DeityTypeBit deity_type_bit)
|
||||||
{
|
{
|
||||||
return IsValid(deity_id) ? deity_names[deity_id] : "UNKNOWN DEITY";
|
switch (deity_type_bit) {
|
||||||
|
case bit_DeityAgnostic:
|
||||||
|
return DeityAgnostic;
|
||||||
|
case bit_DeityBertoxxulous:
|
||||||
|
return DeityBertoxxulous;
|
||||||
|
case bit_DeityBrellSirilis:
|
||||||
|
return DeityBrellSirilis;
|
||||||
|
case bit_DeityCazicThule:
|
||||||
|
return DeityCazicThule;
|
||||||
|
case bit_DeityErollisiMarr:
|
||||||
|
return DeityErollisiMarr;
|
||||||
|
case bit_DeityBristlebane:
|
||||||
|
return DeityBristlebane;
|
||||||
|
case bit_DeityInnoruuk:
|
||||||
|
return DeityInnoruuk;
|
||||||
|
case bit_DeityKarana:
|
||||||
|
return DeityKarana;
|
||||||
|
case bit_DeityMithanielMarr:
|
||||||
|
return DeityMithanielMarr;
|
||||||
|
case bit_DeityPrexus:
|
||||||
|
return DeityPrexus;
|
||||||
|
case bit_DeityQuellious:
|
||||||
|
return DeityQuellious;
|
||||||
|
case bit_DeityRallosZek:
|
||||||
|
return DeityRallosZek;
|
||||||
|
case bit_DeityRodcetNife:
|
||||||
|
return DeityRodcetNife;
|
||||||
|
case bit_DeitySolusekRo:
|
||||||
|
return DeitySolusekRo;
|
||||||
|
case bit_DeityTheTribunal:
|
||||||
|
return DeityTheTribunal;
|
||||||
|
case bit_DeityTunare:
|
||||||
|
return DeityTunare;
|
||||||
|
case bit_DeityVeeshan:
|
||||||
|
return DeityVeeshan;
|
||||||
|
default:
|
||||||
|
return DeityUnknown;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Deity::IsValid(uint32 deity_id)
|
const char* EQ::deity::DeityName(DeityType deity_type)
|
||||||
{
|
{
|
||||||
return deity_names.find(deity_id) != deity_names.end();
|
switch (deity_type) {
|
||||||
|
case DeityBertoxxulous:
|
||||||
|
return "Bertoxxulous";
|
||||||
|
case DeityBrellSirilis:
|
||||||
|
return "Brell Serilis";
|
||||||
|
case DeityCazicThule:
|
||||||
|
return "Cazic-Thule";
|
||||||
|
case DeityErollisiMarr:
|
||||||
|
return "Erollisi Marr";
|
||||||
|
case DeityBristlebane:
|
||||||
|
return "Bristlebane";
|
||||||
|
case DeityInnoruuk:
|
||||||
|
return "Innoruuk";
|
||||||
|
case DeityKarana:
|
||||||
|
return "Karana";
|
||||||
|
case DeityMithanielMarr:
|
||||||
|
return "Mithaniel Marr";
|
||||||
|
case DeityPrexus:
|
||||||
|
return "Prexus";
|
||||||
|
case DeityQuellious:
|
||||||
|
return "Quellious";
|
||||||
|
case DeityRallosZek:
|
||||||
|
return "Rallos Zek";
|
||||||
|
case DeityRodcetNife:
|
||||||
|
return "Rodcet Nife";
|
||||||
|
case DeitySolusekRo:
|
||||||
|
return "Solusek Ro";
|
||||||
|
case DeityTheTribunal:
|
||||||
|
return "The Tribunal";
|
||||||
|
case DeityTunare:
|
||||||
|
return "Tunare";
|
||||||
|
case DeityVeeshan:
|
||||||
|
return "Veeshan";
|
||||||
|
case DeityAgnostic_LB:
|
||||||
|
case DeityAgnostic:
|
||||||
|
return "Agnostic";
|
||||||
|
default:
|
||||||
|
return "Unknown";
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
+51
-87
@@ -21,97 +21,61 @@
|
|||||||
#define COMMON_DEITY_H
|
#define COMMON_DEITY_H
|
||||||
|
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
#include <map>
|
|
||||||
#include <string>
|
|
||||||
#include <limits>
|
|
||||||
|
|
||||||
namespace Deity {
|
|
||||||
constexpr uint32 Unknown = 0;
|
|
||||||
constexpr uint32 Agnostic1 = 140;
|
|
||||||
constexpr uint32 Bertoxxulous = 201;
|
|
||||||
constexpr uint32 BrellSirilis = 202;
|
|
||||||
constexpr uint32 CazicThule = 203;
|
|
||||||
constexpr uint32 ErollisiMarr = 204;
|
|
||||||
constexpr uint32 Bristlebane = 205;
|
|
||||||
constexpr uint32 Innoruuk = 206;
|
|
||||||
constexpr uint32 Karana = 207;
|
|
||||||
constexpr uint32 MithanielMarr = 208;
|
|
||||||
constexpr uint32 Prexus = 209;
|
|
||||||
constexpr uint32 Quellious = 210;
|
|
||||||
constexpr uint32 RallosZek = 211;
|
|
||||||
constexpr uint32 RodcetNife = 212;
|
|
||||||
constexpr uint32 SolusekRo = 213;
|
|
||||||
constexpr uint32 TheTribunal = 214;
|
|
||||||
constexpr uint32 Tunare = 215;
|
|
||||||
constexpr uint32 Veeshan = 216;
|
|
||||||
constexpr uint32 Agnostic2 = 396;
|
|
||||||
|
|
||||||
namespace Bitmask {
|
namespace EQ
|
||||||
constexpr uint32 Agnostic = 1;
|
{
|
||||||
constexpr uint32 Bertoxxulous = 2;
|
namespace deity {
|
||||||
constexpr uint32 BrellSirilis = 4;
|
enum DeityType {
|
||||||
constexpr uint32 CazicThule = 8;
|
DeityUnknown = 0,
|
||||||
constexpr uint32 ErollisiMarr = 16;
|
DeityAgnostic_LB = 140,
|
||||||
constexpr uint32 Bristlebane = 32;
|
DeityBertoxxulous = 201,
|
||||||
constexpr uint32 Innoruuk = 64;
|
DeityBrellSirilis,
|
||||||
constexpr uint32 Karana = 128;
|
DeityCazicThule,
|
||||||
constexpr uint32 MithanielMarr = 256;
|
DeityErollisiMarr,
|
||||||
constexpr uint32 Prexus = 512;
|
DeityBristlebane,
|
||||||
constexpr uint32 Quellious = 1024;
|
DeityInnoruuk,
|
||||||
constexpr uint32 RallosZek = 2048;
|
DeityKarana,
|
||||||
constexpr uint32 RodcetNife = 4096;
|
DeityMithanielMarr,
|
||||||
constexpr uint32 SolusekRo = 8192;
|
DeityPrexus,
|
||||||
constexpr uint32 TheTribunal = 16384;
|
DeityQuellious,
|
||||||
constexpr uint32 Tunare = 32768;
|
DeityRallosZek,
|
||||||
constexpr uint32 Veeshan = 65536;
|
DeityRodcetNife,
|
||||||
constexpr uint32 All = std::numeric_limits<uint32>::max();
|
DeitySolusekRo,
|
||||||
}
|
DeityTheTribunal,
|
||||||
|
DeityTunare,
|
||||||
|
DeityVeeshan,
|
||||||
|
DeityAgnostic = 396
|
||||||
|
};
|
||||||
|
|
||||||
uint32 GetBitmask(uint32 deity_id);
|
enum DeityTypeBit : uint32 {
|
||||||
std::string GetName(uint32 deity_id);
|
bit_DeityNone = 0x00000000,
|
||||||
bool IsValid(uint32 deity_id);
|
bit_DeityAgnostic = 0x00000001,
|
||||||
}
|
bit_DeityBertoxxulous = 0x00000002,
|
||||||
|
bit_DeityBrellSirilis = 0x00000004,
|
||||||
|
bit_DeityCazicThule = 0x00000008,
|
||||||
|
bit_DeityErollisiMarr = 0x00000010,
|
||||||
|
bit_DeityBristlebane = 0x00000020,
|
||||||
|
bit_DeityInnoruuk = 0x00000040,
|
||||||
|
bit_DeityKarana = 0x00000080,
|
||||||
|
bit_DeityMithanielMarr = 0x00000100,
|
||||||
|
bit_DeityPrexus = 0x00000200,
|
||||||
|
bit_DeityQuellious = 0x00000400,
|
||||||
|
bit_DeityRallosZek = 0x00000800,
|
||||||
|
bit_DeityRodcetNife = 0x00001000,
|
||||||
|
bit_DeitySolusekRo = 0x00002000,
|
||||||
|
bit_DeityTheTribunal = 0x00004000,
|
||||||
|
bit_DeityTunare = 0x00008000,
|
||||||
|
bit_DeityVeeshan = 0x00010000,
|
||||||
|
bit_DeityAll = 0xFFFFFFFF
|
||||||
|
};
|
||||||
|
|
||||||
static std::map<uint32, std::string> deity_names = {
|
extern DeityTypeBit ConvertDeityTypeToDeityTypeBit(DeityType deity_type);
|
||||||
{ Deity::Agnostic1, "Agnostic" },
|
extern DeityType ConvertDeityTypeBitToDeityType(DeityTypeBit deity_type_bit);
|
||||||
{ Deity::Agnostic2, "Agnostic" },
|
extern const char* DeityName(DeityType deity_type);
|
||||||
{ Deity::Bertoxxulous, "Bertoxxulous" },
|
|
||||||
{ Deity::BrellSirilis, "Brell Serilis" },
|
|
||||||
{ Deity::Bristlebane, "Bristlebane" },
|
|
||||||
{ Deity::CazicThule, "Cazic-Thule" },
|
|
||||||
{ Deity::ErollisiMarr, "Erollisi Marr" },
|
|
||||||
{ Deity::Innoruuk, "Innoruuk" },
|
|
||||||
{ Deity::Karana, "Karana" },
|
|
||||||
{ Deity::MithanielMarr, "Mithaniel Marr" },
|
|
||||||
{ Deity::Prexus, "Prexus" },
|
|
||||||
{ Deity::Quellious, "Quellious" },
|
|
||||||
{ Deity::RallosZek, "Rallos Zek" },
|
|
||||||
{ Deity::RodcetNife, "Rodcet Nife" },
|
|
||||||
{ Deity::SolusekRo, "Solusek Ro" },
|
|
||||||
{ Deity::TheTribunal, "The Tribunal" },
|
|
||||||
{ Deity::Tunare, "Tunare" },
|
|
||||||
{ Deity::Veeshan, "Veeshan" }
|
|
||||||
};
|
|
||||||
|
|
||||||
static std::map<uint32, uint32> deity_bitmasks = {
|
} /*deity*/
|
||||||
{ Deity::Agnostic1, Deity::Bitmask::Agnostic },
|
|
||||||
{ Deity::Agnostic2, Deity::Bitmask::Agnostic },
|
} /*EQEmu*/
|
||||||
{ Deity::Bertoxxulous, Deity::Bitmask::Bertoxxulous },
|
|
||||||
{ Deity::BrellSirilis, Deity::Bitmask::BrellSirilis },
|
|
||||||
{ Deity::CazicThule, Deity::Bitmask::CazicThule },
|
|
||||||
{ Deity::ErollisiMarr, Deity::Bitmask::ErollisiMarr },
|
|
||||||
{ Deity::Bristlebane, Deity::Bitmask::Bristlebane },
|
|
||||||
{ Deity::Innoruuk, Deity::Bitmask::Innoruuk },
|
|
||||||
{ Deity::Karana, Deity::Bitmask::Karana },
|
|
||||||
{ Deity::MithanielMarr, Deity::Bitmask::MithanielMarr },
|
|
||||||
{ Deity::Prexus, Deity::Bitmask::Prexus },
|
|
||||||
{ Deity::Quellious, Deity::Bitmask::Quellious },
|
|
||||||
{ Deity::RallosZek, Deity::Bitmask::RallosZek },
|
|
||||||
{ Deity::RodcetNife, Deity::Bitmask::RodcetNife },
|
|
||||||
{ Deity::SolusekRo, Deity::Bitmask::SolusekRo },
|
|
||||||
{ Deity::TheTribunal, Deity::Bitmask::TheTribunal },
|
|
||||||
{ Deity::Tunare, Deity::Bitmask::Tunare },
|
|
||||||
{ Deity::Veeshan, Deity::Bitmask::Veeshan }
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif /* COMMON_DEITY_H */
|
#endif /* COMMON_DEITY_H */
|
||||||
|
|||||||
@@ -1,165 +0,0 @@
|
|||||||
#include <cereal/archives/json.hpp>
|
|
||||||
#include <cereal/archives/binary.hpp>
|
|
||||||
#include "discord.h"
|
|
||||||
#include "../http/httplib.h"
|
|
||||||
#include "../json/json.h"
|
|
||||||
#include "../strings.h"
|
|
||||||
#include "../eqemu_logsys.h"
|
|
||||||
#include "../events/player_event_logs.h"
|
|
||||||
|
|
||||||
constexpr int MAX_RETRIES = 10;
|
|
||||||
|
|
||||||
void Discord::SendWebhookMessage(const std::string &message, const std::string &webhook_url)
|
|
||||||
{
|
|
||||||
if (!ValidateWebhookUrl(webhook_url)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// split
|
|
||||||
auto s = Strings::Split(webhook_url, '/');
|
|
||||||
|
|
||||||
// url
|
|
||||||
std::string base_url = fmt::format("{}//{}", s[0], s[2]);
|
|
||||||
std::string endpoint = Strings::Replace(webhook_url, base_url, "");
|
|
||||||
|
|
||||||
// client
|
|
||||||
httplib::Client cli(base_url);
|
|
||||||
cli.set_connection_timeout(0, 15000000); // 15 sec
|
|
||||||
cli.set_read_timeout(15, 0); // 15 seconds
|
|
||||||
cli.set_write_timeout(15, 0); // 15 seconds
|
|
||||||
|
|
||||||
// payload
|
|
||||||
Json::Value p;
|
|
||||||
p["content"] = message;
|
|
||||||
std::stringstream payload;
|
|
||||||
payload << p;
|
|
||||||
|
|
||||||
bool retry = true;
|
|
||||||
int retries = 0;
|
|
||||||
int retry_timer = 1000;
|
|
||||||
while (retry) {
|
|
||||||
if (auto res = cli.Post(endpoint, payload.str(), "application/json")) {
|
|
||||||
if (res->status != 200 && res->status != 204) {
|
|
||||||
LogError("[Discord Client] Code [{}] Error [{}]", res->status, res->body);
|
|
||||||
}
|
|
||||||
if (res->status == 429) {
|
|
||||||
if (!res->body.empty()) {
|
|
||||||
std::stringstream ss(res->body);
|
|
||||||
Json::Value response;
|
|
||||||
|
|
||||||
try {
|
|
||||||
ss >> response;
|
|
||||||
}
|
|
||||||
catch (std::exception const &ex) {
|
|
||||||
LogDiscord("JSON serialization failure [{}] via [{}]", ex.what(), res->body);
|
|
||||||
}
|
|
||||||
|
|
||||||
retry_timer = Strings::ToInt(response["retry_after"].asString()) + 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
LogDiscord("Rate limited... retrying message in [{}ms]", retry_timer);
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(retry_timer + 500));
|
|
||||||
}
|
|
||||||
if (res->status == 204) {
|
|
||||||
retry = false;
|
|
||||||
}
|
|
||||||
if (retries > MAX_RETRIES) {
|
|
||||||
LogDiscord("Retries exceeded for message [{}]", message);
|
|
||||||
retry = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
retries++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Discord::SendPlayerEventMessage(
|
|
||||||
const PlayerEvent::PlayerEventContainer &e,
|
|
||||||
const std::string &webhook_url
|
|
||||||
)
|
|
||||||
{
|
|
||||||
if (!ValidateWebhookUrl(webhook_url)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto s = Strings::Split(webhook_url, '/');
|
|
||||||
|
|
||||||
// url
|
|
||||||
std::string base_url = fmt::format("{}//{}", s[0], s[2]);
|
|
||||||
std::string endpoint = Strings::Replace(webhook_url, base_url, "");
|
|
||||||
|
|
||||||
// client
|
|
||||||
httplib::Client cli(base_url);
|
|
||||||
cli.set_connection_timeout(0, 15000000); // 15 sec
|
|
||||||
cli.set_read_timeout(15, 0); // 15 seconds
|
|
||||||
cli.set_write_timeout(15, 0); // 15 seconds
|
|
||||||
|
|
||||||
std::string payload = PlayerEventLogs::GetDiscordPayloadFromEvent(e);
|
|
||||||
if (payload.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool retry = true;
|
|
||||||
int retries = 0;
|
|
||||||
int retry_timer = 1000;
|
|
||||||
while (retry) {
|
|
||||||
if (auto res = cli.Post(endpoint, payload, "application/json")) {
|
|
||||||
if (res->status != 200 && res->status != 204) {
|
|
||||||
LogError("Code [{}] Error [{}]", res->status, res->body);
|
|
||||||
}
|
|
||||||
if (res->status == 429) {
|
|
||||||
if (!res->body.empty()) {
|
|
||||||
std::stringstream ss(res->body);
|
|
||||||
Json::Value response;
|
|
||||||
|
|
||||||
try {
|
|
||||||
ss >> response;
|
|
||||||
}
|
|
||||||
catch (std::exception const &ex) {
|
|
||||||
LogDiscord("JSON serialization failure [{}] via [{}]", ex.what(), res->body);
|
|
||||||
}
|
|
||||||
|
|
||||||
retry_timer = Strings::ToInt(response["retry_after"].asString()) + 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
LogDiscord("Rate limited... retrying message in [{}ms]", retry_timer);
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(retry_timer + 500));
|
|
||||||
}
|
|
||||||
if (res->status == 204) {
|
|
||||||
retry = false;
|
|
||||||
}
|
|
||||||
if (retries > MAX_RETRIES) {
|
|
||||||
LogDiscord("Retries exceeded for player event message");
|
|
||||||
retry = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
retries++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string Discord::FormatDiscordMessage(uint16 category_id, const std::string &message)
|
|
||||||
{
|
|
||||||
if (category_id == Logs::LogCategory::MySQLQuery) {
|
|
||||||
return fmt::format("```sql\n{}\n```", message);
|
|
||||||
}
|
|
||||||
|
|
||||||
return message + "\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Discord::ValidateWebhookUrl(const std::string &webhook_url)
|
|
||||||
{
|
|
||||||
// validate
|
|
||||||
if (webhook_url.empty()) {
|
|
||||||
LogDiscord("[webhook_url] is empty");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// validate
|
|
||||||
if (!Strings::Contains(webhook_url, "http://") && !Strings::Contains(webhook_url, "https://")) {
|
|
||||||
LogDiscord("[webhook_url] [{}] does not contain a valid http/s prefix.", webhook_url);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
#ifndef EQEMU_DISCORD_H
|
|
||||||
#define EQEMU_DISCORD_H
|
|
||||||
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include "../types.h"
|
|
||||||
#include "../http/httplib.h"
|
|
||||||
#include "../repositories/player_event_logs_repository.h"
|
|
||||||
#include "../events/player_events.h"
|
|
||||||
|
|
||||||
class Discord {
|
|
||||||
public:
|
|
||||||
static void SendWebhookMessage(const std::string& message, const std::string& webhook_url);
|
|
||||||
static std::string FormatDiscordMessage(uint16 category_id, const std::string& message);
|
|
||||||
static void SendPlayerEventMessage(const PlayerEvent::PlayerEventContainer& e, const std::string &webhook_url);
|
|
||||||
static bool ValidateWebhookUrl(const std::string &webhook_url);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#endif //EQEMU_DISCORD_H
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
#include "discord_manager.h"
|
|
||||||
#include "../../common/discord/discord.h"
|
|
||||||
#include "../events/player_event_logs.h"
|
|
||||||
|
|
||||||
void DiscordManager::QueueWebhookMessage(uint32 webhook_id, const std::string &message)
|
|
||||||
{
|
|
||||||
webhook_queue_lock.lock();
|
|
||||||
webhook_message_queue[webhook_id].emplace_back(message);
|
|
||||||
webhook_queue_lock.unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr int MAX_MESSAGE_LENGTH = 1900;
|
|
||||||
|
|
||||||
void DiscordManager::ProcessMessageQueue()
|
|
||||||
{
|
|
||||||
if (webhook_message_queue.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
webhook_queue_lock.lock();
|
|
||||||
for (auto &q: webhook_message_queue) {
|
|
||||||
LogDiscord("Processing [{}] messages in queue for webhook ID [{}]...", q.second.size(), q.first);
|
|
||||||
|
|
||||||
if (q.first >= MAX_DISCORD_WEBHOOK_ID) {
|
|
||||||
LogDiscord("Out of bounds webhook ID [{}] max [{}]", q.first, MAX_DISCORD_WEBHOOK_ID);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto webhook = LogSys.GetDiscordWebhooks()[q.first];
|
|
||||||
std::string message;
|
|
||||||
|
|
||||||
for (auto &m: q.second) {
|
|
||||||
// next message would become too large
|
|
||||||
bool next_message_too_large = ((int) m.length() + (int) message.length()) > MAX_MESSAGE_LENGTH;
|
|
||||||
if (next_message_too_large) {
|
|
||||||
Discord::SendWebhookMessage(
|
|
||||||
message,
|
|
||||||
webhook.webhook_url
|
|
||||||
);
|
|
||||||
message.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
message += m;
|
|
||||||
|
|
||||||
// one single message was too large
|
|
||||||
// this should rarely happen but the message will need to be split
|
|
||||||
if ((int) message.length() > MAX_MESSAGE_LENGTH) {
|
|
||||||
for (unsigned mi = 0; mi < message.length(); mi += MAX_MESSAGE_LENGTH) {
|
|
||||||
Discord::SendWebhookMessage(
|
|
||||||
message.substr(mi, MAX_MESSAGE_LENGTH),
|
|
||||||
webhook.webhook_url
|
|
||||||
);
|
|
||||||
}
|
|
||||||
message.clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// final flush
|
|
||||||
if (!message.empty()) {
|
|
||||||
Discord::SendWebhookMessage(
|
|
||||||
message,
|
|
||||||
webhook.webhook_url
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
webhook_message_queue.clear();
|
|
||||||
webhook_queue_lock.unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DiscordManager::QueuePlayerEventMessage(const PlayerEvent::PlayerEventContainer& e)
|
|
||||||
{
|
|
||||||
auto w = player_event_logs.GetDiscordWebhookUrlFromEventType(e.player_event_log.event_type_id);
|
|
||||||
if (!w.empty()) {
|
|
||||||
Discord::SendPlayerEventMessage(e, w);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
#ifndef EQEMU_DISCORD_MANAGER_H
|
|
||||||
#define EQEMU_DISCORD_MANAGER_H
|
|
||||||
|
|
||||||
#include <mutex>
|
|
||||||
#include <map>
|
|
||||||
#include <vector>
|
|
||||||
#include "../../common/types.h"
|
|
||||||
#include "../repositories/player_event_logs_repository.h"
|
|
||||||
#include "../events/player_events.h"
|
|
||||||
|
|
||||||
class DiscordManager {
|
|
||||||
public:
|
|
||||||
void QueueWebhookMessage(uint32 webhook_id, const std::string& message);
|
|
||||||
void ProcessMessageQueue();
|
|
||||||
void QueuePlayerEventMessage(const PlayerEvent::PlayerEventContainer& e);
|
|
||||||
private:
|
|
||||||
std::mutex webhook_queue_lock{};
|
|
||||||
std::map<uint32, std::vector<std::string>> webhook_message_queue{};
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#endif
|
|
||||||
+48
-267
@@ -5,7 +5,6 @@
|
|||||||
#include "repositories/instance_list_player_repository.h"
|
#include "repositories/instance_list_player_repository.h"
|
||||||
#include "rulesys.h"
|
#include "rulesys.h"
|
||||||
#include "servertalk.h"
|
#include "servertalk.h"
|
||||||
#include "util/uuid.h"
|
|
||||||
|
|
||||||
DynamicZoneBase::DynamicZoneBase(DynamicZonesRepository::DynamicZoneInstance&& entry)
|
DynamicZoneBase::DynamicZoneBase(DynamicZonesRepository::DynamicZoneInstance&& entry)
|
||||||
{
|
{
|
||||||
@@ -14,12 +13,16 @@ DynamicZoneBase::DynamicZoneBase(DynamicZonesRepository::DynamicZoneInstance&& e
|
|||||||
|
|
||||||
uint32_t DynamicZoneBase::Create()
|
uint32_t DynamicZoneBase::Create()
|
||||||
{
|
{
|
||||||
|
if (m_id != 0)
|
||||||
|
{
|
||||||
|
return m_id;
|
||||||
|
}
|
||||||
|
|
||||||
if (GetInstanceID() == 0)
|
if (GetInstanceID() == 0)
|
||||||
{
|
{
|
||||||
CreateInstance();
|
CreateInstance();
|
||||||
}
|
}
|
||||||
|
|
||||||
m_uuid = EQ::Util::UUID::Generate().ToString();
|
|
||||||
m_id = SaveToDatabase();
|
m_id = SaveToDatabase();
|
||||||
|
|
||||||
return m_id;
|
return m_id;
|
||||||
@@ -72,14 +75,8 @@ uint32_t DynamicZoneBase::CreateInstance()
|
|||||||
void DynamicZoneBase::LoadRepositoryResult(DynamicZonesRepository::DynamicZoneInstance&& dz_entry)
|
void DynamicZoneBase::LoadRepositoryResult(DynamicZonesRepository::DynamicZoneInstance&& dz_entry)
|
||||||
{
|
{
|
||||||
m_id = dz_entry.id;
|
m_id = dz_entry.id;
|
||||||
m_uuid = std::move(dz_entry.uuid);
|
|
||||||
m_name = std::move(dz_entry.name);
|
|
||||||
m_leader.id = dz_entry.leader_id;
|
|
||||||
m_min_players = dz_entry.min_players;
|
|
||||||
m_max_players = dz_entry.max_players;
|
|
||||||
m_instance_id = dz_entry.instance_id;
|
m_instance_id = dz_entry.instance_id;
|
||||||
m_type = static_cast<DynamicZoneType>(dz_entry.type);
|
m_type = static_cast<DynamicZoneType>(dz_entry.type);
|
||||||
m_dz_switch_id = dz_entry.dz_switch_id;
|
|
||||||
m_compass.zone_id = dz_entry.compass_zone_id;
|
m_compass.zone_id = dz_entry.compass_zone_id;
|
||||||
m_compass.x = dz_entry.compass_x;
|
m_compass.x = dz_entry.compass_x;
|
||||||
m_compass.y = dz_entry.compass_y;
|
m_compass.y = dz_entry.compass_y;
|
||||||
@@ -107,12 +104,6 @@ void DynamicZoneBase::AddMemberFromRepositoryResult(
|
|||||||
DynamicZoneMembersRepository::MemberWithName&& entry)
|
DynamicZoneMembersRepository::MemberWithName&& entry)
|
||||||
{
|
{
|
||||||
auto status = DynamicZoneMemberStatus::Unknown;
|
auto status = DynamicZoneMemberStatus::Unknown;
|
||||||
|
|
||||||
if (m_leader.id == entry.character_id)
|
|
||||||
{
|
|
||||||
m_leader.name = entry.character_name;
|
|
||||||
}
|
|
||||||
|
|
||||||
AddInternalMember({ entry.character_id, std::move(entry.character_name), status });
|
AddInternalMember({ entry.character_id, std::move(entry.character_name), status });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -123,14 +114,8 @@ uint32_t DynamicZoneBase::SaveToDatabase()
|
|||||||
if (m_instance_id != 0)
|
if (m_instance_id != 0)
|
||||||
{
|
{
|
||||||
auto insert_dz = DynamicZonesRepository::NewEntity();
|
auto insert_dz = DynamicZonesRepository::NewEntity();
|
||||||
insert_dz.uuid = m_uuid;
|
|
||||||
insert_dz.name = m_name;
|
|
||||||
insert_dz.leader_id = m_leader.id;
|
|
||||||
insert_dz.min_players = m_min_players;
|
|
||||||
insert_dz.max_players = m_max_players;
|
|
||||||
insert_dz.instance_id = m_instance_id,
|
insert_dz.instance_id = m_instance_id,
|
||||||
insert_dz.type = static_cast<int>(m_type);
|
insert_dz.type = static_cast<int>(m_type);
|
||||||
insert_dz.dz_switch_id = m_dz_switch_id;
|
|
||||||
insert_dz.compass_zone_id = m_compass.zone_id;
|
insert_dz.compass_zone_id = m_compass.zone_id;
|
||||||
insert_dz.compass_x = m_compass.x;
|
insert_dz.compass_x = m_compass.x;
|
||||||
insert_dz.compass_y = m_compass.y;
|
insert_dz.compass_y = m_compass.y;
|
||||||
@@ -152,80 +137,34 @@ uint32_t DynamicZoneBase::SaveToDatabase()
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DynamicZoneBase::AddMember(const DynamicZoneMember& add_member)
|
void DynamicZoneBase::AddCharacter(uint32_t character_id)
|
||||||
{
|
{
|
||||||
if (HasMember(add_member.id))
|
DynamicZoneMembersRepository::AddMember(GetDatabase(), m_id, character_id);
|
||||||
|
GetDatabase().AddClientToInstance(m_instance_id, character_id);
|
||||||
|
SendInstanceAddRemoveCharacter(character_id, false); // stops client kick timer
|
||||||
|
}
|
||||||
|
|
||||||
|
void DynamicZoneBase::RemoveCharacter(uint32_t character_id)
|
||||||
|
{
|
||||||
|
DynamicZoneMembersRepository::RemoveMember(GetDatabase(), m_id, character_id);
|
||||||
|
GetDatabase().RemoveClientFromInstance(m_instance_id, character_id);
|
||||||
|
SendInstanceAddRemoveCharacter(character_id, true); // start client kick timer
|
||||||
|
}
|
||||||
|
|
||||||
|
void DynamicZoneBase::RemoveAllCharacters(bool enable_removal_timers)
|
||||||
|
{
|
||||||
|
if (GetInstanceID() == 0)
|
||||||
{
|
{
|
||||||
return false;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
DynamicZoneMembersRepository::AddMember(GetDatabase(), m_id, add_member.id);
|
if (enable_removal_timers)
|
||||||
GetDatabase().AddClientToInstance(m_instance_id, add_member.id);
|
|
||||||
|
|
||||||
ProcessMemberAddRemove(add_member, false);
|
|
||||||
SendServerPacket(CreateServerMemberAddRemovePacket(add_member, false).get());
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DynamicZoneBase::RemoveMember(uint32_t character_id)
|
|
||||||
{
|
|
||||||
auto remove_member = GetMemberData(character_id);
|
|
||||||
return RemoveMember(remove_member);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DynamicZoneBase::RemoveMember(const std::string& character_name)
|
|
||||||
{
|
|
||||||
auto remove_member = GetMemberData(character_name);
|
|
||||||
return RemoveMember(remove_member);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DynamicZoneBase::RemoveMember(const DynamicZoneMember& remove_member)
|
|
||||||
{
|
|
||||||
if (remove_member.id == 0)
|
|
||||||
{
|
{
|
||||||
return false;
|
SendInstanceRemoveAllCharacters();
|
||||||
}
|
}
|
||||||
|
|
||||||
DynamicZoneMembersRepository::RemoveMember(GetDatabase(), m_id, remove_member.id);
|
|
||||||
GetDatabase().RemoveClientFromInstance(m_instance_id, remove_member.id);
|
|
||||||
|
|
||||||
ProcessMemberAddRemove(remove_member, true);
|
|
||||||
SendServerPacket(CreateServerMemberAddRemovePacket(remove_member, true).get());
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DynamicZoneBase::SwapMember(
|
|
||||||
const DynamicZoneMember& add_member, const std::string& remove_char_name)
|
|
||||||
{
|
|
||||||
auto remove_member = GetMemberData(remove_char_name);
|
|
||||||
if (!add_member.IsValid() || !remove_member.IsValid())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// make remove and add atomic to avoid racing with separate world messages
|
|
||||||
DynamicZoneMembersRepository::RemoveMember(GetDatabase(), m_id, remove_member.id);
|
|
||||||
GetDatabase().RemoveClientFromInstance(m_instance_id, remove_member.id);
|
|
||||||
|
|
||||||
DynamicZoneMembersRepository::AddMember(GetDatabase(), m_id, add_member.id);
|
|
||||||
GetDatabase().AddClientToInstance(m_instance_id, add_member.id);
|
|
||||||
|
|
||||||
ProcessMemberAddRemove(remove_member, true);
|
|
||||||
ProcessMemberAddRemove(add_member, false);
|
|
||||||
SendServerPacket(CreateServerMemberSwapPacket(remove_member, add_member).get());
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DynamicZoneBase::RemoveAllMembers()
|
|
||||||
{
|
|
||||||
DynamicZoneMembersRepository::RemoveAllMembers(GetDatabase(), m_id);
|
DynamicZoneMembersRepository::RemoveAllMembers(GetDatabase(), m_id);
|
||||||
GetDatabase().RemoveClientsFromInstance(GetInstanceID());
|
GetDatabase().RemoveClientsFromInstance(GetInstanceID());
|
||||||
|
|
||||||
ProcessRemoveAllMembers();
|
|
||||||
SendServerPacket(CreateServerRemoveAllMembersPacket().get());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DynamicZoneBase::SaveMembers(const std::vector<DynamicZoneMember>& members)
|
void DynamicZoneBase::SaveMembers(const std::vector<DynamicZoneMember>& members)
|
||||||
@@ -242,6 +181,7 @@ void DynamicZoneBase::SaveMembers(const std::vector<DynamicZoneMember>& members)
|
|||||||
DynamicZoneMembersRepository::DynamicZoneMembers member_entry{};
|
DynamicZoneMembersRepository::DynamicZoneMembers member_entry{};
|
||||||
member_entry.dynamic_zone_id = m_id;
|
member_entry.dynamic_zone_id = m_id;
|
||||||
member_entry.character_id = member.id;
|
member_entry.character_id = member.id;
|
||||||
|
member_entry.is_current_member = true;
|
||||||
insert_members.emplace_back(member_entry);
|
insert_members.emplace_back(member_entry);
|
||||||
|
|
||||||
InstanceListPlayerRepository::InstanceListPlayer player_entry;
|
InstanceListPlayerRepository::InstanceListPlayer player_entry;
|
||||||
@@ -266,7 +206,7 @@ void DynamicZoneBase::SetCompass(const DynamicZoneLocation& location, bool updat
|
|||||||
DynamicZonesRepository::UpdateCompass(GetDatabase(),
|
DynamicZonesRepository::UpdateCompass(GetDatabase(),
|
||||||
m_id, m_compass.zone_id, m_compass.x, m_compass.y, m_compass.z);
|
m_id, m_compass.zone_id, m_compass.x, m_compass.y, m_compass.z);
|
||||||
|
|
||||||
SendServerPacket(CreateServerDzLocationPacket(ServerOP_DzSetCompass, location).get());
|
SendGlobalLocationChange(ServerOP_DzSetCompass, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -287,7 +227,7 @@ void DynamicZoneBase::SetSafeReturn(const DynamicZoneLocation& location, bool up
|
|||||||
DynamicZonesRepository::UpdateSafeReturn(GetDatabase(), m_id, m_safereturn.zone_id,
|
DynamicZonesRepository::UpdateSafeReturn(GetDatabase(), m_id, m_safereturn.zone_id,
|
||||||
m_safereturn.x, m_safereturn.y, m_safereturn.z, m_safereturn.heading);
|
m_safereturn.x, m_safereturn.y, m_safereturn.z, m_safereturn.heading);
|
||||||
|
|
||||||
SendServerPacket(CreateServerDzLocationPacket(ServerOP_DzSetSafeReturn, location).get());
|
SendGlobalLocationChange(ServerOP_DzSetSafeReturn, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -309,7 +249,7 @@ void DynamicZoneBase::SetZoneInLocation(const DynamicZoneLocation& location, boo
|
|||||||
DynamicZonesRepository::UpdateZoneIn(GetDatabase(), m_id, m_zone_id,
|
DynamicZonesRepository::UpdateZoneIn(GetDatabase(), m_id, m_zone_id,
|
||||||
m_zonein.x, m_zonein.y, m_zonein.z, m_zonein.heading, m_has_zonein);
|
m_zonein.x, m_zonein.y, m_zonein.z, m_zonein.heading, m_has_zonein);
|
||||||
|
|
||||||
SendServerPacket(CreateServerDzLocationPacket(ServerOP_DzSetZoneIn, location).get());
|
SendGlobalLocationChange(ServerOP_DzSetZoneIn, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -318,82 +258,35 @@ void DynamicZoneBase::SetZoneInLocation(float x, float y, float z, float heading
|
|||||||
SetZoneInLocation({ 0, x, y, z, heading }, update_db);
|
SetZoneInLocation({ 0, x, y, z, heading }, update_db);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DynamicZoneBase::SetSwitchID(int dz_switch_id, bool update_db)
|
|
||||||
{
|
|
||||||
m_dz_switch_id = dz_switch_id;
|
|
||||||
|
|
||||||
if (update_db)
|
|
||||||
{
|
|
||||||
DynamicZonesRepository::UpdateSwitchID(GetDatabase(), m_id, dz_switch_id);
|
|
||||||
SendServerPacket(CreateServerDzSwitchIDPacket().get());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void DynamicZoneBase::SetLeader(const DynamicZoneMember& new_leader, bool update_db)
|
|
||||||
{
|
|
||||||
m_leader = new_leader;
|
|
||||||
|
|
||||||
if (update_db)
|
|
||||||
{
|
|
||||||
DynamicZonesRepository::UpdateLeaderID(GetDatabase(), m_id, new_leader.id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t DynamicZoneBase::GetSecondsRemaining() const
|
uint32_t DynamicZoneBase::GetSecondsRemaining() const
|
||||||
{
|
{
|
||||||
auto remaining = std::chrono::duration_cast<std::chrono::seconds>(GetDurationRemaining()).count();
|
auto remaining = std::chrono::duration_cast<std::chrono::seconds>(GetDurationRemaining()).count();
|
||||||
return std::max(0, static_cast<int>(remaining));
|
return std::max(0, static_cast<int>(remaining));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerMemberAddRemovePacket(
|
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerAddRemoveCharacterPacket(
|
||||||
const DynamicZoneMember& member, bool removed)
|
uint32_t character_id, bool removed)
|
||||||
{
|
{
|
||||||
constexpr uint32_t pack_size = sizeof(ServerDzMember_Struct);
|
constexpr uint32_t pack_size = sizeof(ServerDzCharacter_Struct);
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzAddRemoveMember, pack_size);
|
auto pack = std::make_unique<ServerPacket>(ServerOP_DzAddRemoveCharacter, pack_size);
|
||||||
auto buf = reinterpret_cast<ServerDzMember_Struct*>(pack->pBuffer);
|
auto buf = reinterpret_cast<ServerDzCharacter_Struct*>(pack->pBuffer);
|
||||||
buf->dz_id = GetID();
|
buf->zone_id = GetZoneID();
|
||||||
buf->dz_zone_id = GetZoneID();
|
buf->instance_id = GetInstanceID();
|
||||||
buf->dz_instance_id = GetInstanceID();
|
buf->remove = removed;
|
||||||
buf->sender_zone_id = GetCurrentZoneID();
|
buf->character_id = character_id;
|
||||||
buf->sender_instance_id = GetCurrentInstanceID();
|
|
||||||
buf->removed = removed;
|
|
||||||
buf->character_id = member.id;
|
|
||||||
buf->character_status = static_cast<uint8_t>(member.status);
|
|
||||||
strn0cpy(buf->character_name, member.name.c_str(), sizeof(buf->character_name));
|
|
||||||
|
|
||||||
return pack;
|
return pack;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerMemberSwapPacket(
|
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerRemoveAllCharactersPacket()
|
||||||
const DynamicZoneMember& remove_member, const DynamicZoneMember& add_member)
|
|
||||||
{
|
{
|
||||||
constexpr uint32_t pack_size = sizeof(ServerDzMemberSwap_Struct);
|
constexpr uint32_t pack_size = sizeof(ServerDzCharacter_Struct);
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzSwapMembers, pack_size);
|
auto pack = std::make_unique<ServerPacket>(ServerOP_DzRemoveAllCharacters, pack_size);
|
||||||
auto buf = reinterpret_cast<ServerDzMemberSwap_Struct*>(pack->pBuffer);
|
auto buf = reinterpret_cast<ServerDzCharacter_Struct*>(pack->pBuffer);
|
||||||
buf->dz_id = GetID();
|
buf->zone_id = GetZoneID();
|
||||||
buf->dz_zone_id = GetZoneID();
|
buf->instance_id = GetInstanceID();
|
||||||
buf->dz_instance_id = GetInstanceID();
|
buf->remove = true;
|
||||||
buf->sender_zone_id = GetCurrentZoneID();
|
buf->character_id = 0;
|
||||||
buf->sender_instance_id = GetCurrentInstanceID();
|
|
||||||
buf->add_character_status = static_cast<uint8_t>(add_member.status);
|
|
||||||
buf->add_character_id = add_member.id;
|
|
||||||
buf->remove_character_id = remove_member.id;
|
|
||||||
strn0cpy(buf->add_character_name, add_member.name.c_str(), sizeof(buf->add_character_name));
|
|
||||||
strn0cpy(buf->remove_character_name, remove_member.name.c_str(), sizeof(buf->remove_character_name));
|
|
||||||
|
|
||||||
return pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerRemoveAllMembersPacket()
|
|
||||||
{
|
|
||||||
constexpr uint32_t pack_size = sizeof(ServerDzID_Struct);
|
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzRemoveAllMembers, pack_size);
|
|
||||||
auto buf = reinterpret_cast<ServerDzID_Struct*>(pack->pBuffer);
|
|
||||||
buf->dz_id = GetID();
|
|
||||||
buf->dz_zone_id = GetZoneID();
|
|
||||||
buf->dz_instance_id = GetInstanceID();
|
|
||||||
buf->sender_zone_id = GetCurrentZoneID();
|
|
||||||
buf->sender_instance_id = GetCurrentInstanceID();
|
|
||||||
|
|
||||||
return pack;
|
return pack;
|
||||||
}
|
}
|
||||||
@@ -416,36 +309,10 @@ std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerDzLocationPacket(
|
|||||||
return pack;
|
return pack;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerDzSwitchIDPacket()
|
|
||||||
{
|
|
||||||
constexpr uint32_t pack_size = sizeof(ServerDzSwitchID_Struct);
|
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzSetSwitchID, pack_size);
|
|
||||||
auto buf = reinterpret_cast<ServerDzSwitchID_Struct*>(pack->pBuffer);
|
|
||||||
buf->dz_id = GetID();
|
|
||||||
buf->dz_switch_id = GetSwitchID();
|
|
||||||
|
|
||||||
return pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerMemberStatusPacket(
|
|
||||||
uint32_t character_id, DynamicZoneMemberStatus status)
|
|
||||||
{
|
|
||||||
constexpr uint32_t pack_size = sizeof(ServerDzMemberStatus_Struct);
|
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzUpdateMemberStatus, pack_size);
|
|
||||||
auto buf = reinterpret_cast<ServerDzMemberStatus_Struct*>(pack->pBuffer);
|
|
||||||
buf->dz_id = GetID();
|
|
||||||
buf->sender_zone_id = GetCurrentZoneID();
|
|
||||||
buf->sender_instance_id = GetCurrentInstanceID();
|
|
||||||
buf->status = static_cast<uint8_t>(status);
|
|
||||||
buf->character_id = character_id;
|
|
||||||
|
|
||||||
return pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t DynamicZoneBase::GetDatabaseMemberCount()
|
uint32_t DynamicZoneBase::GetDatabaseMemberCount()
|
||||||
{
|
{
|
||||||
return DynamicZoneMembersRepository::GetCountWhere(GetDatabase(),
|
return DynamicZoneMembersRepository::GetCountWhere(GetDatabase(),
|
||||||
fmt::format("dynamic_zone_id = {}", m_id));
|
fmt::format("dynamic_zone_id = {} AND is_current_member = TRUE", m_id));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DynamicZoneBase::HasDatabaseMember(uint32_t character_id)
|
bool DynamicZoneBase::HasDatabaseMember(uint32_t character_id)
|
||||||
@@ -456,7 +323,7 @@ bool DynamicZoneBase::HasDatabaseMember(uint32_t character_id)
|
|||||||
}
|
}
|
||||||
|
|
||||||
auto entries = DynamicZoneMembersRepository::GetWhere(GetDatabase(), fmt::format(
|
auto entries = DynamicZoneMembersRepository::GetWhere(GetDatabase(), fmt::format(
|
||||||
"dynamic_zone_id = {} AND character_id = {}",
|
"dynamic_zone_id = {} AND character_id = {} AND is_current_member = TRUE",
|
||||||
m_id, character_id
|
m_id, character_id
|
||||||
));
|
));
|
||||||
|
|
||||||
@@ -544,33 +411,6 @@ bool DynamicZoneBase::SetInternalMemberStatus(uint32_t character_id, DynamicZone
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DynamicZoneBase::SetMemberStatus(uint32_t character_id, DynamicZoneMemberStatus status)
|
|
||||||
{
|
|
||||||
auto update_member = GetMemberData(character_id);
|
|
||||||
if (update_member.IsValid())
|
|
||||||
{
|
|
||||||
ProcessMemberStatusChange(character_id, status);
|
|
||||||
SendServerPacket(CreateServerMemberStatusPacket(character_id, status).get());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void DynamicZoneBase::ProcessMemberAddRemove(const DynamicZoneMember& member, bool removed)
|
|
||||||
{
|
|
||||||
if (!removed)
|
|
||||||
{
|
|
||||||
AddInternalMember(member);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
RemoveInternalMember(member.id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DynamicZoneBase::ProcessMemberStatusChange(uint32_t character_id, DynamicZoneMemberStatus status)
|
|
||||||
{
|
|
||||||
return SetInternalMemberStatus(character_id, status);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string DynamicZoneBase::GetDynamicZoneTypeName(DynamicZoneType dz_type)
|
std::string DynamicZoneBase::GetDynamicZoneTypeName(DynamicZoneType dz_type)
|
||||||
{
|
{
|
||||||
switch (dz_type)
|
switch (dz_type)
|
||||||
@@ -585,65 +425,6 @@ std::string DynamicZoneBase::GetDynamicZoneTypeName(DynamicZoneType dz_type)
|
|||||||
return "Mission";
|
return "Mission";
|
||||||
case DynamicZoneType::Quest:
|
case DynamicZoneType::Quest:
|
||||||
return "Quest";
|
return "Quest";
|
||||||
default:
|
|
||||||
return "Unknown";
|
|
||||||
}
|
}
|
||||||
}
|
return "Unknown";
|
||||||
|
|
||||||
EQ::Net::DynamicPacket DynamicZoneBase::GetSerializedDzPacket()
|
|
||||||
{
|
|
||||||
EQ::Net::DynamicPacket dyn_pack;
|
|
||||||
dyn_pack.PutSerialize(0, *this);
|
|
||||||
|
|
||||||
LogDynamicZonesDetail("Serialized server dz size [{}]", dyn_pack.Length());
|
|
||||||
return dyn_pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> DynamicZoneBase::CreateServerDzCreatePacket(
|
|
||||||
uint16_t origin_zone_id, uint16_t origin_instance_id)
|
|
||||||
{
|
|
||||||
EQ::Net::DynamicPacket dyn_pack = GetSerializedDzPacket();
|
|
||||||
|
|
||||||
auto pack_size = sizeof(ServerDzCreateSerialized_Struct) + dyn_pack.Length();
|
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_DzCreated, static_cast<uint32_t>(pack_size));
|
|
||||||
auto buf = reinterpret_cast<ServerDzCreateSerialized_Struct*>(pack->pBuffer);
|
|
||||||
buf->origin_zone_id = origin_zone_id;
|
|
||||||
buf->origin_instance_id = origin_instance_id;
|
|
||||||
buf->cereal_size = static_cast<uint32_t>(dyn_pack.Length());
|
|
||||||
memcpy(buf->cereal_data, dyn_pack.Data(), dyn_pack.Length());
|
|
||||||
|
|
||||||
return pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DynamicZoneBase::LoadSerializedDzPacket(char* cereal_data, uint32_t cereal_size)
|
|
||||||
{
|
|
||||||
LogDynamicZonesDetail("Deserializing server dz size [{}]", cereal_size);
|
|
||||||
EQ::Util::MemoryStreamReader ss(cereal_data, cereal_size);
|
|
||||||
cereal::BinaryInputArchive archive(ss);
|
|
||||||
archive(*this);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DynamicZoneBase::LoadTemplate(const DynamicZoneTemplatesRepository::DynamicZoneTemplates& dz_template)
|
|
||||||
{
|
|
||||||
m_zone_id = dz_template.zone_id;
|
|
||||||
m_zone_version = dz_template.zone_version;
|
|
||||||
m_name = dz_template.name;
|
|
||||||
m_min_players = dz_template.min_players;
|
|
||||||
m_max_players = dz_template.max_players;
|
|
||||||
m_duration = std::chrono::seconds(dz_template.duration_seconds);
|
|
||||||
m_dz_switch_id = dz_template.dz_switch_id;
|
|
||||||
m_compass.zone_id = dz_template.compass_zone_id;
|
|
||||||
m_compass.x = dz_template.compass_x;
|
|
||||||
m_compass.y = dz_template.compass_y;
|
|
||||||
m_compass.z = dz_template.compass_z;
|
|
||||||
m_safereturn.zone_id = dz_template.return_zone_id;
|
|
||||||
m_safereturn.x = dz_template.return_x;
|
|
||||||
m_safereturn.y = dz_template.return_y;
|
|
||||||
m_safereturn.z = dz_template.return_z;
|
|
||||||
m_safereturn.heading = dz_template.return_h;
|
|
||||||
m_has_zonein = dz_template.override_zone_in;
|
|
||||||
m_zonein.x = dz_template.zone_in_x;
|
|
||||||
m_zonein.y = dz_template.zone_in_y;
|
|
||||||
m_zonein.z = dz_template.zone_in_z;
|
|
||||||
m_zonein.heading = dz_template.zone_in_h;
|
|
||||||
}
|
}
|
||||||
|
|||||||
+15
-81
@@ -2,10 +2,8 @@
|
|||||||
#define COMMON_DYNAMIC_ZONE_BASE_H
|
#define COMMON_DYNAMIC_ZONE_BASE_H
|
||||||
|
|
||||||
#include "eq_constants.h"
|
#include "eq_constants.h"
|
||||||
#include "net/packet.h"
|
|
||||||
#include "repositories/dynamic_zones_repository.h"
|
#include "repositories/dynamic_zones_repository.h"
|
||||||
#include "repositories/dynamic_zone_members_repository.h"
|
#include "repositories/dynamic_zone_members_repository.h"
|
||||||
#include "repositories/dynamic_zone_templates_repository.h"
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
@@ -20,7 +18,7 @@ struct DynamicZoneMember
|
|||||||
{
|
{
|
||||||
uint32_t id = 0;
|
uint32_t id = 0;
|
||||||
std::string name;
|
std::string name;
|
||||||
DynamicZoneMemberStatus status = DynamicZoneMemberStatus::Unknown;
|
DynamicZoneMemberStatus status = DynamicZoneMemberStatus::Online;
|
||||||
|
|
||||||
DynamicZoneMember() = default;
|
DynamicZoneMember() = default;
|
||||||
DynamicZoneMember(uint32_t id, std::string name_)
|
DynamicZoneMember(uint32_t id, std::string name_)
|
||||||
@@ -31,12 +29,6 @@ struct DynamicZoneMember
|
|||||||
bool IsOnline() const { return status == DynamicZoneMemberStatus::Online ||
|
bool IsOnline() const { return status == DynamicZoneMemberStatus::Online ||
|
||||||
status == DynamicZoneMemberStatus::InDynamicZone; }
|
status == DynamicZoneMemberStatus::InDynamicZone; }
|
||||||
bool IsValid() const { return id != 0 && !name.empty(); }
|
bool IsValid() const { return id != 0 && !name.empty(); }
|
||||||
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive& archive)
|
|
||||||
{
|
|
||||||
archive(id, name, status);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct DynamicZoneLocation
|
struct DynamicZoneLocation
|
||||||
@@ -50,12 +42,6 @@ struct DynamicZoneLocation
|
|||||||
DynamicZoneLocation() = default;
|
DynamicZoneLocation() = default;
|
||||||
DynamicZoneLocation(uint32_t zone_id_, float x_, float y_, float z_, float heading_)
|
DynamicZoneLocation(uint32_t zone_id_, float x_, float y_, float z_, float heading_)
|
||||||
: zone_id(zone_id_), x(x_), y(y_), z(z_), heading(heading_) {}
|
: zone_id(zone_id_), x(x_), y(y_), z(z_), heading(heading_) {}
|
||||||
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive& archive)
|
|
||||||
{
|
|
||||||
archive(zone_id, x, y, z, heading);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class DynamicZoneBase
|
class DynamicZoneBase
|
||||||
@@ -75,11 +61,9 @@ public:
|
|||||||
|
|
||||||
virtual void SetSecondsRemaining(uint32_t seconds_remaining) = 0;
|
virtual void SetSecondsRemaining(uint32_t seconds_remaining) = 0;
|
||||||
|
|
||||||
int GetDuration() const { return static_cast<int>(m_duration.count()); }
|
|
||||||
uint64_t GetExpireTime() const { return std::chrono::system_clock::to_time_t(m_expire_time); }
|
uint64_t GetExpireTime() const { return std::chrono::system_clock::to_time_t(m_expire_time); }
|
||||||
uint32_t GetID() const { return m_id; }
|
uint32_t GetID() const { return m_id; }
|
||||||
uint16_t GetInstanceID() const { return static_cast<uint16_t>(m_instance_id); }
|
uint16_t GetInstanceID() const { return static_cast<uint16_t>(m_instance_id); }
|
||||||
uint32_t GetLeaderID() const { return m_leader.id; }
|
|
||||||
uint32_t GetMaxPlayers() const { return m_max_players; }
|
uint32_t GetMaxPlayers() const { return m_max_players; }
|
||||||
uint32_t GetMemberCount() const { return static_cast<uint32_t>(m_members.size()); }
|
uint32_t GetMemberCount() const { return static_cast<uint32_t>(m_members.size()); }
|
||||||
uint32_t GetMinPlayers() const { return m_min_players; }
|
uint32_t GetMinPlayers() const { return m_min_players; }
|
||||||
@@ -87,11 +71,9 @@ public:
|
|||||||
uint16_t GetZoneID() const { return static_cast<uint16_t>(m_zone_id); }
|
uint16_t GetZoneID() const { return static_cast<uint16_t>(m_zone_id); }
|
||||||
uint32_t GetZoneIndex() const { return (m_instance_id << 16) | (m_zone_id & 0xffff); }
|
uint32_t GetZoneIndex() const { return (m_instance_id << 16) | (m_zone_id & 0xffff); }
|
||||||
uint32_t GetZoneVersion() const { return m_zone_version; }
|
uint32_t GetZoneVersion() const { return m_zone_version; }
|
||||||
int GetSwitchID() const { return m_dz_switch_id; }
|
|
||||||
DynamicZoneType GetType() const { return m_type; }
|
DynamicZoneType GetType() const { return m_type; }
|
||||||
const std::string& GetLeaderName() const { return m_leader.name; }
|
const std::string& GetLeaderName() const { return m_leader.name; }
|
||||||
const std::string& GetName() const { return m_name; }
|
const std::string& GetName() const { return m_name; }
|
||||||
const std::string& GetUUID() const { return m_uuid; }
|
|
||||||
const DynamicZoneMember& GetLeader() const { return m_leader; }
|
const DynamicZoneMember& GetLeader() const { return m_leader; }
|
||||||
const std::vector<DynamicZoneMember>& GetMembers() const { return m_members; }
|
const std::vector<DynamicZoneMember>& GetMembers() const { return m_members; }
|
||||||
const DynamicZoneLocation& GetCompassLocation() const { return m_compass; }
|
const DynamicZoneLocation& GetCompassLocation() const { return m_compass; }
|
||||||
@@ -99,12 +81,14 @@ public:
|
|||||||
const DynamicZoneLocation& GetZoneInLocation() const { return m_zonein; }
|
const DynamicZoneLocation& GetZoneInLocation() const { return m_zonein; }
|
||||||
std::chrono::system_clock::duration GetDurationRemaining() const { return m_expire_time - std::chrono::system_clock::now(); }
|
std::chrono::system_clock::duration GetDurationRemaining() const { return m_expire_time - std::chrono::system_clock::now(); }
|
||||||
|
|
||||||
bool AddMember(const DynamicZoneMember& add_member);
|
void AddCharacter(uint32_t character_id);
|
||||||
|
void AddInternalMember(const DynamicZoneMember& member);
|
||||||
void AddMemberFromRepositoryResult(DynamicZoneMembersRepository::MemberWithName&& entry);
|
void AddMemberFromRepositoryResult(DynamicZoneMembersRepository::MemberWithName&& entry);
|
||||||
|
void ClearInternalMembers() { m_members.clear(); }
|
||||||
|
uint32_t Create();
|
||||||
uint32_t GetDatabaseMemberCount();
|
uint32_t GetDatabaseMemberCount();
|
||||||
DynamicZoneMember GetMemberData(uint32_t character_id);
|
DynamicZoneMember GetMemberData(uint32_t character_id);
|
||||||
DynamicZoneMember GetMemberData(const std::string& character_name);
|
DynamicZoneMember GetMemberData(const std::string& character_name);
|
||||||
EQ::Net::DynamicPacket GetSerializedDzPacket();
|
|
||||||
bool HasDatabaseMember(uint32_t character_id);
|
bool HasDatabaseMember(uint32_t character_id);
|
||||||
bool HasMember(uint32_t character_id);
|
bool HasMember(uint32_t character_id);
|
||||||
bool HasMember(const std::string& character_name);
|
bool HasMember(const std::string& character_name);
|
||||||
@@ -114,56 +98,38 @@ public:
|
|||||||
bool IsInstanceID(uint32_t instance_id) const { return (m_instance_id != 0 && m_instance_id == instance_id); }
|
bool IsInstanceID(uint32_t instance_id) const { return (m_instance_id != 0 && m_instance_id == instance_id); }
|
||||||
bool IsValid() const { return m_instance_id != 0; }
|
bool IsValid() const { return m_instance_id != 0; }
|
||||||
bool IsSameDz(uint32_t zone_id, uint32_t instance_id) const { return zone_id == m_zone_id && instance_id == m_instance_id; }
|
bool IsSameDz(uint32_t zone_id, uint32_t instance_id) const { return zone_id == m_zone_id && instance_id == m_instance_id; }
|
||||||
void LoadSerializedDzPacket(char* cereal_data, uint32_t cereal_size);
|
void RemoveAllCharacters(bool enable_removal_timers = true);
|
||||||
void LoadTemplate(const DynamicZoneTemplatesRepository::DynamicZoneTemplates& dz_template);
|
void RemoveCharacter(uint32_t character_id);
|
||||||
void RemoveAllMembers();
|
void RemoveInternalMember(uint32_t character_id);
|
||||||
bool RemoveMember(uint32_t character_id);
|
|
||||||
bool RemoveMember(const std::string& character_name);
|
|
||||||
bool RemoveMember(const DynamicZoneMember& remove_member);
|
|
||||||
void SaveMembers(const std::vector<DynamicZoneMember>& members);
|
void SaveMembers(const std::vector<DynamicZoneMember>& members);
|
||||||
void SetCompass(const DynamicZoneLocation& location, bool update_db = false);
|
void SetCompass(const DynamicZoneLocation& location, bool update_db = false);
|
||||||
void SetCompass(uint32_t zone_id, float x, float y, float z, bool update_db = false);
|
void SetCompass(uint32_t zone_id, float x, float y, float z, bool update_db = false);
|
||||||
void SetDuration(uint32_t seconds) { m_duration = std::chrono::seconds(seconds); }
|
bool SetInternalMemberStatus(uint32_t character_id, DynamicZoneMemberStatus status);
|
||||||
void SetLeader(const DynamicZoneMember& leader, bool update_db = false);
|
void SetLeader(const DynamicZoneMember& leader) { m_leader = leader; }
|
||||||
void SetMaxPlayers(uint32_t max_players) { m_max_players = max_players; }
|
void SetMaxPlayers(uint32_t max_players) { m_max_players = max_players; }
|
||||||
void SetMemberStatus(uint32_t character_id, DynamicZoneMemberStatus status);
|
|
||||||
void SetMinPlayers(uint32_t min_players) { m_min_players = min_players; }
|
void SetMinPlayers(uint32_t min_players) { m_min_players = min_players; }
|
||||||
void SetName(const std::string& name) { m_name = name; }
|
void SetName(const std::string& name) { m_name = name; }
|
||||||
void SetSafeReturn(const DynamicZoneLocation& location, bool update_db = false);
|
void SetSafeReturn(const DynamicZoneLocation& location, bool update_db = false);
|
||||||
void SetSafeReturn(uint32_t zone_id, float x, float y, float z, float heading, bool update_db = false);
|
void SetSafeReturn(uint32_t zone_id, float x, float y, float z, float heading, bool update_db = false);
|
||||||
void SetSwitchID(int dz_switch_id, bool update_db = false);
|
|
||||||
void SetType(DynamicZoneType type) { m_type = type; }
|
|
||||||
void SetUUID(std::string uuid) { m_uuid = std::move(uuid); }
|
|
||||||
void SetZoneInLocation(const DynamicZoneLocation& location, bool update_db = false);
|
void SetZoneInLocation(const DynamicZoneLocation& location, bool update_db = false);
|
||||||
void SetZoneInLocation(float x, float y, float z, float heading, bool update_db = false);
|
void SetZoneInLocation(float x, float y, float z, float heading, bool update_db = false);
|
||||||
bool SwapMember(const DynamicZoneMember& add_member, const std::string& remove_char_name);
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual uint16_t GetCurrentInstanceID() { return 0; }
|
virtual uint16_t GetCurrentInstanceID() { return 0; }
|
||||||
virtual uint16_t GetCurrentZoneID() { return 0; }
|
virtual uint16_t GetCurrentZoneID() { return 0; }
|
||||||
virtual Database& GetDatabase() = 0;
|
virtual Database& GetDatabase() = 0;
|
||||||
virtual void ProcessCompassChange(const DynamicZoneLocation& location) { m_compass = location; }
|
virtual void ProcessCompassChange(const DynamicZoneLocation& location) { m_compass = location; }
|
||||||
virtual void ProcessMemberAddRemove(const DynamicZoneMember& member, bool removed);
|
virtual void SendInstanceAddRemoveCharacter(uint32_t character_id, bool remove) = 0;
|
||||||
virtual bool ProcessMemberStatusChange(uint32_t member_id, DynamicZoneMemberStatus status);
|
virtual void SendInstanceRemoveAllCharacters() = 0;
|
||||||
virtual void ProcessRemoveAllMembers(bool silent = false) { m_members.clear(); }
|
virtual void SendGlobalLocationChange(uint16_t server_opcode, const DynamicZoneLocation& location) = 0;
|
||||||
virtual void ProcessSetSwitchID(int dz_switch_id) { m_dz_switch_id = dz_switch_id; }
|
|
||||||
virtual bool SendServerPacket(ServerPacket* packet) = 0;
|
|
||||||
|
|
||||||
void AddInternalMember(const DynamicZoneMember& member);
|
|
||||||
uint32_t Create();
|
|
||||||
uint32_t CreateInstance();
|
uint32_t CreateInstance();
|
||||||
void LoadRepositoryResult(DynamicZonesRepository::DynamicZoneInstance&& dz_entry);
|
void LoadRepositoryResult(DynamicZonesRepository::DynamicZoneInstance&& dz_entry);
|
||||||
void RemoveInternalMember(uint32_t character_id);
|
|
||||||
uint32_t SaveToDatabase();
|
uint32_t SaveToDatabase();
|
||||||
bool SetInternalMemberStatus(uint32_t character_id, DynamicZoneMemberStatus status);
|
|
||||||
|
|
||||||
std::unique_ptr<ServerPacket> CreateServerDzCreatePacket(uint16_t origin_zone_id, uint16_t origin_instance_id);
|
std::unique_ptr<ServerPacket> CreateServerAddRemoveCharacterPacket(uint32_t character_id, bool removed);
|
||||||
|
std::unique_ptr<ServerPacket> CreateServerRemoveAllCharactersPacket();
|
||||||
std::unique_ptr<ServerPacket> CreateServerDzLocationPacket(uint16_t server_opcode, const DynamicZoneLocation& location);
|
std::unique_ptr<ServerPacket> CreateServerDzLocationPacket(uint16_t server_opcode, const DynamicZoneLocation& location);
|
||||||
std::unique_ptr<ServerPacket> CreateServerDzSwitchIDPacket();
|
|
||||||
std::unique_ptr<ServerPacket> CreateServerMemberAddRemovePacket(const DynamicZoneMember& member, bool removed);
|
|
||||||
std::unique_ptr<ServerPacket> CreateServerMemberStatusPacket(uint32_t character_id, DynamicZoneMemberStatus status);
|
|
||||||
std::unique_ptr<ServerPacket> CreateServerMemberSwapPacket(const DynamicZoneMember& remove_member, const DynamicZoneMember& add_member);
|
|
||||||
std::unique_ptr<ServerPacket> CreateServerRemoveAllMembersPacket();
|
|
||||||
|
|
||||||
uint32_t m_id = 0;
|
uint32_t m_id = 0;
|
||||||
uint32_t m_zone_id = 0;
|
uint32_t m_zone_id = 0;
|
||||||
@@ -171,12 +137,9 @@ protected:
|
|||||||
uint32_t m_zone_version = 0;
|
uint32_t m_zone_version = 0;
|
||||||
uint32_t m_min_players = 0;
|
uint32_t m_min_players = 0;
|
||||||
uint32_t m_max_players = 0;
|
uint32_t m_max_players = 0;
|
||||||
int m_dz_switch_id = 0;
|
|
||||||
bool m_never_expires = false;
|
bool m_never_expires = false;
|
||||||
bool m_has_zonein = false;
|
bool m_has_zonein = false;
|
||||||
bool m_has_member_statuses = false;
|
|
||||||
std::string m_name;
|
std::string m_name;
|
||||||
std::string m_uuid;
|
|
||||||
DynamicZoneMember m_leader;
|
DynamicZoneMember m_leader;
|
||||||
DynamicZoneType m_type{ DynamicZoneType::None };
|
DynamicZoneType m_type{ DynamicZoneType::None };
|
||||||
DynamicZoneLocation m_compass;
|
DynamicZoneLocation m_compass;
|
||||||
@@ -186,35 +149,6 @@ protected:
|
|||||||
std::chrono::time_point<std::chrono::system_clock> m_start_time;
|
std::chrono::time_point<std::chrono::system_clock> m_start_time;
|
||||||
std::chrono::time_point<std::chrono::system_clock> m_expire_time;
|
std::chrono::time_point<std::chrono::system_clock> m_expire_time;
|
||||||
std::vector<DynamicZoneMember> m_members;
|
std::vector<DynamicZoneMember> m_members;
|
||||||
|
|
||||||
public:
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive& archive)
|
|
||||||
{
|
|
||||||
archive(
|
|
||||||
m_id,
|
|
||||||
m_zone_id,
|
|
||||||
m_instance_id,
|
|
||||||
m_zone_version,
|
|
||||||
m_min_players,
|
|
||||||
m_max_players,
|
|
||||||
m_dz_switch_id,
|
|
||||||
m_never_expires,
|
|
||||||
m_has_zonein,
|
|
||||||
m_has_member_statuses,
|
|
||||||
m_name,
|
|
||||||
m_uuid,
|
|
||||||
m_leader,
|
|
||||||
m_type,
|
|
||||||
m_compass,
|
|
||||||
m_safereturn,
|
|
||||||
m_zonein,
|
|
||||||
m_duration,
|
|
||||||
m_start_time,
|
|
||||||
m_expire_time,
|
|
||||||
m_members
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
+89
-388
@@ -1,5 +1,5 @@
|
|||||||
/* EQEMu: Everquest Server Emulator
|
/* EQEMu: Everquest Server Emulator
|
||||||
|
|
||||||
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
This program is free software; you can redistribute it and/or modify
|
||||||
@@ -18,11 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "emu_constants.h"
|
#include "emu_constants.h"
|
||||||
#include "bodytypes.h"
|
|
||||||
#include "data_verification.h"
|
|
||||||
#include "eqemu_logsys.h"
|
|
||||||
#include "eqemu_logsys_log_aliases.h"
|
|
||||||
#include "rulesys.h"
|
|
||||||
|
|
||||||
int16 EQ::invtype::GetInvTypeSize(int16 inv_type) {
|
int16 EQ::invtype::GetInvTypeSize(int16 inv_type) {
|
||||||
static const int16 local_array[] = {
|
static const int16 local_array[] = {
|
||||||
@@ -59,395 +55,100 @@ int16 EQ::invtype::GetInvTypeSize(int16 inv_type) {
|
|||||||
return local_array[inv_type];
|
return local_array[inv_type];
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 Bug::GetID(const std::string& category_name)
|
const char* EQ::bug::CategoryIDToCategoryName(CategoryID category_id) {
|
||||||
{
|
switch (category_id) {
|
||||||
for (const auto& e : bug_category_names) {
|
case catVideo:
|
||||||
if (e.second == category_name) {
|
return "Video";
|
||||||
return e.first;
|
case catAudio:
|
||||||
}
|
return "Audio";
|
||||||
|
case catPathing:
|
||||||
|
return "Pathing";
|
||||||
|
case catQuest:
|
||||||
|
return "Quest";
|
||||||
|
case catTradeskills:
|
||||||
|
return "Tradeskills";
|
||||||
|
case catSpellStacking:
|
||||||
|
return "Spell stacking";
|
||||||
|
case catDoorsPortals:
|
||||||
|
return "Doors/Portals";
|
||||||
|
case catItems:
|
||||||
|
return "Items";
|
||||||
|
case catNPC:
|
||||||
|
return "NPC";
|
||||||
|
case catDialogs:
|
||||||
|
return "Dialogs";
|
||||||
|
case catLoNTCG:
|
||||||
|
return "LoN - TCG";
|
||||||
|
case catMercenaries:
|
||||||
|
return "Mercenaries";
|
||||||
|
case catOther:
|
||||||
|
default:
|
||||||
|
return "Other";
|
||||||
}
|
}
|
||||||
|
|
||||||
return Bug::Category::Other;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string Bug::GetName(uint32 category_id)
|
EQ::bug::CategoryID EQ::bug::CategoryNameToCategoryID(const char* category_name) {
|
||||||
{
|
if (!category_name)
|
||||||
return IsValid(category_id) ? bug_category_names[category_id] : "UNKNOWN BUG CATEGORY";
|
return catOther;
|
||||||
|
|
||||||
|
if (!strcmp(category_name, "Video"))
|
||||||
|
return catVideo;
|
||||||
|
if (!strcmp(category_name, "Audio"))
|
||||||
|
return catAudio;
|
||||||
|
if (!strcmp(category_name, "Pathing"))
|
||||||
|
return catPathing;
|
||||||
|
if (!strcmp(category_name, "Quest"))
|
||||||
|
return catQuest;
|
||||||
|
if (!strcmp(category_name, "Tradeskills"))
|
||||||
|
return catTradeskills;
|
||||||
|
if (!strcmp(category_name, "Spell stacking"))
|
||||||
|
return catSpellStacking;
|
||||||
|
if (!strcmp(category_name, "Doors/Portals"))
|
||||||
|
return catDoorsPortals;
|
||||||
|
if (!strcmp(category_name, "Items"))
|
||||||
|
return catItems;
|
||||||
|
if (!strcmp(category_name, "NPC"))
|
||||||
|
return catNPC;
|
||||||
|
if (!strcmp(category_name, "Dialogs"))
|
||||||
|
return catDialogs;
|
||||||
|
if (!strcmp(category_name, "LoN - TCG"))
|
||||||
|
return catLoNTCG;
|
||||||
|
if (!strcmp(category_name, "Mercenaries"))
|
||||||
|
return catMercenaries;
|
||||||
|
|
||||||
|
return catOther;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Bug::IsValid(uint32 category_id)
|
const char *EQ::constants::GetStanceName(StanceType stance_type) {
|
||||||
{
|
switch (stance_type) {
|
||||||
return bug_category_names.find(category_id) != bug_category_names.end();
|
case stanceUnknown:
|
||||||
}
|
return "Unknown";
|
||||||
|
case stancePassive:
|
||||||
std::string Stance::GetName(uint8 stance_id)
|
return "Passive";
|
||||||
{
|
case stanceBalanced:
|
||||||
return IsValid(stance_id) ? stance_names[stance_id] : "UNKNOWN STANCE";
|
return "Balanced";
|
||||||
}
|
case stanceEfficient:
|
||||||
|
return "Efficient";
|
||||||
bool Stance::IsValid(uint8 stance_id)
|
case stanceReactive:
|
||||||
{
|
return "Reactive";
|
||||||
return stance_names.find(stance_id) != stance_names.end();
|
case stanceAggressive:
|
||||||
}
|
return "Aggressive";
|
||||||
|
case stanceAssist:
|
||||||
uint8 Stance::GetIndex(uint8 stance_id)
|
return "Assist";
|
||||||
{
|
case stanceBurn:
|
||||||
return IsValid(stance_id) ? (stance_id - Stance::Passive) : 0;
|
return "Burn";
|
||||||
}
|
case stanceEfficient2:
|
||||||
|
return "Efficient2";
|
||||||
const std::map<uint8, std::string>& EQ::constants::GetLanguageMap()
|
case stanceBurnAE:
|
||||||
{
|
return "BurnAE";
|
||||||
static const std::map<uint8, std::string> language_map = {
|
default:
|
||||||
{ Language::CommonTongue, "Common Tongue" },
|
return "Invalid";
|
||||||
{ Language::Barbarian, "Barbarian" },
|
|
||||||
{ Language::Erudian, "Erudian" },
|
|
||||||
{ Language::Elvish, "Elvish" },
|
|
||||||
{ Language::DarkElvish, "Dark Elvish" },
|
|
||||||
{ Language::Dwarvish, "Dwarvish" },
|
|
||||||
{ Language::Troll, "Troll" },
|
|
||||||
{ Language::Ogre, "Ogre" },
|
|
||||||
{ Language::Gnomish, "Gnomish" },
|
|
||||||
{ Language::Halfling, "Halfling" },
|
|
||||||
{ Language::ThievesCant, "Thieves Cant" },
|
|
||||||
{ Language::OldErudian, "Old Erudian" },
|
|
||||||
{ Language::ElderElvish, "Elder Elvish" },
|
|
||||||
{ Language::Froglok, "Froglok" },
|
|
||||||
{ Language::Goblin, "Goblin" },
|
|
||||||
{ Language::Gnoll, "Gnoll" },
|
|
||||||
{ Language::CombineTongue, "Combine Tongue" },
|
|
||||||
{ Language::ElderTeirDal, "Elder Teir'Dal" },
|
|
||||||
{ Language::Lizardman, "Lizardman" },
|
|
||||||
{ Language::Orcish, "Orcish" },
|
|
||||||
{ Language::Faerie, "Faerie" },
|
|
||||||
{ Language::Dragon, "Dragon" },
|
|
||||||
{ Language::ElderDragon, "Elder Dragon" },
|
|
||||||
{ Language::DarkSpeech, "Dark Speech" },
|
|
||||||
{ Language::VahShir, "Vah Shir" },
|
|
||||||
{ Language::Alaran, "Alaran" },
|
|
||||||
{ Language::Hadal, "Hadal" },
|
|
||||||
{ Language::Unknown27, "Unknown" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return language_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetLanguageName(uint8 language_id)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(language_id, Language::CommonTongue, Language::Unknown27)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return EQ::constants::GetLanguageMap().find(language_id)->second;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::map<int8, std::string>& EQ::constants::GetFlyModeMap()
|
int EQ::constants::ConvertStanceTypeToIndex(StanceType stance_type) {
|
||||||
{
|
if (stance_type >= EQ::constants::stancePassive && stance_type <= EQ::constants::stanceBurnAE)
|
||||||
static const std::map<int8, std::string> flymode_map = {
|
return (stance_type - EQ::constants::stancePassive);
|
||||||
{ GravityBehavior::Ground, "Ground" },
|
|
||||||
{ GravityBehavior::Flying, "Flying" },
|
|
||||||
{ GravityBehavior::Levitating, "Levitating" },
|
|
||||||
{ GravityBehavior::Water, "Water" },
|
|
||||||
{ GravityBehavior::Floating, "Floating" },
|
|
||||||
{ GravityBehavior::LevitateWhileRunning, "Levitating While Running" },
|
|
||||||
};
|
|
||||||
|
|
||||||
return flymode_map;
|
return 0;
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetFlyModeName(int8 flymode_id)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(flymode_id, GravityBehavior::Ground, GravityBehavior::LevitateWhileRunning)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetFlyModeMap().find(flymode_id)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string>& EQ::constants::GetConsiderLevelMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> consider_level_map = {
|
|
||||||
{ ConsiderLevel::Ally, "Ally" },
|
|
||||||
{ ConsiderLevel::Warmly, "Warmly" },
|
|
||||||
{ ConsiderLevel::Kindly, "Kindly" },
|
|
||||||
{ ConsiderLevel::Amiably, "Amiably" },
|
|
||||||
{ ConsiderLevel::Indifferently, "Indifferently" },
|
|
||||||
{ ConsiderLevel::Apprehensively, "Apprehensively" },
|
|
||||||
{ ConsiderLevel::Dubiously, "Dubiously" },
|
|
||||||
{ ConsiderLevel::Threateningly, "Threateningly" },
|
|
||||||
{ ConsiderLevel::Scowls, "Scowls" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return consider_level_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetConsiderLevelName(uint8 faction_consider_level)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(faction_consider_level, ConsiderLevel::Ally, ConsiderLevel::Scowls)) {
|
|
||||||
return std::string();;
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetConsiderLevelMap().find(faction_consider_level)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string>& EQ::constants::GetEnvironmentalDamageMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> damage_type_map = {
|
|
||||||
{ EnvironmentalDamage::Lava, "Lava" },
|
|
||||||
{ EnvironmentalDamage::Drowning, "Drowning" },
|
|
||||||
{ EnvironmentalDamage::Falling, "Falling" },
|
|
||||||
{ EnvironmentalDamage::Trap, "Trap" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return damage_type_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetEnvironmentalDamageName(uint8 damage_type)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(damage_type, EnvironmentalDamage::Lava, EnvironmentalDamage::Trap)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetEnvironmentalDamageMap().find(damage_type)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string>& EQ::constants::GetStuckBehaviorMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> stuck_behavior_map = {
|
|
||||||
{ StuckBehavior::RunToTarget, "Run To Target" },
|
|
||||||
{ StuckBehavior::WarpToTarget, "Warp To Target" },
|
|
||||||
{ StuckBehavior::TakeNoAction, "Take No Action" },
|
|
||||||
{ StuckBehavior::EvadeCombat, "Evade Combat" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return stuck_behavior_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetStuckBehaviorName(uint8 behavior_id)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(behavior_id, StuckBehavior::RunToTarget, StuckBehavior::EvadeCombat)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetStuckBehaviorMap().find(behavior_id)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string>& EQ::constants::GetSpawnAnimationMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> spawn_animation_map = {
|
|
||||||
{ SpawnAnimations::Standing, "Standing" },
|
|
||||||
{ SpawnAnimations::Sitting, "Sitting" },
|
|
||||||
{ SpawnAnimations::Crouching, "Crouching" },
|
|
||||||
{ SpawnAnimations::Laying, "Laying" },
|
|
||||||
{ SpawnAnimations::Looting, "Looting" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return spawn_animation_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetSpawnAnimationName(uint8 animation_id)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(animation_id, SpawnAnimations::Standing, SpawnAnimations::Looting)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetSpawnAnimationMap().find(animation_id)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string> &EQ::constants::GetWeatherTypeMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> weather_type_map = {
|
|
||||||
{WeatherTypes::None, "None"},
|
|
||||||
{WeatherTypes::Raining, "Raining"},
|
|
||||||
{WeatherTypes::Snowing, "Snowing"}
|
|
||||||
};
|
|
||||||
|
|
||||||
return weather_type_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetWeatherTypeName(uint8 weather_type)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(weather_type, WeatherTypes::None, WeatherTypes::Snowing)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetWeatherTypeMap().find(weather_type)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string> &EQ::constants::GetEmoteEventTypeMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> emote_event_type_map = {
|
|
||||||
{ EmoteEventTypes::LeaveCombat, "Leave Combat" },
|
|
||||||
{ EmoteEventTypes::EnterCombat, "Enter Combat" },
|
|
||||||
{ EmoteEventTypes::OnDeath, "On Death" },
|
|
||||||
{ EmoteEventTypes::AfterDeath, "After Death" },
|
|
||||||
{ EmoteEventTypes::Hailed, "Hailed" },
|
|
||||||
{ EmoteEventTypes::KilledPC, "Killed PC" },
|
|
||||||
{ EmoteEventTypes::KilledNPC, "Killed NPC" },
|
|
||||||
{ EmoteEventTypes::OnSpawn, "On Spawn" },
|
|
||||||
{ EmoteEventTypes::OnDespawn, "On Despawn" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return emote_event_type_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetEmoteEventTypeName(uint8 emote_event_type)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(emote_event_type, EmoteEventTypes::LeaveCombat, EmoteEventTypes::OnDespawn)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetEmoteEventTypeMap().find(emote_event_type)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint8, std::string> &EQ::constants::GetEmoteTypeMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint8, std::string> emote_type_map = {
|
|
||||||
{ EmoteTypes::Say, "Say" },
|
|
||||||
{ EmoteTypes::Emote, "Emote" },
|
|
||||||
{ EmoteTypes::Shout, "Shout" },
|
|
||||||
{ EmoteTypes::Proximity, "Proximity" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return emote_type_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetEmoteTypeName(uint8 emote_type)
|
|
||||||
{
|
|
||||||
if (!EQ::ValueWithin(emote_type, EmoteTypes::Say, EmoteTypes::Proximity)) {
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
return EQ::constants::GetEmoteTypeMap().find(emote_type)->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint32, std::string>& EQ::constants::GetAppearanceTypeMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint32, std::string> appearance_type_map = {
|
|
||||||
{ AppearanceType::Die, "Die" },
|
|
||||||
{ AppearanceType::WhoLevel, "Who Level" },
|
|
||||||
{ AppearanceType::MaxHealth, "Max Health" },
|
|
||||||
{ AppearanceType::Invisibility, "Invisibility" },
|
|
||||||
{ AppearanceType::PVP, "PVP" },
|
|
||||||
{ AppearanceType::Light, "Light" },
|
|
||||||
{ AppearanceType::Animation, "Animation" },
|
|
||||||
{ AppearanceType::Sneak, "Sneak" },
|
|
||||||
{ AppearanceType::SpawnID, "Spawn ID" },
|
|
||||||
{ AppearanceType::Health, "Health" },
|
|
||||||
{ AppearanceType::Linkdead, "Linkdead" },
|
|
||||||
{ AppearanceType::FlyMode, "Fly Mode" },
|
|
||||||
{ AppearanceType::GM, "GM" },
|
|
||||||
{ AppearanceType::Anonymous, "Anonymous" },
|
|
||||||
{ AppearanceType::GuildID, "Guild ID" },
|
|
||||||
{ AppearanceType::GuildRank, "Guild Rank" },
|
|
||||||
{ AppearanceType::AFK, "AFK" },
|
|
||||||
{ AppearanceType::Pet, "Pet" },
|
|
||||||
{ AppearanceType::Summoned, "Summoned" },
|
|
||||||
{ AppearanceType::Split, "Split" },
|
|
||||||
{ AppearanceType::Size, "Size" },
|
|
||||||
{ AppearanceType::SetType, "Set Type" },
|
|
||||||
{ AppearanceType::NPCName, "NPCName" },
|
|
||||||
{ AppearanceType::AARank, "AARank" },
|
|
||||||
{ AppearanceType::CancelSneakHide, "Cancel Sneak Hide" },
|
|
||||||
{ AppearanceType::AreaHealthRegen, "Area Health Regeneration" },
|
|
||||||
{ AppearanceType::AreaManaRegen, "Area Mana Regeneration" },
|
|
||||||
{ AppearanceType::AreaEnduranceRegen, "Area Endurance Regeneration" },
|
|
||||||
{ AppearanceType::FreezeBeneficialBuffs, "Freeze Beneficial Buffs" },
|
|
||||||
{ AppearanceType::NPCTintIndex, "NPC Tint Index" },
|
|
||||||
{ AppearanceType::GroupAutoConsent, "Group Auto Consent" },
|
|
||||||
{ AppearanceType::RaidAutoConsent, "Raid Auto Consent" },
|
|
||||||
{ AppearanceType::GuildAutoConsent, "Guild Auto Consent" },
|
|
||||||
{ AppearanceType::ShowHelm, "Show Helm" },
|
|
||||||
{ AppearanceType::DamageState, "Damage State" },
|
|
||||||
{ AppearanceType::EQPlayers, "EQ Players" },
|
|
||||||
{ AppearanceType::FindBits, "Find Bits" },
|
|
||||||
{ AppearanceType::TextureType, "Texture Type" },
|
|
||||||
{ AppearanceType::FacePick, "Face Pick" },
|
|
||||||
{ AppearanceType::AntiCheat, "Anti Cheat" },
|
|
||||||
{ AppearanceType::GuildShow, "Guild Show" },
|
|
||||||
{ AppearanceType::OfflineMode, "Offline Mode" }
|
|
||||||
};
|
|
||||||
|
|
||||||
return appearance_type_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetAppearanceTypeName(uint32 appearance_type)
|
|
||||||
{
|
|
||||||
const auto& a = EQ::constants::GetAppearanceTypeMap().find(appearance_type);
|
|
||||||
if (a != EQ::constants::GetAppearanceTypeMap().end()) {
|
|
||||||
return a->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string SpecialAbility::GetName(int ability_id)
|
|
||||||
{
|
|
||||||
return IsValid(ability_id) ? special_ability_names[ability_id] : "UNKNOWN SPECIAL ABILITY";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SpecialAbility::IsValid(int ability_id)
|
|
||||||
{
|
|
||||||
return special_ability_names.find(ability_id) != special_ability_names.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<uint32, std::string>& EQ::constants::GetConsiderColorMap()
|
|
||||||
{
|
|
||||||
static const std::map<uint32, std::string> consider_color_map = {
|
|
||||||
{ ConsiderColor::Green, "Green" },
|
|
||||||
{ ConsiderColor::DarkBlue, "Dark Blue" },
|
|
||||||
{ ConsiderColor::Gray, "Gray" },
|
|
||||||
{ ConsiderColor::White, "White" },
|
|
||||||
{ ConsiderColor::Red, "Red" },
|
|
||||||
{ ConsiderColor::Yellow, "Yellow" },
|
|
||||||
{ ConsiderColor::LightBlue, "Light Blue" },
|
|
||||||
{ ConsiderColor::WhiteTitanium, "White" },
|
|
||||||
};
|
|
||||||
|
|
||||||
return consider_color_map;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string EQ::constants::GetConsiderColorName(uint32 consider_color)
|
|
||||||
{
|
|
||||||
const auto& c = EQ::constants::GetConsiderColorMap().find(consider_color);
|
|
||||||
return c != EQ::constants::GetConsiderColorMap().end() ? c->second : std::string();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string AccountStatus::GetName(uint8 account_status)
|
|
||||||
{
|
|
||||||
for (
|
|
||||||
auto e = account_status_names.rbegin();
|
|
||||||
e != account_status_names.rend();
|
|
||||||
++e
|
|
||||||
) {
|
|
||||||
if (account_status >= e->first) {
|
|
||||||
return e->second;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return "UNKNOWN ACCOUNT STATUS";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string ComparisonType::GetName(uint8 type)
|
|
||||||
{
|
|
||||||
return IsValid(type) ? comparison_types[type] : "UNKNOWN COMPARISON TYPE";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ComparisonType::IsValid(uint8 type)
|
|
||||||
{
|
|
||||||
return comparison_types.find(type) != comparison_types.end();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 LDoNTheme::GetBitmask(uint32 theme_id)
|
|
||||||
{
|
|
||||||
return IsValid(theme_id) ? ldon_theme_names[theme_id].second : LDoNTheme::UnusedBit;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string LDoNTheme::GetName(uint32 theme_id)
|
|
||||||
{
|
|
||||||
return IsValid(theme_id) ? ldon_theme_names[theme_id].first : "UNKNOWN LDON THEME";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LDoNTheme::IsValid(uint32 theme_id)
|
|
||||||
{
|
|
||||||
return ldon_theme_names.find(theme_id) != ldon_theme_names.end();
|
|
||||||
}
|
}
|
||||||
|
|||||||
+95
-551
@@ -1,5 +1,5 @@
|
|||||||
/* EQEMu: Everquest Server Emulator
|
/* EQEMu: Everquest Server Emulator
|
||||||
|
|
||||||
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
This program is free software; you can redistribute it and/or modify
|
||||||
@@ -22,86 +22,19 @@
|
|||||||
|
|
||||||
#include "eq_limits.h"
|
#include "eq_limits.h"
|
||||||
#include "emu_versions.h"
|
#include "emu_versions.h"
|
||||||
#include "bodytypes.h"
|
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
namespace AccountStatus {
|
|
||||||
constexpr uint8 Player = 0;
|
|
||||||
constexpr uint8 Steward = 10;
|
|
||||||
constexpr uint8 ApprenticeGuide = 20;
|
|
||||||
constexpr uint8 Guide = 50;
|
|
||||||
constexpr uint8 QuestTroupe = 80;
|
|
||||||
constexpr uint8 SeniorGuide = 81;
|
|
||||||
constexpr uint8 GMTester = 85;
|
|
||||||
constexpr uint8 EQSupport = 90;
|
|
||||||
constexpr uint8 GMStaff = 95;
|
|
||||||
constexpr uint8 GMAdmin = 100;
|
|
||||||
constexpr uint8 GMLeadAdmin = 150;
|
|
||||||
constexpr uint8 QuestMaster = 160;
|
|
||||||
constexpr uint8 GMAreas = 170;
|
|
||||||
constexpr uint8 GMCoder = 180;
|
|
||||||
constexpr uint8 GMMgmt = 200;
|
|
||||||
constexpr uint8 GMImpossible = 250;
|
|
||||||
constexpr uint8 Max = 255;
|
|
||||||
|
|
||||||
std::string GetName(uint8 account_status);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint8, std::string> account_status_names = {
|
|
||||||
{ AccountStatus::Player, "Player" },
|
|
||||||
{ AccountStatus::Steward, "Steward" },
|
|
||||||
{ AccountStatus::ApprenticeGuide, "Apprentice Guide" },
|
|
||||||
{ AccountStatus::Guide, "Guide" },
|
|
||||||
{ AccountStatus::QuestTroupe, "Quest Troupe" },
|
|
||||||
{ AccountStatus::SeniorGuide, "Senior Guide" },
|
|
||||||
{ AccountStatus::GMTester, "GM Tester" },
|
|
||||||
{ AccountStatus::EQSupport, "EQ Support" },
|
|
||||||
{ AccountStatus::GMStaff, "GM Staff" },
|
|
||||||
{ AccountStatus::GMAdmin, "GM Admin" },
|
|
||||||
{ AccountStatus::GMLeadAdmin, "GM Lead Admin" },
|
|
||||||
{ AccountStatus::QuestMaster, "Quest Master" },
|
|
||||||
{ AccountStatus::GMAreas, "GM Areas" },
|
|
||||||
{ AccountStatus::GMCoder, "GM Coder" },
|
|
||||||
{ AccountStatus::GMMgmt, "GM Mgmt" },
|
|
||||||
{ AccountStatus::GMImpossible, "GM Impossible" },
|
|
||||||
{ AccountStatus::Max, "GM Max" }
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace ComparisonType {
|
|
||||||
constexpr uint8 Equal = 0;
|
|
||||||
constexpr uint8 NotEqual = 1;
|
|
||||||
constexpr uint8 GreaterOrEqual = 2;
|
|
||||||
constexpr uint8 LesserOrEqual = 3;
|
|
||||||
constexpr uint8 Greater = 4;
|
|
||||||
constexpr uint8 Lesser = 5;
|
|
||||||
constexpr uint8 Any = 6;
|
|
||||||
constexpr uint8 NotAny = 7;
|
|
||||||
constexpr uint8 Between = 8;
|
|
||||||
constexpr uint8 NotBetween = 9;
|
|
||||||
|
|
||||||
std::string GetName(uint8 type);
|
|
||||||
bool IsValid(uint8 type);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint8, std::string> comparison_types = {
|
|
||||||
{ ComparisonType::Equal, "Equal" },
|
|
||||||
{ ComparisonType::NotEqual, "Not Equal" },
|
|
||||||
{ ComparisonType::GreaterOrEqual, "Greater or Equal" },
|
|
||||||
{ ComparisonType::LesserOrEqual, "Lesser or Equal" },
|
|
||||||
{ ComparisonType::Greater, "Greater" },
|
|
||||||
{ ComparisonType::Lesser, "Lesser" },
|
|
||||||
{ ComparisonType::Any, "Any" },
|
|
||||||
{ ComparisonType::NotAny, "Not Any" },
|
|
||||||
{ ComparisonType::Between, "Between" },
|
|
||||||
{ ComparisonType::NotBetween, "Not Between" },
|
|
||||||
};
|
|
||||||
|
|
||||||
// local definitions are the result of using hybrid-client or server-only values and methods
|
// local definitions are the result of using hybrid-client or server-only values and methods
|
||||||
namespace EQ
|
namespace EQ
|
||||||
{
|
{
|
||||||
using RoF2::IINVALID;
|
using RoF2::IINVALID;
|
||||||
using RoF2::INULL;
|
using RoF2::INULL;
|
||||||
|
|
||||||
|
namespace inventory {
|
||||||
|
|
||||||
|
} /*inventory*/
|
||||||
|
|
||||||
namespace invtype {
|
namespace invtype {
|
||||||
using namespace RoF2::invtype::enum_;
|
using namespace RoF2::invtype::enum_;
|
||||||
@@ -132,7 +65,7 @@ namespace EQ
|
|||||||
using RoF2::invtype::KRONO_SIZE;
|
using RoF2::invtype::KRONO_SIZE;
|
||||||
using RoF2::invtype::OTHER_SIZE;
|
using RoF2::invtype::OTHER_SIZE;
|
||||||
|
|
||||||
using RoF2::invtype::TRADE_NPC_SIZE;
|
using Titanium::invtype::TRADE_NPC_SIZE;
|
||||||
|
|
||||||
using RoF2::invtype::TYPE_INVALID;
|
using RoF2::invtype::TYPE_INVALID;
|
||||||
using RoF2::invtype::TYPE_BEGIN;
|
using RoF2::invtype::TYPE_BEGIN;
|
||||||
@@ -159,7 +92,7 @@ namespace EQ
|
|||||||
using RoF2::invslot::SLOT_INVALID;
|
using RoF2::invslot::SLOT_INVALID;
|
||||||
using RoF2::invslot::SLOT_BEGIN;
|
using RoF2::invslot::SLOT_BEGIN;
|
||||||
|
|
||||||
using RoF2::invslot::SLOT_TRADESKILL_EXPERIMENT_COMBINE;
|
using Titanium::invslot::SLOT_TRADESKILL_EXPERIMENT_COMBINE;
|
||||||
|
|
||||||
const int16 SLOT_AUGMENT_GENERIC_RETURN = 1001; // clients don't appear to use this method... (internal inventory return value)
|
const int16 SLOT_AUGMENT_GENERIC_RETURN = 1001; // clients don't appear to use this method... (internal inventory return value)
|
||||||
|
|
||||||
@@ -179,28 +112,28 @@ namespace EQ
|
|||||||
using RoF2::invslot::BONUS_STAT_END;
|
using RoF2::invslot::BONUS_STAT_END;
|
||||||
using RoF2::invslot::BONUS_SKILL_END;
|
using RoF2::invslot::BONUS_SKILL_END;
|
||||||
|
|
||||||
using RoF2::invslot::BANK_BEGIN;
|
using Titanium::invslot::BANK_BEGIN;
|
||||||
using RoF2::invslot::BANK_END;
|
using SoF::invslot::BANK_END;
|
||||||
|
|
||||||
using RoF2::invslot::SHARED_BANK_BEGIN;
|
using Titanium::invslot::SHARED_BANK_BEGIN;
|
||||||
using RoF2::invslot::SHARED_BANK_END;
|
using Titanium::invslot::SHARED_BANK_END;
|
||||||
|
|
||||||
using RoF2::invslot::TRADE_BEGIN;
|
using Titanium::invslot::TRADE_BEGIN;
|
||||||
using RoF2::invslot::TRADE_END;
|
using Titanium::invslot::TRADE_END;
|
||||||
|
|
||||||
using RoF2::invslot::TRADE_NPC_END;
|
using Titanium::invslot::TRADE_NPC_END;
|
||||||
|
|
||||||
using RoF2::invslot::WORLD_BEGIN;
|
using Titanium::invslot::WORLD_BEGIN;
|
||||||
using RoF2::invslot::WORLD_END;
|
using Titanium::invslot::WORLD_END;
|
||||||
|
|
||||||
using RoF2::invslot::TRIBUTE_BEGIN;
|
using Titanium::invslot::TRIBUTE_BEGIN;
|
||||||
using RoF2::invslot::TRIBUTE_END;
|
using Titanium::invslot::TRIBUTE_END;
|
||||||
|
|
||||||
using RoF2::invslot::GUILD_TRIBUTE_BEGIN;
|
using Titanium::invslot::GUILD_TRIBUTE_BEGIN;
|
||||||
using RoF2::invslot::GUILD_TRIBUTE_END;
|
using Titanium::invslot::GUILD_TRIBUTE_END;
|
||||||
|
|
||||||
const int16 CORPSE_BEGIN = invslot::slotGeneral1;
|
const int16 CORPSE_BEGIN = invslot::slotGeneral1;
|
||||||
const int16 CORPSE_END = CORPSE_BEGIN + invslot::slotCursor;
|
const int16 CORPSE_END = CORPSE_BEGIN + invslot::slotCursor;
|
||||||
|
|
||||||
using RoF2::invslot::EQUIPMENT_BITMASK;
|
using RoF2::invslot::EQUIPMENT_BITMASK;
|
||||||
using RoF2::invslot::GENERAL_BITMASK;
|
using RoF2::invslot::GENERAL_BITMASK;
|
||||||
@@ -214,40 +147,38 @@ namespace EQ
|
|||||||
} // namespace invslot
|
} // namespace invslot
|
||||||
|
|
||||||
namespace invbag {
|
namespace invbag {
|
||||||
using RoF2::invbag::SLOT_INVALID;
|
using Titanium::invbag::SLOT_INVALID;
|
||||||
using RoF2::invbag::SLOT_BEGIN;
|
using Titanium::invbag::SLOT_BEGIN;
|
||||||
using RoF2::invbag::SLOT_END;
|
using Titanium::invbag::SLOT_END;
|
||||||
using RoF2::invbag::SLOT_COUNT;
|
using Titanium::invbag::SLOT_COUNT;
|
||||||
|
|
||||||
using RoF2::invslot::WORLD_END;
|
using Titanium::invbag::GENERAL_BAGS_BEGIN;
|
||||||
|
|
||||||
const int16 GENERAL_BAGS_BEGIN = WORLD_END + 1;
|
|
||||||
const int16 GENERAL_BAGS_COUNT = invslot::GENERAL_COUNT * SLOT_COUNT;
|
const int16 GENERAL_BAGS_COUNT = invslot::GENERAL_COUNT * SLOT_COUNT;
|
||||||
const int16 GENERAL_BAGS_END = (GENERAL_BAGS_BEGIN + GENERAL_BAGS_COUNT) - 1;
|
const int16 GENERAL_BAGS_END = (GENERAL_BAGS_BEGIN + GENERAL_BAGS_COUNT) - 1;
|
||||||
|
|
||||||
const int16 GENERAL_BAGS_8_COUNT = 8 * SLOT_COUNT;
|
const int16 GENERAL_BAGS_8_COUNT = 8 * SLOT_COUNT;
|
||||||
const int16 GENERAL_BAGS_8_END = (GENERAL_BAGS_BEGIN + GENERAL_BAGS_8_COUNT) - 1;
|
const int16 GENERAL_BAGS_8_END = (GENERAL_BAGS_BEGIN + GENERAL_BAGS_8_COUNT) - 1;
|
||||||
|
|
||||||
const int16 CURSOR_BAG_BEGIN = GENERAL_BAGS_END + 1;
|
const int16 CURSOR_BAG_BEGIN = 351;
|
||||||
const int16 CURSOR_BAG_COUNT = SLOT_COUNT;
|
const int16 CURSOR_BAG_COUNT = SLOT_COUNT;
|
||||||
const int16 CURSOR_BAG_END = (CURSOR_BAG_BEGIN + CURSOR_BAG_COUNT) - 1;
|
const int16 CURSOR_BAG_END = (CURSOR_BAG_BEGIN + CURSOR_BAG_COUNT) - 1;
|
||||||
|
|
||||||
const int16 BANK_BAGS_BEGIN = CURSOR_BAG_END + 1;
|
using Titanium::invbag::BANK_BAGS_BEGIN;
|
||||||
const int16 BANK_BAGS_COUNT = (invtype::BANK_SIZE * SLOT_COUNT);
|
const int16 BANK_BAGS_COUNT = (invtype::BANK_SIZE * SLOT_COUNT);
|
||||||
const int16 BANK_BAGS_END = (BANK_BAGS_BEGIN + BANK_BAGS_COUNT) - 1;
|
const int16 BANK_BAGS_END = (BANK_BAGS_BEGIN + BANK_BAGS_COUNT) - 1;
|
||||||
|
|
||||||
const int16 BANK_BAGS_16_COUNT = 16 * SLOT_COUNT;
|
const int16 BANK_BAGS_16_COUNT = 16 * SLOT_COUNT;
|
||||||
const int16 BANK_BAGS_16_END = (BANK_BAGS_BEGIN + BANK_BAGS_16_COUNT) - 1;
|
const int16 BANK_BAGS_16_END = (BANK_BAGS_BEGIN + BANK_BAGS_16_COUNT) - 1;
|
||||||
|
|
||||||
const int16 SHARED_BANK_BAGS_BEGIN = BANK_BAGS_END + 1;
|
using Titanium::invbag::SHARED_BANK_BAGS_BEGIN;
|
||||||
const int16 SHARED_BANK_BAGS_COUNT = invtype::SHARED_BANK_SIZE * SLOT_COUNT;
|
const int16 SHARED_BANK_BAGS_COUNT = invtype::SHARED_BANK_SIZE * SLOT_COUNT;
|
||||||
const int16 SHARED_BANK_BAGS_END = (SHARED_BANK_BAGS_BEGIN + SHARED_BANK_BAGS_COUNT) - 1;
|
const int16 SHARED_BANK_BAGS_END = (SHARED_BANK_BAGS_BEGIN + SHARED_BANK_BAGS_COUNT) - 1;
|
||||||
|
|
||||||
const int16 TRADE_BAGS_BEGIN = SHARED_BANK_BAGS_END + 1;
|
using Titanium::invbag::TRADE_BAGS_BEGIN;
|
||||||
const int16 TRADE_BAGS_COUNT = invtype::TRADE_SIZE * SLOT_COUNT;
|
const int16 TRADE_BAGS_COUNT = invtype::TRADE_SIZE * SLOT_COUNT;
|
||||||
const int16 TRADE_BAGS_END = (TRADE_BAGS_BEGIN + TRADE_BAGS_COUNT) - 1;
|
const int16 TRADE_BAGS_END = (TRADE_BAGS_BEGIN + TRADE_BAGS_COUNT) - 1;
|
||||||
|
|
||||||
using RoF2::invbag::GetInvBagIndexName;
|
using Titanium::invbag::GetInvBagIndexName;
|
||||||
|
|
||||||
} // namespace invbag
|
} // namespace invbag
|
||||||
|
|
||||||
@@ -269,127 +200,39 @@ namespace EQ
|
|||||||
using RoF2::constants::EXPANSIONS_MASK;
|
using RoF2::constants::EXPANSIONS_MASK;
|
||||||
|
|
||||||
using RoF2::constants::CHARACTER_CREATION_LIMIT;
|
using RoF2::constants::CHARACTER_CREATION_LIMIT;
|
||||||
|
|
||||||
const size_t SAY_LINK_OPENER_SIZE = 1;
|
const size_t SAY_LINK_OPENER_SIZE = 1;
|
||||||
using RoF2::constants::SAY_LINK_BODY_SIZE;
|
using RoF2::constants::SAY_LINK_BODY_SIZE;
|
||||||
const size_t SAY_LINK_TEXT_SIZE = 256; // this may be varied until it breaks something (tested:374) - the others are constant
|
const size_t SAY_LINK_TEXT_SIZE = 256; // this may be varied until it breaks something (tested:374) - the others are constant
|
||||||
const size_t SAY_LINK_CLOSER_SIZE = 1;
|
const size_t SAY_LINK_CLOSER_SIZE = 1;
|
||||||
const size_t SAY_LINK_MAXIMUM_SIZE = (SAY_LINK_OPENER_SIZE + SAY_LINK_BODY_SIZE + SAY_LINK_TEXT_SIZE + SAY_LINK_CLOSER_SIZE);
|
const size_t SAY_LINK_MAXIMUM_SIZE = (SAY_LINK_OPENER_SIZE + SAY_LINK_BODY_SIZE + SAY_LINK_TEXT_SIZE + SAY_LINK_CLOSER_SIZE);
|
||||||
|
|
||||||
enum BotSpellIDs : int {
|
enum StanceType : int {
|
||||||
Warrior = 3001,
|
stanceUnknown = 0,
|
||||||
Cleric,
|
stancePassive,
|
||||||
Paladin,
|
stanceBalanced,
|
||||||
Ranger,
|
stanceEfficient,
|
||||||
Shadowknight,
|
stanceReactive,
|
||||||
Druid,
|
stanceAggressive,
|
||||||
Monk,
|
stanceAssist,
|
||||||
Bard,
|
stanceBurn,
|
||||||
Rogue,
|
stanceEfficient2,
|
||||||
Shaman,
|
stanceBurnAE
|
||||||
Necromancer,
|
|
||||||
Wizard,
|
|
||||||
Magician,
|
|
||||||
Enchanter,
|
|
||||||
Beastlord,
|
|
||||||
Berserker
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum GravityBehavior : int8 {
|
const char *GetStanceName(StanceType stance_type);
|
||||||
Ground,
|
int ConvertStanceTypeToIndex(StanceType stance_type);
|
||||||
Flying,
|
|
||||||
Levitating,
|
|
||||||
Water,
|
|
||||||
Floating,
|
|
||||||
LevitateWhileRunning
|
|
||||||
};
|
|
||||||
|
|
||||||
enum EnvironmentalDamage : uint8 {
|
const int STANCE_TYPE_FIRST = stancePassive;
|
||||||
Lava = 250,
|
const int STANCE_TYPE_LAST = stanceBurnAE;
|
||||||
Drowning,
|
const int STANCE_TYPE_COUNT = stanceBurnAE;
|
||||||
Falling,
|
|
||||||
Trap
|
|
||||||
};
|
|
||||||
|
|
||||||
enum StuckBehavior : uint8 {
|
|
||||||
RunToTarget,
|
|
||||||
WarpToTarget,
|
|
||||||
TakeNoAction,
|
|
||||||
EvadeCombat
|
|
||||||
};
|
|
||||||
|
|
||||||
enum SpawnAnimations : uint8 {
|
|
||||||
Standing,
|
|
||||||
Sitting,
|
|
||||||
Crouching,
|
|
||||||
Laying,
|
|
||||||
Looting
|
|
||||||
};
|
|
||||||
|
|
||||||
enum WeatherTypes : uint8 {
|
|
||||||
None,
|
|
||||||
Raining,
|
|
||||||
Snowing
|
|
||||||
};
|
|
||||||
|
|
||||||
enum EmoteEventTypes : uint8 {
|
|
||||||
LeaveCombat,
|
|
||||||
EnterCombat,
|
|
||||||
OnDeath,
|
|
||||||
AfterDeath,
|
|
||||||
Hailed,
|
|
||||||
KilledPC,
|
|
||||||
KilledNPC,
|
|
||||||
OnSpawn,
|
|
||||||
OnDespawn
|
|
||||||
};
|
|
||||||
|
|
||||||
enum EmoteTypes : uint8 {
|
|
||||||
Say,
|
|
||||||
Emote,
|
|
||||||
Shout,
|
|
||||||
Proximity
|
|
||||||
};
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetLanguageMap();
|
|
||||||
std::string GetLanguageName(uint8 language_id);
|
|
||||||
|
|
||||||
extern const std::map<int8, std::string>& GetFlyModeMap();
|
|
||||||
std::string GetFlyModeName(int8 flymode_id);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetConsiderLevelMap();
|
|
||||||
std::string GetConsiderLevelName(uint8 consider_level);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetEnvironmentalDamageMap();
|
|
||||||
std::string GetEnvironmentalDamageName(uint8 damage_type);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetStuckBehaviorMap();
|
|
||||||
std::string GetStuckBehaviorName(uint8 behavior_id);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetSpawnAnimationMap();
|
|
||||||
std::string GetSpawnAnimationName(uint8 animation_id);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetWeatherTypeMap();
|
|
||||||
std::string GetWeatherTypeName(uint8 weather_type);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetEmoteEventTypeMap();
|
|
||||||
std::string GetEmoteEventTypeName(uint8 emote_event_type);
|
|
||||||
|
|
||||||
extern const std::map<uint8, std::string>& GetEmoteTypeMap();
|
|
||||||
std::string GetEmoteTypeName(uint8 emote_type);
|
|
||||||
|
|
||||||
extern const std::map<uint32, std::string>& GetAppearanceTypeMap();
|
|
||||||
std::string GetAppearanceTypeName(uint32 animation_type);
|
|
||||||
|
|
||||||
extern const std::map<uint32, std::string>& GetConsiderColorMap();
|
|
||||||
std::string GetConsiderColorName(uint32 consider_color);
|
|
||||||
|
|
||||||
} /*constants*/
|
} /*constants*/
|
||||||
|
|
||||||
namespace profile {
|
namespace profile {
|
||||||
using RoF2::profile::BANDOLIERS_SIZE;
|
using RoF2::profile::BANDOLIERS_SIZE;
|
||||||
using RoF2::profile::BANDOLIER_ITEM_COUNT;
|
using RoF2::profile::BANDOLIER_ITEM_COUNT;
|
||||||
|
|
||||||
using RoF2::profile::POTION_BELT_SIZE;
|
using RoF2::profile::POTION_BELT_SIZE;
|
||||||
|
|
||||||
using RoF2::profile::SKILL_ARRAY_SIZE;
|
using RoF2::profile::SKILL_ARRAY_SIZE;
|
||||||
@@ -437,6 +280,37 @@ namespace EQ
|
|||||||
|
|
||||||
} // namespace spells
|
} // namespace spells
|
||||||
|
|
||||||
|
namespace bug {
|
||||||
|
enum CategoryID : uint32 {
|
||||||
|
catOther = 0,
|
||||||
|
catVideo,
|
||||||
|
catAudio,
|
||||||
|
catPathing,
|
||||||
|
catQuest,
|
||||||
|
catTradeskills,
|
||||||
|
catSpellStacking,
|
||||||
|
catDoorsPortals,
|
||||||
|
catItems,
|
||||||
|
catNPC,
|
||||||
|
catDialogs,
|
||||||
|
catLoNTCG,
|
||||||
|
catMercenaries
|
||||||
|
};
|
||||||
|
|
||||||
|
enum OptionalInfoFlag : uint32 {
|
||||||
|
infoNoOptionalInfo = 0x0,
|
||||||
|
infoCanDuplicate = 0x1,
|
||||||
|
infoCrashBug = 0x2,
|
||||||
|
infoTargetInfo = 0x4,
|
||||||
|
infoCharacterFlags = 0x8,
|
||||||
|
infoUnknownValue = 0xFFFFFFF0
|
||||||
|
};
|
||||||
|
|
||||||
|
const char* CategoryIDToCategoryName(CategoryID category_id);
|
||||||
|
CategoryID CategoryNameToCategoryID(const char* category_name);
|
||||||
|
|
||||||
|
} // namespace bug
|
||||||
|
|
||||||
enum WaypointStatus : int {
|
enum WaypointStatus : int {
|
||||||
RoamBoxPauseInProgress = -3,
|
RoamBoxPauseInProgress = -3,
|
||||||
QuestControlNoGrid = -2,
|
QuestControlNoGrid = -2,
|
||||||
@@ -450,344 +324,14 @@ namespace EQ
|
|||||||
Raid,
|
Raid,
|
||||||
Guild
|
Guild
|
||||||
};
|
};
|
||||||
};
|
}; // namespace consent
|
||||||
|
|
||||||
} /*EQEmu*/
|
} /*EQEmu*/
|
||||||
|
|
||||||
enum ServerLockType : int {
|
|
||||||
List,
|
|
||||||
Lock,
|
|
||||||
Unlock
|
|
||||||
};
|
|
||||||
|
|
||||||
enum Invisibility : uint8 {
|
|
||||||
Visible,
|
|
||||||
Invisible,
|
|
||||||
Special = 255
|
|
||||||
};
|
|
||||||
|
|
||||||
enum AugmentActions : int {
|
|
||||||
Insert,
|
|
||||||
Remove,
|
|
||||||
Swap,
|
|
||||||
Destroy
|
|
||||||
};
|
|
||||||
|
|
||||||
enum ConsiderLevel : uint8 {
|
|
||||||
Ally = 1,
|
|
||||||
Warmly,
|
|
||||||
Kindly,
|
|
||||||
Amiably,
|
|
||||||
Indifferently,
|
|
||||||
Apprehensively,
|
|
||||||
Dubiously,
|
|
||||||
Threateningly,
|
|
||||||
Scowls
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace ConsiderColor {
|
|
||||||
constexpr uint32 Green = 2;
|
|
||||||
constexpr uint32 DarkBlue = 4;
|
|
||||||
constexpr uint32 Gray = 6;
|
|
||||||
constexpr uint32 White = 10;
|
|
||||||
constexpr uint32 Red = 13;
|
|
||||||
constexpr uint32 Yellow = 15;
|
|
||||||
constexpr uint32 LightBlue = 18;
|
|
||||||
constexpr uint32 WhiteTitanium = 20;
|
|
||||||
};
|
|
||||||
|
|
||||||
enum TargetDescriptionType : uint8 {
|
|
||||||
LCSelf,
|
|
||||||
UCSelf,
|
|
||||||
LCYou,
|
|
||||||
UCYou,
|
|
||||||
LCYour,
|
|
||||||
UCYour
|
|
||||||
};
|
|
||||||
|
|
||||||
enum ReloadWorld : uint8 {
|
|
||||||
NoRepop = 0,
|
|
||||||
Repop,
|
|
||||||
ForceRepop
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class EntityFilterType {
|
|
||||||
All,
|
|
||||||
Bots,
|
|
||||||
Clients,
|
|
||||||
NPCs
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class ApplySpellType {
|
|
||||||
Solo,
|
|
||||||
Group,
|
|
||||||
Raid
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace SpecialAbility {
|
|
||||||
constexpr int Summon = 1;
|
|
||||||
constexpr int Enrage = 2;
|
|
||||||
constexpr int Rampage = 3;
|
|
||||||
constexpr int AreaRampage = 4;
|
|
||||||
constexpr int Flurry = 5;
|
|
||||||
constexpr int TripleAttack = 6;
|
|
||||||
constexpr int QuadrupleAttack = 7;
|
|
||||||
constexpr int DualWield = 8;
|
|
||||||
constexpr int BaneAttack = 9;
|
|
||||||
constexpr int MagicalAttack = 10;
|
|
||||||
constexpr int RangedAttack = 11;
|
|
||||||
constexpr int SlowImmunity = 12;
|
|
||||||
constexpr int MesmerizeImmunity = 13;
|
|
||||||
constexpr int CharmImmunity = 14;
|
|
||||||
constexpr int StunImmunity = 15;
|
|
||||||
constexpr int SnareImmunity = 16;
|
|
||||||
constexpr int FearImmunity = 17;
|
|
||||||
constexpr int DispellImmunity = 18;
|
|
||||||
constexpr int MeleeImmunity = 19;
|
|
||||||
constexpr int MagicImmunity = 20;
|
|
||||||
constexpr int FleeingImmunity = 21;
|
|
||||||
constexpr int MeleeImmunityExceptBane = 22;
|
|
||||||
constexpr int MeleeImmunityExceptMagical = 23;
|
|
||||||
constexpr int AggroImmunity = 24;
|
|
||||||
constexpr int BeingAggroImmunity = 25;
|
|
||||||
constexpr int CastingFromRangeImmunity = 26;
|
|
||||||
constexpr int FeignDeathImmunity = 27;
|
|
||||||
constexpr int TauntImmunity = 28;
|
|
||||||
constexpr int TunnelVision = 29;
|
|
||||||
constexpr int NoBuffHealFriends = 30;
|
|
||||||
constexpr int PacifyImmunity = 31;
|
|
||||||
constexpr int Leash = 32;
|
|
||||||
constexpr int Tether = 33;
|
|
||||||
constexpr int DestructibleObject = 34;
|
|
||||||
constexpr int HarmFromClientImmunity = 35;
|
|
||||||
constexpr int AlwaysFlee = 36;
|
|
||||||
constexpr int FleePercent = 37;
|
|
||||||
constexpr int AllowBeneficial = 38;
|
|
||||||
constexpr int DisableMelee = 39;
|
|
||||||
constexpr int NPCChaseDistance = 40;
|
|
||||||
constexpr int AllowedToTank = 41;
|
|
||||||
constexpr int IgnoreRootAggroRules = 42;
|
|
||||||
constexpr int CastingResistDifficulty = 43;
|
|
||||||
constexpr int CounterAvoidDamage = 44;
|
|
||||||
constexpr int ProximityAggro = 45;
|
|
||||||
constexpr int RangedAttackImmunity = 46;
|
|
||||||
constexpr int ClientDamageImmunity = 47;
|
|
||||||
constexpr int NPCDamageImmunity = 48;
|
|
||||||
constexpr int ClientAggroImmunity = 49;
|
|
||||||
constexpr int NPCAggroImmunity = 50;
|
|
||||||
constexpr int ModifyAvoidDamage = 51;
|
|
||||||
constexpr int MemoryFadeImmunity = 52;
|
|
||||||
constexpr int OpenImmunity = 53;
|
|
||||||
constexpr int AssassinateImmunity = 54;
|
|
||||||
constexpr int HeadshotImmunity = 55;
|
|
||||||
constexpr int BotAggroImmunity = 56;
|
|
||||||
constexpr int BotDamageImmunity = 57;
|
|
||||||
constexpr int Max = 58;
|
|
||||||
|
|
||||||
constexpr int MaxParameters = 9;
|
|
||||||
|
|
||||||
std::string GetName(int ability_id);
|
|
||||||
bool IsValid(int ability_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<int, std::string> special_ability_names = {
|
|
||||||
{ SpecialAbility::Summon, "Summon" },
|
|
||||||
{ SpecialAbility::Enrage, "Enrage" },
|
|
||||||
{ SpecialAbility::Rampage, "Rampage" },
|
|
||||||
{ SpecialAbility::AreaRampage, "Area Rampage" },
|
|
||||||
{ SpecialAbility::Flurry, "Flurry" },
|
|
||||||
{ SpecialAbility::TripleAttack, "Triple Attack" },
|
|
||||||
{ SpecialAbility::QuadrupleAttack, "Quadruple Attack" },
|
|
||||||
{ SpecialAbility::DualWield, "Dual Wield" },
|
|
||||||
{ SpecialAbility::BaneAttack, "Bane Attack" },
|
|
||||||
{ SpecialAbility::MagicalAttack, "Magical Attack" },
|
|
||||||
{ SpecialAbility::RangedAttack, "Ranged Attack" },
|
|
||||||
{ SpecialAbility::SlowImmunity, "Immune to Slow" },
|
|
||||||
{ SpecialAbility::MesmerizeImmunity, "Immune to Mesmerize" },
|
|
||||||
{ SpecialAbility::CharmImmunity, "Immune to Charm" },
|
|
||||||
{ SpecialAbility::StunImmunity, "Immune to Stun" },
|
|
||||||
{ SpecialAbility::SnareImmunity, "Immune to Snare" },
|
|
||||||
{ SpecialAbility::FearImmunity, "Immune to Fear" },
|
|
||||||
{ SpecialAbility::DispellImmunity, "Immune to Dispell" },
|
|
||||||
{ SpecialAbility::MeleeImmunity, "Immune to Melee" },
|
|
||||||
{ SpecialAbility::MagicImmunity, "Immune to Magic" },
|
|
||||||
{ SpecialAbility::FleeingImmunity, "Immune to Fleeing" },
|
|
||||||
{ SpecialAbility::MeleeImmunityExceptBane, "Immune to Melee except Bane" },
|
|
||||||
{ SpecialAbility::MeleeImmunityExceptMagical, "Immune to Non-Magical Melee" },
|
|
||||||
{ SpecialAbility::AggroImmunity, "Immune to Aggro" },
|
|
||||||
{ SpecialAbility::BeingAggroImmunity, "Immune to Being Aggro" },
|
|
||||||
{ SpecialAbility::CastingFromRangeImmunity, "Immune to Ranged Spells" },
|
|
||||||
{ SpecialAbility::FeignDeathImmunity, "Immune to Feign Death" },
|
|
||||||
{ SpecialAbility::TauntImmunity, "Immune to Taunt" },
|
|
||||||
{ SpecialAbility::TunnelVision, "Tunnel Vision" },
|
|
||||||
{ SpecialAbility::NoBuffHealFriends, "Does Not Heal or Buff Allies" },
|
|
||||||
{ SpecialAbility::PacifyImmunity, "Immune to Pacify" },
|
|
||||||
{ SpecialAbility::Leash, "Leashed" },
|
|
||||||
{ SpecialAbility::Tether, "Tethered" },
|
|
||||||
{ SpecialAbility::DestructibleObject, "Destructible Object" },
|
|
||||||
{ SpecialAbility::HarmFromClientImmunity, "Immune to Harm from Client" },
|
|
||||||
{ SpecialAbility::AlwaysFlee, "Always Flees" },
|
|
||||||
{ SpecialAbility::FleePercent, "Flee Percentage" },
|
|
||||||
{ SpecialAbility::AllowBeneficial, "Allows Beneficial Spells" },
|
|
||||||
{ SpecialAbility::DisableMelee, "Melee is Disabled" },
|
|
||||||
{ SpecialAbility::NPCChaseDistance, "Chase Distance" },
|
|
||||||
{ SpecialAbility::AllowedToTank, "Allowed to Tank" },
|
|
||||||
{ SpecialAbility::IgnoreRootAggroRules, "Ignores Root Aggro" },
|
|
||||||
{ SpecialAbility::CastingResistDifficulty, "Casting Resist Difficulty" },
|
|
||||||
{ SpecialAbility::CounterAvoidDamage, "Counter Damage Avoidance" },
|
|
||||||
{ SpecialAbility::ProximityAggro, "Proximity Aggro" },
|
|
||||||
{ SpecialAbility::RangedAttackImmunity, "Immune to Ranged Attacks" },
|
|
||||||
{ SpecialAbility::ClientDamageImmunity, "Immune to Client Damage" },
|
|
||||||
{ SpecialAbility::NPCDamageImmunity, "Immune to NPC Damage" },
|
|
||||||
{ SpecialAbility::ClientAggroImmunity, "Immune to Client Aggro" },
|
|
||||||
{ SpecialAbility::NPCAggroImmunity, "Immune to NPC Aggro" },
|
|
||||||
{ SpecialAbility::ModifyAvoidDamage, "Modify Damage Avoidance" },
|
|
||||||
{ SpecialAbility::MemoryFadeImmunity, "Immune to Memory Fades" },
|
|
||||||
{ SpecialAbility::OpenImmunity, "Immune to Open" },
|
|
||||||
{ SpecialAbility::AssassinateImmunity, "Immune to Assassinate" },
|
|
||||||
{ SpecialAbility::HeadshotImmunity, "Immune to Headshot" },
|
|
||||||
{ SpecialAbility::BotAggroImmunity, "Immune to Bot Aggro" },
|
|
||||||
{ SpecialAbility::BotDamageImmunity, "Immune to Bot Damage" },
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace HeroicBonusBucket
|
|
||||||
{
|
|
||||||
const std::string WisMaxMana = "HWIS-MaxMana";
|
|
||||||
const std::string WisManaRegen = "HWIS-ManaRegen";
|
|
||||||
const std::string WisHealAmt = "HWIS-HealAmt";
|
|
||||||
const std::string IntMaxMana = "HINT-MaxMana";
|
|
||||||
const std::string IntManaRegen = "HINT-ManaRegen";
|
|
||||||
const std::string IntSpellDmg = "HINT-SpellDmg";
|
|
||||||
const std::string StrMeleeDamage = "HSTR-MeleeDamage";
|
|
||||||
const std::string StrShieldAC = "HSTR-ShieldAC";
|
|
||||||
const std::string StrMaxEndurance = "HSTR-MaxEndurance";
|
|
||||||
const std::string StrEnduranceRegen = "HSTR-EnduranceRegen";
|
|
||||||
const std::string StaMaxHP = "HSTA-MaxHP";
|
|
||||||
const std::string StaHPRegen = "HSTA-HPRegen";
|
|
||||||
const std::string StaMaxEndurance = "HSTA-MaxEndurance";
|
|
||||||
const std::string StaEnduranceRegen = "HSTA-EnduranceRegen";
|
|
||||||
const std::string AgiAvoidance = "HAGI-Avoidance";
|
|
||||||
const std::string AgiMaxEndurance = "HAGI-MaxEndurance";
|
|
||||||
const std::string AgiEnduranceRegen = "HAGI-EnduranceRegen";
|
|
||||||
const std::string DexRangedDamage = "HDEX-RangedDamage";
|
|
||||||
const std::string DexMaxEndurance = "HDEX-MaxEndurance";
|
|
||||||
const std::string DexEnduranceRegen = "HDEX-EnduranceRegen";
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Bug {
|
|
||||||
namespace Category {
|
|
||||||
constexpr uint32 Other = 0;
|
|
||||||
constexpr uint32 Video = 1;
|
|
||||||
constexpr uint32 Audio = 2;
|
|
||||||
constexpr uint32 Pathing = 3;
|
|
||||||
constexpr uint32 Quest = 4;
|
|
||||||
constexpr uint32 Tradeskills = 5;
|
|
||||||
constexpr uint32 SpellStacking = 6;
|
|
||||||
constexpr uint32 DoorsPortals = 7;
|
|
||||||
constexpr uint32 Items = 8;
|
|
||||||
constexpr uint32 NPC = 9;
|
|
||||||
constexpr uint32 Dialogs = 10;
|
|
||||||
constexpr uint32 LoNTCG = 11;
|
|
||||||
constexpr uint32 Mercenaries = 12;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace InformationFlag {
|
|
||||||
constexpr uint32 None = 0;
|
|
||||||
constexpr uint32 Repeatable = 1;
|
|
||||||
constexpr uint32 Crash = 2;
|
|
||||||
constexpr uint32 TargetInfo = 4;
|
|
||||||
constexpr uint32 CharacterFlags = 8;
|
|
||||||
constexpr uint32 Unknown = 4294967280;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 GetID(const std::string& category_name);
|
|
||||||
std::string GetName(uint32 category_id);
|
|
||||||
bool IsValid(uint32 category_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint32, std::string> bug_category_names = {
|
|
||||||
{ Bug::Category::Other, "Other" },
|
|
||||||
{ Bug::Category::Video, "Video" },
|
|
||||||
{ Bug::Category::Audio, "Audio" },
|
|
||||||
{ Bug::Category::Pathing, "Pathing" },
|
|
||||||
{ Bug::Category::Quest, "Quest" },
|
|
||||||
{ Bug::Category::Tradeskills, "Tradeskills" },
|
|
||||||
{ Bug::Category::SpellStacking, "Spell Stacking" },
|
|
||||||
{ Bug::Category::DoorsPortals, "Doors and Portals" },
|
|
||||||
{ Bug::Category::Items, "Items" },
|
|
||||||
{ Bug::Category::NPC, "NPC" },
|
|
||||||
{ Bug::Category::Dialogs, "Dialogs" },
|
|
||||||
{ Bug::Category::LoNTCG, "LoN - TCG" },
|
|
||||||
{ Bug::Category::Mercenaries, "Mercenaries" }
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace Stance {
|
|
||||||
constexpr uint32 Unknown = 0;
|
|
||||||
constexpr uint32 Passive = 1;
|
|
||||||
constexpr uint32 Balanced = 2;
|
|
||||||
constexpr uint32 Efficient = 3;
|
|
||||||
constexpr uint32 Reactive = 4;
|
|
||||||
constexpr uint32 Aggressive = 5;
|
|
||||||
constexpr uint32 Assist = 6;
|
|
||||||
constexpr uint32 Burn = 7;
|
|
||||||
constexpr uint32 Efficient2 = 8;
|
|
||||||
constexpr uint32 AEBurn = 9;
|
|
||||||
|
|
||||||
std::string GetName(uint8 stance_id);
|
|
||||||
uint8 GetIndex(uint8 stance_id);
|
|
||||||
bool IsValid(uint8 stance_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint32, std::string> stance_names = {
|
|
||||||
{ Stance::Unknown, "Unknown" },
|
|
||||||
{ Stance::Passive, "Passive" },
|
|
||||||
{ Stance::Balanced, "Balanced" },
|
|
||||||
{ Stance::Efficient, "Efficient" },
|
|
||||||
{ Stance::Reactive, "Reactive" },
|
|
||||||
{ Stance::Aggressive, "Aggressive" },
|
|
||||||
{ Stance::Assist, "Assist" },
|
|
||||||
{ Stance::Burn, "Burn" },
|
|
||||||
{ Stance::Efficient2, "Efficient" },
|
|
||||||
{ Stance::AEBurn, "AE Burn" }
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace LDoNTheme {
|
|
||||||
constexpr uint32 Unused = 0;
|
|
||||||
constexpr uint32 GUK = 1;
|
|
||||||
constexpr uint32 MIR = 2;
|
|
||||||
constexpr uint32 MMC = 3;
|
|
||||||
constexpr uint32 RUJ = 4;
|
|
||||||
constexpr uint32 TAK = 5;
|
|
||||||
|
|
||||||
constexpr uint32 UnusedBit = 0;
|
|
||||||
constexpr uint32 GUKBit = 1;
|
|
||||||
constexpr uint32 MIRBit = 2;
|
|
||||||
constexpr uint32 MMCBit = 4;
|
|
||||||
constexpr uint32 RUJBit = 8;
|
|
||||||
constexpr uint32 TAKBit = 16;
|
|
||||||
|
|
||||||
uint32 GetBitmask(uint32 theme_id);
|
|
||||||
std::string GetName(uint32 theme_id);
|
|
||||||
bool IsValid(uint32 theme_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::map<uint32, std::pair<std::string, uint32>> ldon_theme_names = {
|
|
||||||
{ LDoNTheme::Unused, { "Unused", LDoNTheme::UnusedBit }, },
|
|
||||||
{ LDoNTheme::GUK, { "Deepest Guk", LDoNTheme::GUKBit }, },
|
|
||||||
{ LDoNTheme::MIR, { "Miragul's Menagerie", LDoNTheme::MIRBit }, },
|
|
||||||
{ LDoNTheme::MMC, { "Mistmoore Catacombs", LDoNTheme::MMCBit }, },
|
|
||||||
{ LDoNTheme::RUJ, { "Rujarkian Hills", LDoNTheme::RUJBit }, },
|
|
||||||
{ LDoNTheme::TAK, { "Takish-Hiz", LDoNTheme::TAKBit }, },
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace PCNPCOnlyFlagType {
|
|
||||||
constexpr int PC = 1;
|
|
||||||
constexpr int NPC = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace BookType {
|
|
||||||
constexpr uint8 Scroll = 0;
|
|
||||||
constexpr uint8 Book = 1;
|
|
||||||
constexpr uint8 ItemInfo = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /*COMMON_EMU_CONSTANTS_H*/
|
#endif /*COMMON_EMU_CONSTANTS_H*/
|
||||||
|
|
||||||
|
/* hack list to prevent circular references
|
||||||
|
|
||||||
|
eq_limits.h:EQ::inventory::LookupEntry::InventoryTypeSize[n];
|
||||||
|
|
||||||
|
*/
|
||||||
|
|||||||
@@ -45,7 +45,6 @@ typedef enum { //EQEmu internal opcodes list
|
|||||||
_maxEmuOpcode
|
_maxEmuOpcode
|
||||||
} EmuOpcode;
|
} EmuOpcode;
|
||||||
|
|
||||||
constexpr int format_as(EmuOpcode opcode) { return static_cast<int>(opcode); }
|
|
||||||
extern const char *OpcodeNames[_maxEmuOpcode+1];
|
extern const char *OpcodeNames[_maxEmuOpcode+1];
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
+6
-63
@@ -35,7 +35,7 @@ N(OP_AltCurrencyMerchantRequest),
|
|||||||
N(OP_AltCurrencyPurchase),
|
N(OP_AltCurrencyPurchase),
|
||||||
N(OP_AltCurrencyReclaim),
|
N(OP_AltCurrencyReclaim),
|
||||||
N(OP_AltCurrencySell),
|
N(OP_AltCurrencySell),
|
||||||
N(OP_AltCurrencySellSelection), // Used by eqstr_us.txt 8066, 8068, 8069
|
N(OP_AltCurrencySellSelection),
|
||||||
N(OP_Animation),
|
N(OP_Animation),
|
||||||
N(OP_AnnoyingZoneUnknown),
|
N(OP_AnnoyingZoneUnknown),
|
||||||
N(OP_ApplyPoison),
|
N(OP_ApplyPoison),
|
||||||
@@ -62,22 +62,18 @@ N(OP_BeginCast),
|
|||||||
N(OP_Bind_Wound),
|
N(OP_Bind_Wound),
|
||||||
N(OP_BlockedBuffs),
|
N(OP_BlockedBuffs),
|
||||||
N(OP_BoardBoat),
|
N(OP_BoardBoat),
|
||||||
N(OP_BookButton),
|
|
||||||
N(OP_Buff),
|
N(OP_Buff),
|
||||||
N(OP_BuffCreate),
|
N(OP_BuffCreate),
|
||||||
N(OP_BuffRemoveRequest),
|
N(OP_BuffRemoveRequest),
|
||||||
N(OP_Bug),
|
N(OP_Bug),
|
||||||
N(OP_BuyerItems),
|
|
||||||
N(OP_CameraEffect),
|
N(OP_CameraEffect),
|
||||||
N(OP_Camp),
|
N(OP_Camp),
|
||||||
N(OP_CancelSneakHide),
|
N(OP_CancelSneakHide),
|
||||||
N(OP_CancelTask),
|
N(OP_CancelTask),
|
||||||
N(OP_CancelTrade),
|
N(OP_CancelTrade),
|
||||||
N(OP_CashReward),
|
|
||||||
N(OP_CastSpell),
|
N(OP_CastSpell),
|
||||||
N(OP_ChangeSize),
|
N(OP_ChangeSize),
|
||||||
N(OP_ChannelMessage),
|
N(OP_ChannelMessage),
|
||||||
N(OP_ChangePetName),
|
|
||||||
N(OP_CharacterCreate),
|
N(OP_CharacterCreate),
|
||||||
N(OP_CharacterCreateRequest),
|
N(OP_CharacterCreateRequest),
|
||||||
N(OP_CharInventory),
|
N(OP_CharInventory),
|
||||||
@@ -133,8 +129,8 @@ N(OP_DisciplineTimer),
|
|||||||
N(OP_DisciplineUpdate),
|
N(OP_DisciplineUpdate),
|
||||||
N(OP_DiscordMerchantInventory),
|
N(OP_DiscordMerchantInventory),
|
||||||
N(OP_DoGroupLeadershipAbility),
|
N(OP_DoGroupLeadershipAbility),
|
||||||
N(OP_DuelDecline),
|
N(OP_DuelResponse),
|
||||||
N(OP_DuelAccept),
|
N(OP_DuelResponse2),
|
||||||
N(OP_DumpName),
|
N(OP_DumpName),
|
||||||
N(OP_Dye),
|
N(OP_Dye),
|
||||||
N(OP_DynamicWall),
|
N(OP_DynamicWall),
|
||||||
@@ -163,7 +159,6 @@ N(OP_EnduranceUpdate),
|
|||||||
N(OP_EnterChat),
|
N(OP_EnterChat),
|
||||||
N(OP_EnterWorld),
|
N(OP_EnterWorld),
|
||||||
N(OP_EnvDamage),
|
N(OP_EnvDamage),
|
||||||
N(OP_EvolveItem),
|
|
||||||
N(OP_ExpansionInfo),
|
N(OP_ExpansionInfo),
|
||||||
N(OP_ExpUpdate),
|
N(OP_ExpUpdate),
|
||||||
N(OP_FaceChange),
|
N(OP_FaceChange),
|
||||||
@@ -231,7 +226,6 @@ N(OP_GuildBank),
|
|||||||
N(OP_GuildBankItemList),
|
N(OP_GuildBankItemList),
|
||||||
N(OP_GuildCreate),
|
N(OP_GuildCreate),
|
||||||
N(OP_GuildDelete),
|
N(OP_GuildDelete),
|
||||||
N(OP_GuildDeleteGuild),
|
|
||||||
N(OP_GuildDemote),
|
N(OP_GuildDemote),
|
||||||
N(OP_GuildInvite),
|
N(OP_GuildInvite),
|
||||||
N(OP_GuildInviteAccept),
|
N(OP_GuildInviteAccept),
|
||||||
@@ -243,33 +237,15 @@ N(OP_GuildManageStatus),
|
|||||||
N(OP_GuildMemberLevelUpdate),
|
N(OP_GuildMemberLevelUpdate),
|
||||||
N(OP_GuildMemberList),
|
N(OP_GuildMemberList),
|
||||||
N(OP_GuildMemberUpdate),
|
N(OP_GuildMemberUpdate),
|
||||||
N(OP_GuildMemberLevel),
|
|
||||||
N(OP_GuildMemberRankAltBanker),
|
|
||||||
N(OP_GuildMemberPublicNote),
|
|
||||||
N(OP_GuildMemberAdd),
|
|
||||||
N(OP_GuildMemberRename),
|
|
||||||
N(OP_GuildMemberDelete),
|
|
||||||
N(OP_GuildMemberDetails),
|
|
||||||
N(OP_GuildRenameGuild),
|
|
||||||
N(OP_GuildMOTD),
|
N(OP_GuildMOTD),
|
||||||
N(OP_GuildPeace),
|
N(OP_GuildPeace),
|
||||||
N(OP_GuildPromote),
|
N(OP_GuildPromote),
|
||||||
N(OP_GuildPublicNote),
|
N(OP_GuildPublicNote),
|
||||||
N(OP_GuildRemove),
|
N(OP_GuildRemove),
|
||||||
N(OP_GuildSelectTribute),
|
|
||||||
N(OP_GuildModifyBenefits),
|
|
||||||
N(OP_GuildTributeToggleReq),
|
|
||||||
N(OP_GuildTributeToggleReply),
|
|
||||||
N(OP_GuildOptInOut),
|
|
||||||
N(OP_GuildSaveActiveTributes),
|
|
||||||
N(OP_GuildSendActiveTributes),
|
|
||||||
N(OP_GuildTributeFavorAndTimer),
|
|
||||||
N(OP_GuildsList),
|
N(OP_GuildsList),
|
||||||
N(OP_GuildStatus),
|
N(OP_GuildStatus),
|
||||||
N(OP_GuildTributeInfo),
|
N(OP_GuildTributeInfo),
|
||||||
N(OP_GuildUpdate),
|
N(OP_GuildUpdateURLAndChannel),
|
||||||
N(OP_GuildTributeDonateItem),
|
|
||||||
N(OP_GuildTributeDonatePlat),
|
|
||||||
N(OP_GuildWar),
|
N(OP_GuildWar),
|
||||||
N(OP_Heartbeat),
|
N(OP_Heartbeat),
|
||||||
N(OP_Hide),
|
N(OP_Hide),
|
||||||
@@ -285,15 +261,12 @@ N(OP_InspectMessageUpdate),
|
|||||||
N(OP_InspectRequest),
|
N(OP_InspectRequest),
|
||||||
N(OP_InstillDoubt),
|
N(OP_InstillDoubt),
|
||||||
N(OP_InterruptCast),
|
N(OP_InterruptCast),
|
||||||
N(OP_InvokeChangePetName),
|
|
||||||
N(OP_InvokeChangePetNameImmediate),
|
|
||||||
N(OP_ItemLinkClick),
|
N(OP_ItemLinkClick),
|
||||||
N(OP_ItemLinkResponse),
|
N(OP_ItemLinkResponse),
|
||||||
N(OP_ItemLinkText),
|
N(OP_ItemLinkText),
|
||||||
N(OP_ItemName),
|
N(OP_ItemName),
|
||||||
N(OP_ItemPacket),
|
N(OP_ItemPacket),
|
||||||
N(OP_ItemPreview),
|
N(OP_ItemPreview),
|
||||||
N(OP_ItemPreviewRequest),
|
|
||||||
N(OP_ItemRecastDelay),
|
N(OP_ItemRecastDelay),
|
||||||
N(OP_ItemVerifyReply),
|
N(OP_ItemVerifyReply),
|
||||||
N(OP_ItemVerifyRequest),
|
N(OP_ItemVerifyRequest),
|
||||||
@@ -330,7 +303,6 @@ N(OP_LockoutTimerInfo),
|
|||||||
N(OP_Login),
|
N(OP_Login),
|
||||||
N(OP_LoginAccepted),
|
N(OP_LoginAccepted),
|
||||||
N(OP_LoginComplete),
|
N(OP_LoginComplete),
|
||||||
N(OP_LoginExpansionPacketData), //added for Rof2 client to send expansion data packet. Requires login_opcodes_sod.conf to be updated.
|
|
||||||
N(OP_LoginUnknown1),
|
N(OP_LoginUnknown1),
|
||||||
N(OP_LoginUnknown2),
|
N(OP_LoginUnknown2),
|
||||||
N(OP_Logout),
|
N(OP_Logout),
|
||||||
@@ -342,7 +314,6 @@ N(OP_LootRequest),
|
|||||||
N(OP_ManaChange),
|
N(OP_ManaChange),
|
||||||
N(OP_ManaUpdate),
|
N(OP_ManaUpdate),
|
||||||
N(OP_MarkNPC),
|
N(OP_MarkNPC),
|
||||||
N(OP_MarkRaidNPC),
|
|
||||||
N(OP_Marquee),
|
N(OP_Marquee),
|
||||||
N(OP_MemorizeSpell),
|
N(OP_MemorizeSpell),
|
||||||
N(OP_Mend),
|
N(OP_Mend),
|
||||||
@@ -384,6 +355,7 @@ N(OP_OpenContainer),
|
|||||||
N(OP_OpenDiscordMerchant),
|
N(OP_OpenDiscordMerchant),
|
||||||
N(OP_OpenGuildTributeMaster),
|
N(OP_OpenGuildTributeMaster),
|
||||||
N(OP_OpenInventory),
|
N(OP_OpenInventory),
|
||||||
|
N(OP_OpenNewTasksWindow),
|
||||||
N(OP_OpenTributeMaster),
|
N(OP_OpenTributeMaster),
|
||||||
N(OP_PDeletePetition),
|
N(OP_PDeletePetition),
|
||||||
N(OP_PetBuffWindow),
|
N(OP_PetBuffWindow),
|
||||||
@@ -405,8 +377,6 @@ N(OP_PetitionSearchText),
|
|||||||
N(OP_PetitionUnCheckout),
|
N(OP_PetitionUnCheckout),
|
||||||
N(OP_PetitionUpdate),
|
N(OP_PetitionUpdate),
|
||||||
N(OP_PickPocket),
|
N(OP_PickPocket),
|
||||||
N(OP_PickZone),
|
|
||||||
N(OP_PickZoneWindow),
|
|
||||||
N(OP_PlayerProfile),
|
N(OP_PlayerProfile),
|
||||||
N(OP_PlayerStateAdd),
|
N(OP_PlayerStateAdd),
|
||||||
N(OP_PlayerStateRemove),
|
N(OP_PlayerStateRemove),
|
||||||
@@ -427,8 +397,6 @@ N(OP_PVPLeaderBoardRequest),
|
|||||||
N(OP_PVPStats),
|
N(OP_PVPStats),
|
||||||
N(OP_QueryResponseThing),
|
N(OP_QueryResponseThing),
|
||||||
N(OP_QueryUCSServerStatus),
|
N(OP_QueryUCSServerStatus),
|
||||||
N(OP_RaidDelegateAbility),
|
|
||||||
N(OP_RaidClearNPCMarks),
|
|
||||||
N(OP_RaidInvite),
|
N(OP_RaidInvite),
|
||||||
N(OP_RaidJoin),
|
N(OP_RaidJoin),
|
||||||
N(OP_RaidUpdate),
|
N(OP_RaidUpdate),
|
||||||
@@ -452,7 +420,6 @@ N(OP_ReqClientSpawn),
|
|||||||
N(OP_ReqNewZone),
|
N(OP_ReqNewZone),
|
||||||
N(OP_RequestClientZoneChange),
|
N(OP_RequestClientZoneChange),
|
||||||
N(OP_RequestDuel),
|
N(OP_RequestDuel),
|
||||||
N(OP_RequestGuildTributes),
|
|
||||||
N(OP_RequestKnowledgeBase),
|
N(OP_RequestKnowledgeBase),
|
||||||
N(OP_RequestTitles),
|
N(OP_RequestTitles),
|
||||||
N(OP_RespawnWindow),
|
N(OP_RespawnWindow),
|
||||||
@@ -489,7 +456,6 @@ N(OP_ServerListResponse),
|
|||||||
N(OP_SessionReady),
|
N(OP_SessionReady),
|
||||||
N(OP_SetChatServer),
|
N(OP_SetChatServer),
|
||||||
N(OP_SetChatServer2),
|
N(OP_SetChatServer2),
|
||||||
N(OP_SetFace),
|
|
||||||
N(OP_SetGroupTarget),
|
N(OP_SetGroupTarget),
|
||||||
N(OP_SetGuildMOTD),
|
N(OP_SetGuildMOTD),
|
||||||
N(OP_SetGuildRank),
|
N(OP_SetGuildRank),
|
||||||
@@ -498,19 +464,6 @@ N(OP_SetServerFilter),
|
|||||||
N(OP_SetStartCity),
|
N(OP_SetStartCity),
|
||||||
N(OP_SetTitle),
|
N(OP_SetTitle),
|
||||||
N(OP_SetTitleReply),
|
N(OP_SetTitleReply),
|
||||||
N(OP_SharedTaskMemberList),
|
|
||||||
N(OP_SharedTaskAddPlayer),
|
|
||||||
N(OP_SharedTaskRemovePlayer),
|
|
||||||
N(OP_SharedTaskMakeLeader),
|
|
||||||
N(OP_SharedTaskMemberInvite),
|
|
||||||
N(OP_SharedTaskInvite),
|
|
||||||
N(OP_SharedTaskInviteResponse),
|
|
||||||
N(OP_SharedTaskAcceptNew),
|
|
||||||
N(OP_SharedTaskMemberChange),
|
|
||||||
N(OP_SharedTaskPlayerList),
|
|
||||||
N(OP_SharedTaskSelectWindow),
|
|
||||||
N(OP_SharedTaskQuit),
|
|
||||||
N(OP_TaskTimers),
|
|
||||||
N(OP_Shielding),
|
N(OP_Shielding),
|
||||||
N(OP_ShopDelItem),
|
N(OP_ShopDelItem),
|
||||||
N(OP_ShopEnd),
|
N(OP_ShopEnd),
|
||||||
@@ -518,11 +471,6 @@ N(OP_ShopEndConfirm),
|
|||||||
N(OP_ShopItem),
|
N(OP_ShopItem),
|
||||||
N(OP_ShopPlayerBuy),
|
N(OP_ShopPlayerBuy),
|
||||||
N(OP_ShopPlayerSell),
|
N(OP_ShopPlayerSell),
|
||||||
N(OP_ShopSendParcel),
|
|
||||||
N(OP_ShopDeleteParcel),
|
|
||||||
N(OP_ShopRespondParcel),
|
|
||||||
N(OP_ShopRetrieveParcel),
|
|
||||||
N(OP_ShopParcelIcon),
|
|
||||||
N(OP_ShopRequest),
|
N(OP_ShopRequest),
|
||||||
N(OP_SimpleMessage),
|
N(OP_SimpleMessage),
|
||||||
N(OP_SkillUpdate),
|
N(OP_SkillUpdate),
|
||||||
@@ -541,7 +489,6 @@ N(OP_Stamina),
|
|||||||
N(OP_Stun),
|
N(OP_Stun),
|
||||||
N(OP_Surname),
|
N(OP_Surname),
|
||||||
N(OP_SwapSpell),
|
N(OP_SwapSpell),
|
||||||
N(OP_SystemFingerprint),
|
|
||||||
N(OP_TargetBuffs),
|
N(OP_TargetBuffs),
|
||||||
N(OP_TargetCommand),
|
N(OP_TargetCommand),
|
||||||
N(OP_TargetHoTT),
|
N(OP_TargetHoTT),
|
||||||
@@ -552,8 +499,7 @@ N(OP_TaskActivityComplete),
|
|||||||
N(OP_TaskDescription),
|
N(OP_TaskDescription),
|
||||||
N(OP_TaskHistoryReply),
|
N(OP_TaskHistoryReply),
|
||||||
N(OP_TaskHistoryRequest),
|
N(OP_TaskHistoryRequest),
|
||||||
N(OP_TaskRequestTimer),
|
N(OP_TaskMemberList),
|
||||||
N(OP_TaskSelectWindow),
|
|
||||||
N(OP_Taunt),
|
N(OP_Taunt),
|
||||||
N(OP_TestBuff),
|
N(OP_TestBuff),
|
||||||
N(OP_TGB),
|
N(OP_TGB),
|
||||||
@@ -566,7 +512,6 @@ N(OP_TradeBusy),
|
|||||||
N(OP_TradeCoins),
|
N(OP_TradeCoins),
|
||||||
N(OP_TradeMoneyUpdate),
|
N(OP_TradeMoneyUpdate),
|
||||||
N(OP_Trader),
|
N(OP_Trader),
|
||||||
N(OP_TraderBulkSend),
|
|
||||||
N(OP_TraderBuy),
|
N(OP_TraderBuy),
|
||||||
N(OP_TraderDelItem),
|
N(OP_TraderDelItem),
|
||||||
N(OP_TradeRequest),
|
N(OP_TradeRequest),
|
||||||
@@ -600,7 +545,6 @@ N(OP_WhoAllRequest),
|
|||||||
N(OP_WhoAllResponse),
|
N(OP_WhoAllResponse),
|
||||||
N(OP_World_Client_CRC1),
|
N(OP_World_Client_CRC1),
|
||||||
N(OP_World_Client_CRC2),
|
N(OP_World_Client_CRC2),
|
||||||
N(OP_World_Client_CRC3),
|
|
||||||
N(OP_WorldClientReady),
|
N(OP_WorldClientReady),
|
||||||
N(OP_WorldComplete),
|
N(OP_WorldComplete),
|
||||||
N(OP_WorldLogout),
|
N(OP_WorldLogout),
|
||||||
@@ -623,5 +567,4 @@ N(OP_ZoneServerReady),
|
|||||||
N(OP_ZoneSpawns),
|
N(OP_ZoneSpawns),
|
||||||
N(OP_ZoneUnavail),
|
N(OP_ZoneUnavail),
|
||||||
N(OP_ResetAA),
|
N(OP_ResetAA),
|
||||||
N(OP_UnderWorld),
|
|
||||||
// mail and chat opcodes located in ../mail_oplist.h
|
// mail and chat opcodes located in ../mail_oplist.h
|
||||||
|
|||||||
+131
-756
@@ -23,61 +23,60 @@
|
|||||||
#include "skills.h"
|
#include "skills.h"
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
|
|
||||||
namespace AppearanceType {
|
|
||||||
constexpr uint32 Die = 0; // Causes the client to keel over and zone to bind point (default action)
|
|
||||||
constexpr uint32 WhoLevel = 1; // Level that shows up on /who
|
|
||||||
constexpr uint32 MaxHealth = 2;
|
|
||||||
constexpr uint32 Invisibility = 3; // 0 = Visible, 1 = Invisible
|
|
||||||
constexpr uint32 PVP = 4; // 0 = Non-PVP, 1 = PVP
|
|
||||||
constexpr uint32 Light = 5; // Light type emitted by player (lightstone, shiny shield)
|
|
||||||
constexpr uint32 Animation = 14; // 100 = Standing, 102 = Freeze, 105 = Looting, 110 = Sitting, 111 = Crouching, 115 = Lying
|
|
||||||
constexpr uint32 Sneak = 15; // 0 = Normal, 1 = Sneaking
|
|
||||||
constexpr uint32 SpawnID = 16; // Server -> Client, sets player spawn ID
|
|
||||||
constexpr uint32 Health = 17; // Client->Server, my HP has changed (like regen tic)
|
|
||||||
constexpr uint32 Linkdead = 18; // 0 = Normal, 1 = Linkdead
|
|
||||||
constexpr uint32 FlyMode = 19; // 0 = Off, 1 = Flying, 2 = Levitating, 3 = Water, 4 = Floating, 5 = Levitating while Running
|
|
||||||
constexpr uint32 GM = 20; // 0 = Non-GM, 1 = GM
|
|
||||||
constexpr uint32 Anonymous = 21; // 0 = Non-Anonymous, 1 = Anonymous, 2 = Roleplaying
|
|
||||||
constexpr uint32 GuildID = 22;
|
|
||||||
constexpr uint32 GuildRank = 23;
|
|
||||||
constexpr uint32 AFK = 24; // 0 = Non-AFK, 1 = AFK
|
|
||||||
constexpr uint32 Pet = 25; // Parameter is Entity ID of owner, or 0 for when charm breaks
|
|
||||||
constexpr uint32 Summoned = 27;
|
|
||||||
constexpr uint32 Split = 28; // 0 = No Split, 1 = Auto Split
|
|
||||||
constexpr uint32 Size = 29; // Spawn's Size
|
|
||||||
constexpr uint32 SetType = 30; // 0 = PC, 1 = NPC, 2 = Corpse
|
|
||||||
constexpr uint32 NPCName = 31; // Change PC name color to NPC name color
|
|
||||||
constexpr uint32 AARank = 32; // AA Rank Title ID, title in /who?
|
|
||||||
constexpr uint32 CancelSneakHide = 33; // Turns off Hide and Sneak
|
|
||||||
constexpr uint32 AreaHealthRegen = 35; // Guild Hall Regeneration Pool sets to value * 0.001
|
|
||||||
constexpr uint32 AreaManaRegen = 36; // Guild Hall Regeneration Pool sets to value * 0.001
|
|
||||||
constexpr uint32 AreaEnduranceRegen = 37; // Guild Hall Regeneration Pool sets to value * 0.001
|
|
||||||
constexpr uint32 FreezeBeneficialBuffs = 38; // Freezes beneficial buff timers for PCs
|
|
||||||
constexpr uint32 NPCTintIndex = 39;
|
|
||||||
constexpr uint32 GroupAutoConsent = 40; // Auto Consent Group
|
|
||||||
constexpr uint32 RaidAutoConsent = 41; // Auto Consent Raid
|
|
||||||
constexpr uint32 GuildAutoConsent = 42; // Auto Consent Guild
|
|
||||||
constexpr uint32 ShowHelm = 43; // 0 = Hide, 1 = Show
|
|
||||||
constexpr uint32 DamageState = 44; // The damage state of a destructible object (0 through 10) plays sound IDs, most only have 2 or 4 states though
|
|
||||||
constexpr uint32 EQPlayers = 45; // EQ Players Update
|
|
||||||
constexpr uint32 FindBits = 46; // Set Find Bits?
|
|
||||||
constexpr uint32 TextureType = 48; // Texture Type?
|
|
||||||
constexpr uint32 FacePick = 49; // Turns off face pick window?
|
|
||||||
constexpr uint32 AntiCheat = 51; // Sent by the client randomly telling the server how long since last action has occurred
|
|
||||||
constexpr uint32 GuildShow = 52;
|
|
||||||
constexpr uint32 OfflineMode = 53; // Offline Mode
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Animation {
|
//SpawnAppearance types: (compared two clients for server-originating types: SoF & RoF2)
|
||||||
constexpr uint32 Standing = 100;
|
#define AT_Die 0 // this causes the client to keel over and zone to bind point (default action)
|
||||||
constexpr uint32 Freeze = 102;
|
#define AT_WhoLevel 1 // the level that shows up on /who
|
||||||
constexpr uint32 Looting = 105;
|
#define AT_HPMax 2 // idk
|
||||||
constexpr uint32 Sitting = 110;
|
#define AT_Invis 3 // 0 = visible, 1 = invisible
|
||||||
constexpr uint32 Crouching = 111;
|
#define AT_PVP 4 // 0 = blue, 1 = pvp (red)
|
||||||
constexpr uint32 Lying = 115;
|
#define AT_Light 5 // light type emitted by player (lightstone, shiny shield)
|
||||||
}
|
#define AT_Anim 14 // 100=standing, 110=sitting, 111=ducking, 115=feigned, 105=looting
|
||||||
|
#define AT_Sneak 15 // 0 = normal, 1 = sneaking
|
||||||
|
#define AT_SpawnID 16 // server to client, sets player spawn id
|
||||||
|
#define AT_HP 17 // Client->Server, my HP has changed (like regen tic)
|
||||||
|
#define AT_Linkdead 18 // 0 = normal, 1 = linkdead
|
||||||
|
#define AT_Levitate 19 // 0=off, 1=flymode, 2=levitate max 5, see GravityBehavior enum
|
||||||
|
#define AT_GM 20 // 0 = normal, 1 = GM - all odd numbers seem to make it GM
|
||||||
|
#define AT_Anon 21 // 0 = normal, 1 = anon, 2 = roleplay
|
||||||
|
#define AT_GuildID 22
|
||||||
|
#define AT_GuildRank 23 // 0=member, 1=officer, 2=leader
|
||||||
|
#define AT_AFK 24 // 0 = normal, 1 = afk
|
||||||
|
#define AT_Pet 25 // Param is EntityID of owner, or 0 for when charm breaks
|
||||||
|
#define AT_Summoned 27 // Unsure
|
||||||
|
#define AT_Split 28 // 0 = normal, 1 = autosplit on (not showing in SoF+) (client-to-server only)
|
||||||
|
#define AT_Size 29 // spawn's size (present: SoF, absent: RoF2)
|
||||||
|
#define AT_SetType 30 // 0 = PC, 1 = NPC, 2 <= = corpse
|
||||||
|
#define AT_NPCName 31 // change PC's name's color to NPC color 0 = normal, 1 = npc name, Trader on RoF2?
|
||||||
|
#define AT_AARank 32 // AA Rank Title ID thingy, does is this the title in /who?
|
||||||
|
#define AT_CancelSneakHide 33 // Turns off Hide and Sneak
|
||||||
|
//#define AT_34 34 // unknown (present: SoF, absent: RoF2)
|
||||||
|
#define AT_AreaHPRegen 35 // guild hall regen pool sets to value * 0.001
|
||||||
|
#define AT_AreaManaRegen 36 // guild hall regen pool sets to value * 0.001
|
||||||
|
#define AT_AreaEndRegen 37 // guild hall regen pool sets to value * 0.001
|
||||||
|
#define AT_FreezeBuffs 38 // Freezes beneficial buff timers
|
||||||
|
#define AT_NpcTintIndex 39 // not 100% sure
|
||||||
|
#define AT_GroupConsent 40 // auto consent group
|
||||||
|
#define AT_RaidConsent 41 // auto consent raid
|
||||||
|
#define AT_GuildConsent 42 // auto consent guild
|
||||||
|
#define AT_ShowHelm 43 // 0 = hide graphic, 1 = show graphic
|
||||||
|
#define AT_DamageState 44 // The damage state of a destructible object (0 through 10) plays soundids most only have 2 or 4 states though
|
||||||
|
#define AT_EQPlayers 45 // /eqplayersupdate
|
||||||
|
#define AT_FindBits 46 // set FindBits, whatever those are!
|
||||||
|
#define AT_TextureType 48 // TextureType
|
||||||
|
#define AT_FacePick 49 // Turns off face pick window? maybe ...
|
||||||
|
#define AT_GuildShow 52 // this is what MQ2 call sit, not sure
|
||||||
|
#define AT_Offline 53 // Offline mode
|
||||||
|
|
||||||
constexpr int16 RECAST_TYPE_UNLINKED_ITEM = -1;
|
//#define AT_Trader 300 // Bazaar Trader Mode (not present in SoF or RoF2)
|
||||||
|
|
||||||
|
// animations for AT_Anim
|
||||||
|
#define ANIM_FREEZE 102
|
||||||
|
#define ANIM_STAND 0x64
|
||||||
|
#define ANIM_SIT 0x6e
|
||||||
|
#define ANIM_CROUCH 0x6f
|
||||||
|
#define ANIM_DEATH 0x73
|
||||||
|
#define ANIM_LOOT 0x69
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
eaStanding = 0,
|
eaStanding = 0,
|
||||||
@@ -198,539 +197,14 @@ namespace Chat {
|
|||||||
const uint16 Stun = 340;
|
const uint16 Stun = 340;
|
||||||
};
|
};
|
||||||
|
|
||||||
// generation SQL:
|
//ZoneChange_Struct->success values
|
||||||
// SELECT CONCAT(' constexpr uint16 ', UPPER(short_name), ' = ' , zoneidnumber, '; // ', long_name) from zone group by zoneidnumber ORDER BY zoneidnumber;
|
#define ZONE_ERROR_NOMSG 0
|
||||||
namespace Zones {
|
#define ZONE_ERROR_NOTREADY -1
|
||||||
constexpr uint16 QEYNOS = 1; // South Qeynos
|
#define ZONE_ERROR_VALIDPC -2
|
||||||
constexpr uint16 QEYNOS2 = 2; // North Qeynos
|
#define ZONE_ERROR_STORYZONE -3
|
||||||
constexpr uint16 QRG = 3; // The Surefall Glade
|
#define ZONE_ERROR_NOEXPANSION -6
|
||||||
constexpr uint16 QEYTOQRG = 4; // The Qeynos Hills
|
#define ZONE_ERROR_NOEXPERIENCE -7
|
||||||
constexpr uint16 HIGHPASS = 5; // Highpass Hold
|
|
||||||
constexpr uint16 HIGHKEEP = 6; // High Keep
|
|
||||||
constexpr uint16 FREPORTN = 8; // North Freeport
|
|
||||||
constexpr uint16 FREPORTW = 9; // West Freeport
|
|
||||||
constexpr uint16 FREPORTE = 10; // East Freeport
|
|
||||||
constexpr uint16 RUNNYEYE = 11; // The Liberated Citadel of Runnyeye
|
|
||||||
constexpr uint16 QEY2HH1 = 12; // The Western Plains of Karana
|
|
||||||
constexpr uint16 NORTHKARANA = 13; // The Northern Plains of Karana
|
|
||||||
constexpr uint16 SOUTHKARANA = 14; // The Southern Plains of Karana
|
|
||||||
constexpr uint16 EASTKARANA = 15; // Eastern Plains of Karana
|
|
||||||
constexpr uint16 BEHOLDER = 16; // Gorge of King Xorbb
|
|
||||||
constexpr uint16 BLACKBURROW = 17; // Blackburrow
|
|
||||||
constexpr uint16 PAW = 18; // The Lair of the Splitpaw
|
|
||||||
constexpr uint16 RIVERVALE = 19; // Rivervale
|
|
||||||
constexpr uint16 KITHICOR = 20; // Kithicor Forest
|
|
||||||
constexpr uint16 COMMONS = 21; // West Commonlands
|
|
||||||
constexpr uint16 ECOMMONS = 22; // East Commonlands
|
|
||||||
constexpr uint16 ERUDNINT = 23; // The Erudin Palace
|
|
||||||
constexpr uint16 ERUDNEXT = 24; // Erudin
|
|
||||||
constexpr uint16 NEKTULOS = 25; // The Nektulos Forest
|
|
||||||
constexpr uint16 CSHOME = 26; // Sunset Home
|
|
||||||
constexpr uint16 LAVASTORM = 27; // The Lavastorm Mountains
|
|
||||||
constexpr uint16 NEKTROPOS = 28; // Nektropos
|
|
||||||
constexpr uint16 HALAS = 29; // Halas
|
|
||||||
constexpr uint16 EVERFROST = 30; // Everfrost Peaks
|
|
||||||
constexpr uint16 SOLDUNGA = 31; // Solusek's Eye
|
|
||||||
constexpr uint16 SOLDUNGB = 32; // Nagafen's Lair
|
|
||||||
constexpr uint16 MISTY = 33; // Misty Thicket
|
|
||||||
constexpr uint16 NRO = 34; // Northern Desert of Ro
|
|
||||||
constexpr uint16 SRO = 35; // Southern Desert of Ro
|
|
||||||
constexpr uint16 BEFALLEN = 36; // Befallen
|
|
||||||
constexpr uint16 OASIS = 37; // Oasis of Marr
|
|
||||||
constexpr uint16 TOX = 38; // Toxxulia Forest
|
|
||||||
constexpr uint16 HOLE = 39; // The Hole
|
|
||||||
constexpr uint16 NERIAKA = 40; // Neriak - Foreign Quarter
|
|
||||||
constexpr uint16 NERIAKB = 41; // Neriak - Commons
|
|
||||||
constexpr uint16 NERIAKC = 42; // Neriak - 3rd Gate
|
|
||||||
constexpr uint16 NERIAKD = 43; // Neriak Palace
|
|
||||||
constexpr uint16 NAJENA = 44; // Najena
|
|
||||||
constexpr uint16 QCAT = 45; // The Qeynos Aqueduct System
|
|
||||||
constexpr uint16 INNOTHULE = 46; // Innothule Swamp
|
|
||||||
constexpr uint16 FEERROTT = 47; // The Feerrott
|
|
||||||
constexpr uint16 CAZICTHULE = 48; // Accursed Temple of CazicThule
|
|
||||||
constexpr uint16 OGGOK = 49; // Oggok
|
|
||||||
constexpr uint16 RATHEMTN = 50; // The Rathe Mountains
|
|
||||||
constexpr uint16 LAKERATHE = 51; // Lake Rathetear
|
|
||||||
constexpr uint16 GROBB = 52; // Grobb
|
|
||||||
constexpr uint16 AVIAK = 53; // Aviak Village
|
|
||||||
constexpr uint16 GFAYDARK = 54; // The Greater Faydark
|
|
||||||
constexpr uint16 AKANON = 55; // Ak'Anon
|
|
||||||
constexpr uint16 STEAMFONT = 56; // Steamfont Mountains
|
|
||||||
constexpr uint16 LFAYDARK = 57; // The Lesser Faydark
|
|
||||||
constexpr uint16 CRUSHBONE = 58; // Crushbone
|
|
||||||
constexpr uint16 MISTMOORE = 59; // The Castle of Mistmoore
|
|
||||||
constexpr uint16 KALADIMA = 60; // South Kaladim
|
|
||||||
constexpr uint16 FELWITHEA = 61; // Northern Felwithe
|
|
||||||
constexpr uint16 FELWITHEB = 62; // Southern Felwithe
|
|
||||||
constexpr uint16 UNREST = 63; // The Estate of Unrest
|
|
||||||
constexpr uint16 KEDGE = 64; // Kedge Keep
|
|
||||||
constexpr uint16 GUKTOP = 65; // The City of Guk
|
|
||||||
constexpr uint16 GUKBOTTOM = 66; // The Ruins of Old Guk
|
|
||||||
constexpr uint16 KALADIMB = 67; // North Kaladim
|
|
||||||
constexpr uint16 BUTCHER = 68; // Butcherblock Mountains
|
|
||||||
constexpr uint16 OOT = 69; // Ocean of Tears
|
|
||||||
constexpr uint16 CAULDRON = 70; // Dagnor's Cauldron
|
|
||||||
constexpr uint16 AIRPLANE = 71; // The Plane of Sky
|
|
||||||
constexpr uint16 FEARPLANE = 72; // The Plane of Fear
|
|
||||||
constexpr uint16 PERMAFROST = 73; // The Permafrost Caverns
|
|
||||||
constexpr uint16 KERRARIDGE = 74; // Kerra Isle
|
|
||||||
constexpr uint16 PAINEEL = 75; // Paineel
|
|
||||||
constexpr uint16 HATEPLANE = 76; // Plane of Hate
|
|
||||||
constexpr uint16 ARENA = 77; // The Arena
|
|
||||||
constexpr uint16 FIELDOFBONE = 78; // The Field of Bone
|
|
||||||
constexpr uint16 WARSLIKSWOOD = 79; // The Warsliks Woods
|
|
||||||
constexpr uint16 SOLTEMPLE = 80; // The Temple of Solusek Ro
|
|
||||||
constexpr uint16 DROGA = 81; // The Temple of Droga
|
|
||||||
constexpr uint16 CABWEST = 82; // Cabilis West
|
|
||||||
constexpr uint16 SWAMPOFNOHOPE = 83; // The Swamp of No Hope
|
|
||||||
constexpr uint16 FIRIONA = 84; // Firiona Vie
|
|
||||||
constexpr uint16 LAKEOFILLOMEN = 85; // Lake of Ill Omen
|
|
||||||
constexpr uint16 DREADLANDS = 86; // The Dreadlands
|
|
||||||
constexpr uint16 BURNINGWOOD = 87; // The Burning Wood
|
|
||||||
constexpr uint16 KAESORA = 88; // Kaesora
|
|
||||||
constexpr uint16 SEBILIS = 89; // The Ruins of Sebilis
|
|
||||||
constexpr uint16 CITYMIST = 90; // The City of Mist
|
|
||||||
constexpr uint16 SKYFIRE = 91; // The Skyfire Mountains
|
|
||||||
constexpr uint16 FRONTIERMTNS = 92; // Frontier Mountains
|
|
||||||
constexpr uint16 OVERTHERE = 93; // The Overthere
|
|
||||||
constexpr uint16 EMERALDJUNGLE = 94; // The Emerald Jungle
|
|
||||||
constexpr uint16 TRAKANON = 95; // Trakanon's Teeth
|
|
||||||
constexpr uint16 TIMOROUS = 96; // Timorous Deep
|
|
||||||
constexpr uint16 KURN = 97; // Kurn's Tower
|
|
||||||
constexpr uint16 ERUDSXING = 98; // Erud's Crossing
|
|
||||||
constexpr uint16 STONEBRUNT = 100; // The Stonebrunt Mountains
|
|
||||||
constexpr uint16 WARRENS = 101; // The Warrens
|
|
||||||
constexpr uint16 KARNOR = 102; // Karnor's Castle
|
|
||||||
constexpr uint16 CHARDOK = 103; // Chardok
|
|
||||||
constexpr uint16 DALNIR = 104; // The Crypt of Dalnir
|
|
||||||
constexpr uint16 CHARASIS = 105; // The Howling Stones
|
|
||||||
constexpr uint16 CABEAST = 106; // Cabilis East
|
|
||||||
constexpr uint16 NURGA = 107; // The Mines of Nurga
|
|
||||||
constexpr uint16 VEESHAN = 108; // Veeshan's Peak
|
|
||||||
constexpr uint16 VEKSAR = 109; // Veksar
|
|
||||||
constexpr uint16 ICECLAD = 110; // The Iceclad Ocean
|
|
||||||
constexpr uint16 FROZENSHADOW = 111; // The Tower of Frozen Shadow
|
|
||||||
constexpr uint16 VELKETOR = 112; // Velketor's Labyrinth
|
|
||||||
constexpr uint16 KAEL = 113; // Kael Drakkel
|
|
||||||
constexpr uint16 SKYSHRINE = 114; // Skyshrine
|
|
||||||
constexpr uint16 THURGADINA = 115; // The City of Thurgadin
|
|
||||||
constexpr uint16 EASTWASTES = 116; // Eastern Wastes
|
|
||||||
constexpr uint16 COBALTSCAR = 117; // Cobaltscar
|
|
||||||
constexpr uint16 GREATDIVIDE = 118; // The Great Divide
|
|
||||||
constexpr uint16 WAKENING = 119; // The Wakening Land
|
|
||||||
constexpr uint16 WESTWASTES = 120; // The Western Wastes
|
|
||||||
constexpr uint16 CRYSTAL = 121; // The Crystal Caverns
|
|
||||||
constexpr uint16 NECROPOLIS = 123; // Dragon Necropolis
|
|
||||||
constexpr uint16 TEMPLEVEESHAN = 124; // The Temple of Veeshan
|
|
||||||
constexpr uint16 SIRENS = 125; // Siren's Grotto
|
|
||||||
constexpr uint16 MISCHIEFPLANE = 126; // The Plane of Mischief
|
|
||||||
constexpr uint16 GROWTHPLANE = 127; // The Plane of Growth
|
|
||||||
constexpr uint16 SLEEPER = 128; // The Sleeper's Tomb
|
|
||||||
constexpr uint16 THURGADINB = 129; // Icewell Keep
|
|
||||||
constexpr uint16 ERUDSXING2 = 130; // Marauders Mire
|
|
||||||
constexpr uint16 SHADOWHAVEN = 150; // Shadow Haven
|
|
||||||
constexpr uint16 BAZAAR = 151; // The Bazaar
|
|
||||||
constexpr uint16 NEXUS = 152; // Nexus
|
|
||||||
constexpr uint16 ECHO_ = 153; // The Echo Caverns
|
|
||||||
constexpr uint16 ACRYLIA = 154; // The Acrylia Caverns
|
|
||||||
constexpr uint16 SHARVAHL = 155; // The City of Shar Vahl
|
|
||||||
constexpr uint16 PALUDAL = 156; // The Paludal Caverns
|
|
||||||
constexpr uint16 FUNGUSGROVE = 157; // The Fungus Grove
|
|
||||||
constexpr uint16 VEXTHAL = 158; // Vex Thal
|
|
||||||
constexpr uint16 SSERU = 159; // Sanctus Seru
|
|
||||||
constexpr uint16 KATTA = 160; // Katta Castellum
|
|
||||||
constexpr uint16 NETHERBIAN = 161; // Netherbian Lair
|
|
||||||
constexpr uint16 SSRATEMPLE = 162; // Ssraeshza Temple
|
|
||||||
constexpr uint16 GRIEGSEND = 163; // Grieg's End
|
|
||||||
constexpr uint16 THEDEEP = 164; // The Deep
|
|
||||||
constexpr uint16 SHADEWEAVER = 165; // Shadeweaver's Thicket
|
|
||||||
constexpr uint16 HOLLOWSHADE = 166; // Hollowshade Moor
|
|
||||||
constexpr uint16 GRIMLING = 167; // Grimling Forest
|
|
||||||
constexpr uint16 MSERU = 168; // Marus Seru
|
|
||||||
constexpr uint16 LETALIS = 169; // Mons Letalis
|
|
||||||
constexpr uint16 TWILIGHT = 170; // The Twilight Sea
|
|
||||||
constexpr uint16 THEGREY = 171; // The Grey
|
|
||||||
constexpr uint16 TENEBROUS = 172; // The Tenebrous Mountains
|
|
||||||
constexpr uint16 MAIDEN = 173; // The Maiden's Eye
|
|
||||||
constexpr uint16 DAWNSHROUD = 174; // The Dawnshroud Peaks
|
|
||||||
constexpr uint16 SCARLET = 175; // The Scarlet Desert
|
|
||||||
constexpr uint16 UMBRAL = 176; // The Umbral Plains
|
|
||||||
constexpr uint16 AKHEVA = 179; // The Akheva Ruins
|
|
||||||
constexpr uint16 ARENA2 = 180; // The Arena Two
|
|
||||||
constexpr uint16 JAGGEDPINE = 181; // The Jaggedpine Forest
|
|
||||||
constexpr uint16 NEDARIA = 182; // Nedaria's Landing
|
|
||||||
constexpr uint16 TUTORIAL = 183; // EverQuest Tutorial
|
|
||||||
constexpr uint16 LOAD = 184; // Loading Zone
|
|
||||||
constexpr uint16 LOAD2 = 185; // New Loading Zone
|
|
||||||
constexpr uint16 HATEPLANEB = 186; // The Plane of Hate
|
|
||||||
constexpr uint16 SHADOWREST = 187; // Shadowrest
|
|
||||||
constexpr uint16 TUTORIALA = 188; // The Mines of Gloomingdeep
|
|
||||||
constexpr uint16 TUTORIALB = 189; // The Mines of Gloomingdeep
|
|
||||||
constexpr uint16 CLZ = 190; // Loading
|
|
||||||
constexpr uint16 CODECAY = 200; // The Crypt of Decay
|
|
||||||
constexpr uint16 POJUSTICE = 201; // The Plane of Justice
|
|
||||||
constexpr uint16 POKNOWLEDGE = 202; // The Plane of Knowledge
|
|
||||||
constexpr uint16 POTRANQUILITY = 203; // The Plane of Tranquility
|
|
||||||
constexpr uint16 PONIGHTMARE = 204; // The Plane of Nightmares
|
|
||||||
constexpr uint16 PODISEASE = 205; // The Plane of Disease
|
|
||||||
constexpr uint16 POINNOVATION = 206; // The Plane of Innovation
|
|
||||||
constexpr uint16 POTORMENT = 207; // Torment, the Plane of Pain
|
|
||||||
constexpr uint16 POVALOR = 208; // The Plane of Valor
|
|
||||||
constexpr uint16 BOTHUNDER = 209; // Bastion of Thunder
|
|
||||||
constexpr uint16 POSTORMS = 210; // The Plane of Storms
|
|
||||||
constexpr uint16 HOHONORA = 211; // The Halls of Honor
|
|
||||||
constexpr uint16 SOLROTOWER = 212; // The Tower of Solusek Ro
|
|
||||||
constexpr uint16 POWAR = 213; // Plane of War
|
|
||||||
constexpr uint16 POTACTICS = 214; // Drunder, the Fortress of Zek
|
|
||||||
constexpr uint16 POAIR = 215; // The Plane of Air
|
|
||||||
constexpr uint16 POWATER = 216; // The Plane of Water
|
|
||||||
constexpr uint16 POFIRE = 217; // The Plane of Fire
|
|
||||||
constexpr uint16 POEARTHA = 218; // The Plane of Earth
|
|
||||||
constexpr uint16 POTIMEA = 219; // The Plane of Time
|
|
||||||
constexpr uint16 HOHONORB = 220; // The Temple of Marr
|
|
||||||
constexpr uint16 NIGHTMAREB = 221; // The Lair of Terris Thule
|
|
||||||
constexpr uint16 POEARTHB = 222; // The Plane of Earth
|
|
||||||
constexpr uint16 POTIMEB = 223; // The Plane of Time
|
|
||||||
constexpr uint16 GUNTHAK = 224; // The Gulf of Gunthak
|
|
||||||
constexpr uint16 DULAK = 225; // Dulak's Harbor
|
|
||||||
constexpr uint16 TORGIRAN = 226; // The Torgiran Mines
|
|
||||||
constexpr uint16 NADOX = 227; // The Crypt of Nadox
|
|
||||||
constexpr uint16 HATESFURY = 228; // Hate's Fury
|
|
||||||
constexpr uint16 GUKA = 229; // Deepest Guk: Cauldron of Lost Souls
|
|
||||||
constexpr uint16 RUJA = 230; // The Rujarkian Hills: Bloodied Quarries
|
|
||||||
constexpr uint16 TAKA = 231; // Takish-Hiz: Sunken Library
|
|
||||||
constexpr uint16 MIRA = 232; // Miragul's Menagerie: Silent Gallery
|
|
||||||
constexpr uint16 MMCA = 233; // Mistmoore's Catacombs: Forlorn Caverns
|
|
||||||
constexpr uint16 GUKB = 234; // The Drowning Crypt
|
|
||||||
constexpr uint16 RUJB = 235; // The Rujarkian Hills: Halls of War
|
|
||||||
constexpr uint16 TAKB = 236; // Takish-Hiz: Shifting Tower
|
|
||||||
constexpr uint16 MIRB = 237; // Miragul's Menagerie: Frozen Nightmare
|
|
||||||
constexpr uint16 MMCB = 238; // Mistmoore's Catacombs: Dreary Grotto
|
|
||||||
constexpr uint16 GUKC = 239; // Deepest Guk: Ancient Aqueducts
|
|
||||||
constexpr uint16 RUJC = 240; // The Rujarkian Hills: Wind Bridges
|
|
||||||
constexpr uint16 TAKC = 241; // Takish-Hiz: Within the Compact
|
|
||||||
constexpr uint16 MIRC = 242; // The Spider Den
|
|
||||||
constexpr uint16 MMCC = 243; // Mistmoore's Catacombs: Struggles within the Progeny
|
|
||||||
constexpr uint16 GUKD = 244; // The Mushroom Grove
|
|
||||||
constexpr uint16 RUJD = 245; // The Rujarkian Hills: Prison Break
|
|
||||||
constexpr uint16 TAKD = 246; // Takish-Hiz: Royal Observatory
|
|
||||||
constexpr uint16 MIRD = 247; // Miragul's Menagerie: Hushed Banquet
|
|
||||||
constexpr uint16 MMCD = 248; // Mistmoore's Catacombs: Chambers of Eternal Affliction
|
|
||||||
constexpr uint16 GUKE = 249; // Deepest Guk: The Curse Reborn
|
|
||||||
constexpr uint16 RUJE = 250; // The Rujarkian Hills: Drudge Hollows
|
|
||||||
constexpr uint16 TAKE = 251; // Takish-Hiz: River of Recollection
|
|
||||||
constexpr uint16 MIRE = 252; // The Frosted Halls
|
|
||||||
constexpr uint16 MMCE = 253; // Mistmoore's Catacombs: Sepulcher of the Damned
|
|
||||||
constexpr uint16 GUKF = 254; // Deepest Guk: Chapel of the Witnesses
|
|
||||||
constexpr uint16 RUJF = 255; // The Rujarkian Hills: Fortified Lair of the Taskmasters
|
|
||||||
constexpr uint16 TAKF = 256; // Takish-Hiz: Sandfall Corridors
|
|
||||||
constexpr uint16 MIRF = 257; // The Forgotten Wastes
|
|
||||||
constexpr uint16 MMCF = 258; // Mistmoore's Catacombs: Scion Lair of Fury
|
|
||||||
constexpr uint16 GUKG = 259; // The Root Garden
|
|
||||||
constexpr uint16 RUJG = 260; // The Rujarkian Hills: Hidden Vale of Deceit
|
|
||||||
constexpr uint16 TAKG = 261; // Takish-Hiz: Balancing Chamber
|
|
||||||
constexpr uint16 MIRG = 262; // Miragul's Menagerie: Heart of the Menagerie
|
|
||||||
constexpr uint16 MMCG = 263; // Mistmoore's Catacombs: Cesspits of Putrescence
|
|
||||||
constexpr uint16 GUKH = 264; // Deepest Guk: Accursed Sanctuary
|
|
||||||
constexpr uint16 RUJH = 265; // The Rujarkian Hills: Blazing Forge
|
|
||||||
constexpr uint16 TAKH = 266; // Takish-Hiz: Sweeping Tides
|
|
||||||
constexpr uint16 MIRH = 267; // The Morbid Laboratory
|
|
||||||
constexpr uint16 MMCH = 268; // Mistmoore's Catacombs: Aisles of Blood
|
|
||||||
constexpr uint16 RUJI = 269; // The Rujarkian Hills: Arena of Chance
|
|
||||||
constexpr uint16 TAKI = 270; // Takish-Hiz: Antiquated Palace
|
|
||||||
constexpr uint16 MIRI = 271; // The Theater of Imprisoned Horror
|
|
||||||
constexpr uint16 MMCI = 272; // Mistmoore's Catacombs: Halls of Sanguinary Rites
|
|
||||||
constexpr uint16 RUJJ = 273; // The Rujarkian Hills: Barracks of War
|
|
||||||
constexpr uint16 TAKJ = 274; // Takish-Hiz: Prismatic Corridors
|
|
||||||
constexpr uint16 MIRJ = 275; // Miragul's Menagerie: Grand Library
|
|
||||||
constexpr uint16 MMCJ = 276; // Mistmoore's Catacombs: Infernal Sanctuary
|
|
||||||
constexpr uint16 CHARDOKB = 277; // Chardok: The Halls of Betrayal
|
|
||||||
constexpr uint16 SOLDUNGC = 278; // The Caverns of Exile
|
|
||||||
constexpr uint16 ABYSMAL = 279; // The Abysmal Sea
|
|
||||||
constexpr uint16 NATIMBI = 280; // Natimbi, the Broken Shores
|
|
||||||
constexpr uint16 QINIMI = 281; // Qinimi, Court of Nihilia
|
|
||||||
constexpr uint16 RIWWI = 282; // Riwwi, Coliseum of Games
|
|
||||||
constexpr uint16 BARINDU = 283; // Barindu, Hanging Gardens
|
|
||||||
constexpr uint16 FERUBI = 284; // Ferubi, Forgotten Temple of Taelosia
|
|
||||||
constexpr uint16 SNPOOL = 285; // Sewers of Nihilia, Pool of Sludg
|
|
||||||
constexpr uint16 SNLAIR = 286; // Sewers of Nihilia, Lair of Trapp
|
|
||||||
constexpr uint16 SNPLANT = 287; // Sewers of Nihilia, Purifying Pla
|
|
||||||
constexpr uint16 SNCREMATORY = 288; // Sewers of Nihilia, Emanating Cre
|
|
||||||
constexpr uint16 TIPT = 289; // Tipt, Treacherous Crags
|
|
||||||
constexpr uint16 VXED = 290; // Vxed, the Crumbling Caverns
|
|
||||||
constexpr uint16 YXTTA = 291; // Yxtta, Pulpit of Exiles
|
|
||||||
constexpr uint16 UQUA = 292; // Uqua, the Ocean God Chantry
|
|
||||||
constexpr uint16 KODTAZ = 293; // Kod'Taz, Broken Trial Grounds
|
|
||||||
constexpr uint16 IKKINZ = 294; // Ikkinz, Chambers of Transcendence
|
|
||||||
constexpr uint16 QVIC = 295; // Qvic, Prayer Grounds of Calling
|
|
||||||
constexpr uint16 INKTUTA = 296; // Inktu'Ta, the Unmasked Chapel
|
|
||||||
constexpr uint16 TXEVU = 297; // Txevu, Lair of the Elite
|
|
||||||
constexpr uint16 TACVI = 298; // Tacvi, The Broken Temple
|
|
||||||
constexpr uint16 QVICB = 299; // Qvic, the Hidden Vault
|
|
||||||
constexpr uint16 WALLOFSLAUGHTER = 300; // Wall of Slaughter
|
|
||||||
constexpr uint16 BLOODFIELDS = 301; // The Bloodfields
|
|
||||||
constexpr uint16 DRANIKSSCAR = 302; // Dranik's Scar
|
|
||||||
constexpr uint16 CAUSEWAY = 303; // Nobles' Causeway
|
|
||||||
constexpr uint16 CHAMBERSA = 304; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 CHAMBERSB = 305; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 CHAMBERSC = 306; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 CHAMBERSD = 307; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 CHAMBERSE = 308; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 CHAMBERSF = 309; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 PROVINGGROUNDS = 316; // Muramite Proving Grounds
|
|
||||||
constexpr uint16 ANGUISH = 317; // Anguish, the Fallen Palace
|
|
||||||
constexpr uint16 DRANIKHOLLOWSA = 318; // Dranik's Hollows
|
|
||||||
constexpr uint16 DRANIKHOLLOWSB = 319; // Dranik's Hollows
|
|
||||||
constexpr uint16 DRANIKHOLLOWSC = 320; // Dranik's Hollows
|
|
||||||
constexpr uint16 DRANIKCATACOMBSA = 328; // Catacombs of Dranik
|
|
||||||
constexpr uint16 DRANIKCATACOMBSB = 329; // Catacombs of Dranik
|
|
||||||
constexpr uint16 DRANIKCATACOMBSC = 330; // Catacombs of Dranik
|
|
||||||
constexpr uint16 DRANIKSEWERSA = 331; // Sewers of Dranik
|
|
||||||
constexpr uint16 DRANIKSEWERSB = 332; // Sewers of Dranik
|
|
||||||
constexpr uint16 DRANIKSEWERSC = 333; // Sewers of Dranik
|
|
||||||
constexpr uint16 RIFTSEEKERS = 334; // Riftseekers' Sanctum
|
|
||||||
constexpr uint16 HARBINGERS = 335; // Harbinger's Spire
|
|
||||||
constexpr uint16 DRANIK = 336; // The Ruined City of Dranik
|
|
||||||
constexpr uint16 BROODLANDS = 337; // The Broodlands
|
|
||||||
constexpr uint16 STILLMOONA = 338; // Stillmoon Temple
|
|
||||||
constexpr uint16 STILLMOONB = 339; // The Ascent
|
|
||||||
constexpr uint16 THUNDERCREST = 340; // Thundercrest Isles
|
|
||||||
constexpr uint16 DELVEA = 341; // Lavaspinner's Lair
|
|
||||||
constexpr uint16 DELVEB = 342; // Tirranun's Delve
|
|
||||||
constexpr uint16 THENEST = 343; // The Nest
|
|
||||||
constexpr uint16 GUILDLOBBY = 344; // Guild Lobby
|
|
||||||
constexpr uint16 GUILDHALL = 345; // Guild Hall
|
|
||||||
constexpr uint16 BARTER = 346; // The Barter Hall
|
|
||||||
constexpr uint16 ILLSALIN = 347; // Ruins of Illsalin
|
|
||||||
constexpr uint16 ILLSALINA = 348; // Illsalin Marketplace
|
|
||||||
constexpr uint16 ILLSALINB = 349; // Temple of Korlach
|
|
||||||
constexpr uint16 ILLSALINC = 350; // The Nargil Pits
|
|
||||||
constexpr uint16 DREADSPIRE = 351; // Dreadspire Keep
|
|
||||||
constexpr uint16 DRACHNIDHIVE = 354; // The Hive
|
|
||||||
constexpr uint16 DRACHNIDHIVEA = 355; // The Hatchery
|
|
||||||
constexpr uint16 DRACHNIDHIVEB = 356; // The Cocoons
|
|
||||||
constexpr uint16 DRACHNIDHIVEC = 357; // Queen Sendaii`s Lair
|
|
||||||
constexpr uint16 WESTKORLACH = 358; // Stoneroot Falls
|
|
||||||
constexpr uint16 WESTKORLACHA = 359; // Prince's Manor
|
|
||||||
constexpr uint16 WESTKORLACHB = 360; // Caverns of the Lost
|
|
||||||
constexpr uint16 WESTKORLACHC = 361; // Lair of the Korlach
|
|
||||||
constexpr uint16 EASTKORLACH = 362; // The Undershore
|
|
||||||
constexpr uint16 EASTKORLACHA = 363; // Snarlstone Dens
|
|
||||||
constexpr uint16 SHADOWSPINE = 364; // Shadow Spine
|
|
||||||
constexpr uint16 CORATHUS = 365; // Corathus Creep
|
|
||||||
constexpr uint16 CORATHUSA = 366; // Sporali Caverns
|
|
||||||
constexpr uint16 CORATHUSB = 367; // The Corathus Mines
|
|
||||||
constexpr uint16 NEKTULOSA = 368; // Shadowed Grove
|
|
||||||
constexpr uint16 ARCSTONE = 369; // Arcstone, Isle of Spirits
|
|
||||||
constexpr uint16 RELIC = 370; // Relic, the Artifact City
|
|
||||||
constexpr uint16 SKYLANCE = 371; // Skylance
|
|
||||||
constexpr uint16 DEVASTATION = 372; // The Devastation
|
|
||||||
constexpr uint16 DEVASTATIONA = 373; // The Seething Wall
|
|
||||||
constexpr uint16 RAGE = 374; // Sverag, Stronghold of Rage
|
|
||||||
constexpr uint16 RAGEA = 375; // Razorthorn, Tower of Sullon Zek
|
|
||||||
constexpr uint16 TAKISHRUINS = 376; // Ruins of Takish-Hiz
|
|
||||||
constexpr uint16 TAKISHRUINSA = 377; // The Root of Ro
|
|
||||||
constexpr uint16 ELDDAR = 378; // The Elddar Forest
|
|
||||||
constexpr uint16 ELDDARA = 379; // Tunare's Shrine
|
|
||||||
constexpr uint16 THEATER = 380; // Theater of Blood
|
|
||||||
constexpr uint16 THEATERA = 381; // Deathknell, Tower of Dissonance
|
|
||||||
constexpr uint16 FREEPORTEAST = 382; // East Freeport
|
|
||||||
constexpr uint16 FREEPORTWEST = 383; // West Freeport
|
|
||||||
constexpr uint16 FREEPORTSEWERS = 384; // Freeport Sewers
|
|
||||||
constexpr uint16 FREEPORTACADEMY = 385; // Academy of Arcane Sciences
|
|
||||||
constexpr uint16 FREEPORTTEMPLE = 386; // Temple of Marr
|
|
||||||
constexpr uint16 FREEPORTMILITIA = 387; // Freeport Militia House: My Precious
|
|
||||||
constexpr uint16 FREEPORTARENA = 388; // Arena
|
|
||||||
constexpr uint16 FREEPORTCITYHALL = 389; // City Hall
|
|
||||||
constexpr uint16 FREEPORTTHEATER = 390; // Theater of the Tranquil
|
|
||||||
constexpr uint16 FREEPORTHALL = 391; // Hall of Truth: Bounty
|
|
||||||
constexpr uint16 NORTHRO = 392; // North Desert of Ro
|
|
||||||
constexpr uint16 SOUTHRO = 393; // South Desert of Ro
|
|
||||||
constexpr uint16 CRESCENT = 394; // Crescent Reach
|
|
||||||
constexpr uint16 MOORS = 395; // Blightfire Moors
|
|
||||||
constexpr uint16 STONEHIVE = 396; // Stone Hive
|
|
||||||
constexpr uint16 MESA = 397; // Goru`kar Mesa
|
|
||||||
constexpr uint16 ROOST = 398; // Blackfeather Roost
|
|
||||||
constexpr uint16 STEPPES = 399; // The Steppes
|
|
||||||
constexpr uint16 ICEFALL = 400; // Icefall Glacier
|
|
||||||
constexpr uint16 VALDEHOLM = 401; // Valdeholm
|
|
||||||
constexpr uint16 FROSTCRYPT = 402; // Frostcrypt, Throne of the Shade King
|
|
||||||
constexpr uint16 SUNDEROCK = 403; // Sunderock Springs
|
|
||||||
constexpr uint16 VERGALID = 404; // Vergalid Mines
|
|
||||||
constexpr uint16 DIREWIND = 405; // Direwind Cliffs
|
|
||||||
constexpr uint16 ASHENGATE = 406; // Ashengate, Reliquary of the Scale
|
|
||||||
constexpr uint16 HIGHPASSHOLD = 407; // Highpass Hold
|
|
||||||
constexpr uint16 COMMONLANDS = 408; // The Commonlands
|
|
||||||
constexpr uint16 OCEANOFTEARS = 409; // The Ocean of Tears
|
|
||||||
constexpr uint16 KITHFOREST = 410; // Kithicor Forest
|
|
||||||
constexpr uint16 BEFALLENB = 411; // Befallen
|
|
||||||
constexpr uint16 HIGHPASSKEEP = 412; // HighKeep
|
|
||||||
constexpr uint16 INNOTHULEB = 413; // The Innothule Swamp
|
|
||||||
constexpr uint16 TOXXULIA = 414; // Toxxulia Forest
|
|
||||||
constexpr uint16 MISTYTHICKET = 415; // The Misty Thicket
|
|
||||||
constexpr uint16 KATTACASTRUM = 416; // Katta Castrum
|
|
||||||
constexpr uint16 THALASSIUS = 417; // Thalassius, the Coral Keep
|
|
||||||
constexpr uint16 ATIIKI = 418; // Jewel of Atiiki
|
|
||||||
constexpr uint16 ZHISZA = 419; // Zhisza, the Shissar Sanctuary
|
|
||||||
constexpr uint16 SILYSSAR = 420; // Silyssar, New Chelsith
|
|
||||||
constexpr uint16 SOLTERIS = 421; // Solteris, the Throne of Ro
|
|
||||||
constexpr uint16 BARREN = 422; // Barren Coast
|
|
||||||
constexpr uint16 BURIEDSEA = 423; // The Buried Sea
|
|
||||||
constexpr uint16 JARDELSHOOK = 424; // Jardel's Hook
|
|
||||||
constexpr uint16 MONKEYROCK = 425; // Monkey Rock
|
|
||||||
constexpr uint16 SUNCREST = 426; // Suncrest Isle
|
|
||||||
constexpr uint16 DEADBONE = 427; // Deadbone Reef
|
|
||||||
constexpr uint16 BLACKSAIL = 428; // Blacksail Folly
|
|
||||||
constexpr uint16 MAIDENSGRAVE = 429; // Maiden's Grave
|
|
||||||
constexpr uint16 REDFEATHER = 430; // Redfeather Isle
|
|
||||||
constexpr uint16 SHIPMVP = 431; // The Open Sea
|
|
||||||
constexpr uint16 SHIPMVU = 432; // The Open Sea
|
|
||||||
constexpr uint16 SHIPPVU = 433; // The Open Sea
|
|
||||||
constexpr uint16 SHIPUVU = 434; // The Open Sea
|
|
||||||
constexpr uint16 SHIPMVM = 435; // The Open Sea
|
|
||||||
constexpr uint16 MECHANOTUS = 436; // Fortress Mechanotus
|
|
||||||
constexpr uint16 MANSION = 437; // Meldrath's Majestic Mansion
|
|
||||||
constexpr uint16 STEAMFACTORY = 438; // The Steam Factory
|
|
||||||
constexpr uint16 SHIPWORKSHOP = 439; // S.H.I.P. Workshop
|
|
||||||
constexpr uint16 GYROSPIREB = 440; // Gyrospire Beza
|
|
||||||
constexpr uint16 GYROSPIREZ = 441; // Gyrospire Zeka
|
|
||||||
constexpr uint16 DRAGONSCALE = 442; // Dragonscale Hills
|
|
||||||
constexpr uint16 LOPINGPLAINS = 443; // Loping Plains
|
|
||||||
constexpr uint16 HILLSOFSHADE = 444; // Hills of Shade
|
|
||||||
constexpr uint16 BLOODMOON = 445; // Bloodmoon Keep
|
|
||||||
constexpr uint16 CRYSTALLOS = 446; // Crystallos, Lair of the Awakened
|
|
||||||
constexpr uint16 GUARDIAN = 447; // The Mechamatic Guardian
|
|
||||||
constexpr uint16 STEAMFONTMTS = 448; // The Steamfont Mountains
|
|
||||||
constexpr uint16 CRYPTOFSHADE = 449; // Crypt of Shade
|
|
||||||
constexpr uint16 DRAGONSCALEB = 451; // Deepscar's Den
|
|
||||||
constexpr uint16 OLDFIELDOFBONE = 452; // Field of Scale
|
|
||||||
constexpr uint16 OLDKAESORAA = 453; // Kaesora Library
|
|
||||||
constexpr uint16 OLDKAESORAB = 454; // Kaesora Hatchery
|
|
||||||
constexpr uint16 OLDKURN = 455; // Kurn's Tower
|
|
||||||
constexpr uint16 OLDKITHICOR = 456; // Bloody Kithicor
|
|
||||||
constexpr uint16 OLDCOMMONS = 457; // Old Commonlands
|
|
||||||
constexpr uint16 OLDHIGHPASS = 458; // Highpass Hold
|
|
||||||
constexpr uint16 THEVOIDA = 459; // The Void
|
|
||||||
constexpr uint16 THEVOIDB = 460; // The Void
|
|
||||||
constexpr uint16 THEVOIDC = 461; // The Void
|
|
||||||
constexpr uint16 THEVOIDD = 462; // The Void
|
|
||||||
constexpr uint16 THEVOIDE = 463; // The Void
|
|
||||||
constexpr uint16 THEVOIDF = 464; // The Void
|
|
||||||
constexpr uint16 THEVOIDG = 465; // The Void
|
|
||||||
constexpr uint16 OCEANGREENHILLS = 466; // Oceangreen Hills
|
|
||||||
constexpr uint16 OCEANGREENVILLAGE = 467; // Oceangreen Village
|
|
||||||
constexpr uint16 OLDBLACKBURROW = 468; // BlackBurrow
|
|
||||||
constexpr uint16 BERTOXTEMPLE = 469; // Temple of Bertoxxulous
|
|
||||||
constexpr uint16 DISCORD = 470; // Korafax, Home of the Riders
|
|
||||||
constexpr uint16 DISCORDTOWER = 471; // Citadel of the Worldslayer
|
|
||||||
constexpr uint16 OLDBLOODFIELD = 472; // Old Bloodfields
|
|
||||||
constexpr uint16 PRECIPICEOFWAR = 473; // The Precipice of War
|
|
||||||
constexpr uint16 OLDDRANIK = 474; // City of Dranik
|
|
||||||
constexpr uint16 TOSKIRAKK = 475; // Toskirakk
|
|
||||||
constexpr uint16 KORASCIAN = 476; // Korascian Warrens
|
|
||||||
constexpr uint16 RATHECHAMBER = 477; // Rathe Council Chamber
|
|
||||||
constexpr uint16 BRELLSREST = 480; // Brell's Rest
|
|
||||||
constexpr uint16 FUNGALFOREST = 481; // Fungal Forest
|
|
||||||
constexpr uint16 UNDERQUARRY = 482; // The Underquarry
|
|
||||||
constexpr uint16 COOLINGCHAMBER = 483; // The Cooling Chamber
|
|
||||||
constexpr uint16 SHININGCITY = 484; // Kernagir, the Shining City
|
|
||||||
constexpr uint16 ARTHICREX = 485; // Arthicrex
|
|
||||||
constexpr uint16 FOUNDATION = 486; // The Foundation
|
|
||||||
constexpr uint16 LICHENCREEP = 487; // Lichen Creep
|
|
||||||
constexpr uint16 PELLUCID = 488; // Pellucid Grotto
|
|
||||||
constexpr uint16 STONESNAKE = 489; // Volska's Husk
|
|
||||||
constexpr uint16 BRELLSTEMPLE = 490; // Brell's Temple
|
|
||||||
constexpr uint16 CONVORTEUM = 491; // The Convorteum
|
|
||||||
constexpr uint16 BRELLSARENA = 492; // Brell's Arena
|
|
||||||
constexpr uint16 WEDDINGCHAPEL = 493; // Wedding Chapel
|
|
||||||
constexpr uint16 WEDDINGCHAPELDARK = 494; // Wedding Chapel
|
|
||||||
constexpr uint16 DRAGONCRYPT = 495; // Lair of the Risen
|
|
||||||
constexpr uint16 FEERROTT2 = 700; // The Feerrott
|
|
||||||
constexpr uint16 THULEHOUSE1 = 701; // House of Thule
|
|
||||||
constexpr uint16 THULEHOUSE2 = 702; // House of Thule, Upper Floors
|
|
||||||
constexpr uint16 HOUSEGARDEN = 703; // The Grounds
|
|
||||||
constexpr uint16 THULELIBRARY = 704; // The Library
|
|
||||||
constexpr uint16 WELL = 705; // The Well
|
|
||||||
constexpr uint16 FALLEN = 706; // Erudin Burning
|
|
||||||
constexpr uint16 MORELLCASTLE = 707; // Morell's Castle
|
|
||||||
constexpr uint16 SOMNIUM = 708; // Sanctum Somnium
|
|
||||||
constexpr uint16 ALKABORMARE = 709; // Al'Kabor's Nightmare
|
|
||||||
constexpr uint16 MIRAGULMARE = 710; // Miragul's Nightmare
|
|
||||||
constexpr uint16 THULEDREAM = 711; // Fear Itself
|
|
||||||
constexpr uint16 NEIGHBORHOOD = 712; // Sunrise Hills
|
|
||||||
constexpr uint16 ARGATH = 724; // Argath, Bastion of Illdaera
|
|
||||||
constexpr uint16 ARELIS = 725; // Valley of Lunanyn
|
|
||||||
constexpr uint16 SARITHCITY = 726; // Sarith, City of Tides
|
|
||||||
constexpr uint16 RUBAK = 727; // Rubak Oseka, Temple of the Sea
|
|
||||||
constexpr uint16 BEASTDOMAIN = 728; // Beasts' Domain
|
|
||||||
constexpr uint16 RESPLENDENT = 729; // The Resplendent Temple
|
|
||||||
constexpr uint16 PILLARSALRA = 730; // Pillars of Alra
|
|
||||||
constexpr uint16 WINDSONG = 731; // Windsong Sanctuary
|
|
||||||
constexpr uint16 CITYOFBRONZE = 732; // Erillion, City of Bronze
|
|
||||||
constexpr uint16 SEPULCHER = 733; // Sepulcher of Order
|
|
||||||
constexpr uint16 EASTSEPULCHER = 734; // Sepulcher East
|
|
||||||
constexpr uint16 WESTSEPULCHER = 735; // Sepulcher West
|
|
||||||
constexpr uint16 SHARDSLANDING = 752; // Shard's Landing
|
|
||||||
constexpr uint16 XORBB = 753; // Valley of King Xorbb
|
|
||||||
constexpr uint16 KAELSHARD = 754; // Kael Drakkel: The King's Madness
|
|
||||||
constexpr uint16 EASTWASTESSHARD = 755; // East Wastes: Zeixshi-Kar's Awakening
|
|
||||||
constexpr uint16 CRYSTALSHARD = 756; // The Crystal Caverns: Fragment of Fear
|
|
||||||
constexpr uint16 BREEDINGGROUNDS = 757; // The Breeding Grounds
|
|
||||||
constexpr uint16 EVILTREE = 758; // Evantil, the Vile Oak
|
|
||||||
constexpr uint16 GRELLETH = 759; // Grelleth's Palace, the Chateau of Filth
|
|
||||||
constexpr uint16 CHAPTERHOUSE = 760; // Chapterhouse of the Fallen
|
|
||||||
constexpr uint16 ARTTEST = 996; // Art Testing Domain
|
|
||||||
constexpr uint16 FHALLS = 998; // The Forgotten Halls
|
|
||||||
constexpr uint16 APPRENTICE = 999; // Designer Apprentice
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Language {
|
|
||||||
constexpr uint8 CommonTongue = 0;
|
|
||||||
constexpr uint8 Barbarian = 1;
|
|
||||||
constexpr uint8 Erudian = 2;
|
|
||||||
constexpr uint8 Elvish = 3;
|
|
||||||
constexpr uint8 DarkElvish = 4;
|
|
||||||
constexpr uint8 Dwarvish = 5;
|
|
||||||
constexpr uint8 Troll = 6;
|
|
||||||
constexpr uint8 Ogre = 7;
|
|
||||||
constexpr uint8 Gnomish = 8;
|
|
||||||
constexpr uint8 Halfling = 9;
|
|
||||||
constexpr uint8 ThievesCant = 10;
|
|
||||||
constexpr uint8 OldErudian = 11;
|
|
||||||
constexpr uint8 ElderElvish = 12;
|
|
||||||
constexpr uint8 Froglok = 13;
|
|
||||||
constexpr uint8 Goblin = 14;
|
|
||||||
constexpr uint8 Gnoll = 15;
|
|
||||||
constexpr uint8 CombineTongue = 16;
|
|
||||||
constexpr uint8 ElderTeirDal = 17;
|
|
||||||
constexpr uint8 Lizardman = 18;
|
|
||||||
constexpr uint8 Orcish = 19;
|
|
||||||
constexpr uint8 Faerie = 20;
|
|
||||||
constexpr uint8 Dragon = 21;
|
|
||||||
constexpr uint8 ElderDragon = 22;
|
|
||||||
constexpr uint8 DarkSpeech = 23;
|
|
||||||
constexpr uint8 VahShir = 24;
|
|
||||||
constexpr uint8 Alaran = 25;
|
|
||||||
constexpr uint8 Hadal = 26;
|
|
||||||
constexpr uint8 Unknown27 = 27;
|
|
||||||
|
|
||||||
constexpr uint8 MaxValue = 100;
|
|
||||||
}
|
|
||||||
namespace PetInfoType {
|
|
||||||
constexpr int Current = 0;
|
|
||||||
constexpr int Suspended = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace BuffEffectType {
|
|
||||||
constexpr uint8 None = 0;
|
|
||||||
constexpr uint8 Buff = 2;
|
|
||||||
constexpr uint8 InverseBuff = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace AlternateCurrencyMode {
|
|
||||||
constexpr uint32 Update = 7;
|
|
||||||
constexpr uint32 Populate = 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
FilterNone = 0,
|
FilterNone = 0,
|
||||||
@@ -757,11 +231,11 @@ typedef enum {
|
|||||||
FilterPetMisses = 21, //0=show, 1=hide
|
FilterPetMisses = 21, //0=show, 1=hide
|
||||||
FilterFocusEffects = 22, //0=show, 1=hide
|
FilterFocusEffects = 22, //0=show, 1=hide
|
||||||
FilterPetSpells = 23, //0=show, 1=hide
|
FilterPetSpells = 23, //0=show, 1=hide
|
||||||
FilterHealOverTime = 24, //0=show, 1=mine only, 2=hide
|
FilterHealOverTime = 24, //0=show, 1=hide
|
||||||
FilterItemSpeech = 25, //0=show, 1=hide // RoF2 Confirmed
|
FilterUnknown25 = 25,
|
||||||
FilterStrikethrough = 26, //0=show, 1=hide // RoF2 Confirmed
|
FilterUnknown26 = 26,
|
||||||
FilterStuns = 27, //0=show, 1=hide // RoF2 Confirmed
|
FilterUnknown27 = 27,
|
||||||
FilterBardSongsOnPets = 28, //0=show, 1=hide // RoF2 Confirmed
|
FilterUnknown28 = 28,
|
||||||
_FilterCount
|
_FilterCount
|
||||||
} eqFilterType;
|
} eqFilterType;
|
||||||
|
|
||||||
@@ -772,47 +246,67 @@ typedef enum {
|
|||||||
FilterShowSelfOnly
|
FilterShowSelfOnly
|
||||||
} eqFilterMode;
|
} eqFilterMode;
|
||||||
|
|
||||||
#define STAT_STR 0
|
#define STAT_STR 0
|
||||||
#define STAT_STA 1
|
#define STAT_STA 1
|
||||||
#define STAT_AGI 2
|
#define STAT_AGI 2
|
||||||
#define STAT_DEX 3
|
#define STAT_DEX 3
|
||||||
#define STAT_INT 4
|
#define STAT_INT 4
|
||||||
#define STAT_WIS 5
|
#define STAT_WIS 5
|
||||||
#define STAT_CHA 6
|
#define STAT_CHA 6
|
||||||
#define STAT_MAGIC 7
|
#define STAT_MAGIC 7
|
||||||
#define STAT_COLD 8
|
#define STAT_COLD 8
|
||||||
#define STAT_FIRE 9
|
#define STAT_FIRE 9
|
||||||
#define STAT_POISON 10
|
#define STAT_POISON 10
|
||||||
#define STAT_DISEASE 11
|
#define STAT_DISEASE 11
|
||||||
#define STAT_MANA 12
|
#define STAT_MANA 12
|
||||||
#define STAT_HP 13
|
#define STAT_HP 13
|
||||||
#define STAT_AC 14
|
#define STAT_AC 14
|
||||||
#define STAT_ENDURANCE 15
|
#define STAT_ENDURANCE 15
|
||||||
#define STAT_ATTACK 16
|
#define STAT_ATTACK 16
|
||||||
#define STAT_HP_REGEN 17
|
#define STAT_HP_REGEN 17
|
||||||
#define STAT_MANA_REGEN 18
|
#define STAT_MANA_REGEN 18
|
||||||
#define STAT_HASTE 19
|
#define STAT_HASTE 19
|
||||||
#define STAT_DAMAGE_SHIELD 20
|
#define STAT_DAMAGE_SHIELD 20
|
||||||
#define STAT_DS_MITIGATION 22
|
|
||||||
#define STAT_HEAL_AMOUNT 23
|
/*
|
||||||
#define STAT_SPELL_DAMAGE 24
|
** Recast timer types. Used as an off set to charProfileStruct timers.
|
||||||
#define STAT_CLAIRVOYANCE 25
|
**
|
||||||
#define STAT_HEROIC_AGILITY 26
|
** (Another orphaned enumeration...)
|
||||||
#define STAT_HEROIC_CHARISMA 27
|
*/
|
||||||
#define STAT_HEROIC_DEXTERITY 28
|
enum RecastTimerTypes
|
||||||
#define STAT_HEROIC_INTELLIGENCE 29
|
{
|
||||||
#define STAT_HEROIC_STAMINA 30
|
RecTimer_0 = 0,
|
||||||
#define STAT_HEROIC_STRENGTH 31
|
RecTimer_1,
|
||||||
#define STAT_HEROIC_WISDOM 32
|
RecTimer_WeaponHealClick, // 2
|
||||||
#define STAT_BASH 33
|
RecTimer_MuramiteBaneNukeClick, // 3
|
||||||
#define STAT_BACKSTAB 34
|
RecTimer_4,
|
||||||
#define STAT_DRAGON_PUNCH 35
|
RecTimer_DispellClick, // 5 (also click heal orbs?)
|
||||||
#define STAT_EAGLE_STRIKE 36
|
RecTimer_Epic, // 6
|
||||||
#define STAT_FLYING_KICK 37
|
RecTimer_OoWBPClick, // 7
|
||||||
#define STAT_KICK 38
|
RecTimer_VishQuestClassItem, // 8
|
||||||
#define STAT_ROUND_KICK 39
|
RecTimer_HealPotion, // 9
|
||||||
#define STAT_TIGER_CLAW 40
|
RecTimer_10,
|
||||||
#define STAT_FRENZY 41
|
RecTimer_11,
|
||||||
|
RecTimer_12,
|
||||||
|
RecTimer_13,
|
||||||
|
RecTimer_14,
|
||||||
|
RecTimer_15,
|
||||||
|
RecTimer_16,
|
||||||
|
RecTimer_17,
|
||||||
|
RecTimer_18,
|
||||||
|
RecTimer_ModRod, // 19
|
||||||
|
_RecTimerCount
|
||||||
|
};
|
||||||
|
|
||||||
|
enum GroupUpdateAction
|
||||||
|
{
|
||||||
|
GUA_Joined = 0,
|
||||||
|
GUA_Left = 1,
|
||||||
|
GUA_LastLeft = 6,
|
||||||
|
GUA_FullGroupInfo = 7,
|
||||||
|
GUA_MakeLeader = 8,
|
||||||
|
GUA_Started = 9
|
||||||
|
};
|
||||||
|
|
||||||
static const uint8 DamageTypeSomething = 0x1C; //0x1c is something...
|
static const uint8 DamageTypeSomething = 0x1C; //0x1c is something...
|
||||||
static const uint8 DamageTypeFalling = 0xFC;
|
static const uint8 DamageTypeFalling = 0xFC;
|
||||||
@@ -944,9 +438,12 @@ static const uint8 SkillDamageTypes[EQ::skills::HIGHEST_SKILL + 1] = // change t
|
|||||||
|
|
||||||
static const uint32 MAX_SPELL_DB_ID_VAL = 65535;
|
static const uint32 MAX_SPELL_DB_ID_VAL = 65535;
|
||||||
|
|
||||||
static const uint32 DB_FACTION_GEM_CHOPPERS = 255;
|
static const uint32 DB_SPELL_CAZIC_TOUCH = 982;
|
||||||
static const uint32 DB_FACTION_HERETICS = 265;
|
static const uint32 DB_SPELL_TOUCH_OF_VINITRAS = 2859;
|
||||||
static const uint32 DB_FACTION_KING_AKANON = 333;
|
|
||||||
|
static const uint32 DB_FACTION_GEM_CHOPPERS = 255;
|
||||||
|
static const uint32 DB_FACTION_HERETICS = 265;
|
||||||
|
static const uint32 DB_FACTION_KING_AKANON = 333;
|
||||||
|
|
||||||
enum ChatChannelNames : uint16
|
enum ChatChannelNames : uint16
|
||||||
{
|
{
|
||||||
@@ -993,126 +490,4 @@ enum class DynamicZoneMemberStatus : uint8_t
|
|||||||
LinkDead
|
LinkDead
|
||||||
};
|
};
|
||||||
|
|
||||||
enum StartZoneIndex {
|
|
||||||
Odus = 0,
|
|
||||||
Qeynos,
|
|
||||||
Halas,
|
|
||||||
Rivervale,
|
|
||||||
Freeport,
|
|
||||||
Neriak,
|
|
||||||
Grobb,
|
|
||||||
Oggok,
|
|
||||||
Kaladim,
|
|
||||||
GreaterFaydark,
|
|
||||||
Felwithe,
|
|
||||||
Akanon,
|
|
||||||
Cabilis,
|
|
||||||
SharVahl
|
|
||||||
};
|
|
||||||
|
|
||||||
enum FVNoDropFlagRule
|
|
||||||
{
|
|
||||||
Disabled = 0,
|
|
||||||
Enabled = 1,
|
|
||||||
AdminOnly = 2
|
|
||||||
};
|
|
||||||
|
|
||||||
enum Anonymity : uint8
|
|
||||||
{
|
|
||||||
NotAnonymous,
|
|
||||||
Anonymous,
|
|
||||||
Roleplaying
|
|
||||||
};
|
|
||||||
|
|
||||||
enum ZoningMessage : int8 {
|
|
||||||
ZoneNoMessage = 0,
|
|
||||||
ZoneSuccess = 1,
|
|
||||||
ZoneNotReady = -1,
|
|
||||||
ZoneValidPC = -2,
|
|
||||||
ZoneStoryZone = -3,
|
|
||||||
ZoneNoExpansion = -6,
|
|
||||||
ZoneNoExperience = -7
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class RecipeCountType : uint8
|
|
||||||
{
|
|
||||||
Component,
|
|
||||||
Container,
|
|
||||||
Fail,
|
|
||||||
Salvage,
|
|
||||||
Success
|
|
||||||
};
|
|
||||||
|
|
||||||
#define ALT_CURRENCY_ID_RADIANT 4
|
|
||||||
#define ALT_CURRENCY_ID_EBON 5
|
|
||||||
|
|
||||||
enum ResurrectionActions
|
|
||||||
{
|
|
||||||
Decline,
|
|
||||||
Accept
|
|
||||||
};
|
|
||||||
|
|
||||||
enum ScribeSpellActions
|
|
||||||
{
|
|
||||||
Scribe,
|
|
||||||
Memorize,
|
|
||||||
Unmemorize
|
|
||||||
};
|
|
||||||
|
|
||||||
enum SpellTimeRestrictions
|
|
||||||
{
|
|
||||||
NoRestriction,
|
|
||||||
Day,
|
|
||||||
Night
|
|
||||||
};
|
|
||||||
|
|
||||||
enum MoneyTypes
|
|
||||||
{
|
|
||||||
Copper,
|
|
||||||
Silver,
|
|
||||||
Gold,
|
|
||||||
Platinum
|
|
||||||
};
|
|
||||||
|
|
||||||
enum MoneySubtypes
|
|
||||||
{
|
|
||||||
Personal,
|
|
||||||
Bank,
|
|
||||||
Cursor,
|
|
||||||
SharedBank // Platinum Only
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace RaidLootType {
|
|
||||||
constexpr uint32 LeaderOnly = 1;
|
|
||||||
constexpr uint32 LeaderAndGroupLeadersOnly = 2;
|
|
||||||
constexpr uint32 LeaderSelected = 3;
|
|
||||||
constexpr uint32 EntireRaid = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace LeadershipAbilitySlot {
|
|
||||||
constexpr uint16 HealthOfTargetsTarget = 14;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum ExpSource
|
|
||||||
{
|
|
||||||
Quest,
|
|
||||||
GM,
|
|
||||||
Kill,
|
|
||||||
Death,
|
|
||||||
Resurrection,
|
|
||||||
LDoNChest,
|
|
||||||
Task,
|
|
||||||
Sacrifice
|
|
||||||
};
|
|
||||||
|
|
||||||
#define PARCEL_SEND_ITEMS 0
|
|
||||||
#define PARCEL_SEND_MONEY 1
|
|
||||||
#define PARCEL_MONEY_ITEM_ID 99990 // item id of money
|
|
||||||
#define PARCEL_LIMIT 5
|
|
||||||
#define PARCEL_BEGIN_SLOT 1
|
|
||||||
|
|
||||||
namespace DoorType {
|
|
||||||
constexpr uint32 BuyerStall = 155;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /*COMMON_EQ_CONSTANTS_H*/
|
#endif /*COMMON_EQ_CONSTANTS_H*/
|
||||||
|
|||||||
+22
-30
@@ -1,5 +1,5 @@
|
|||||||
/* EQEMu: Everquest Server Emulator
|
/* EQEMu: Everquest Server Emulator
|
||||||
|
|
||||||
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
Copyright (C) 2001-2016 EQEMu Development Team (http://eqemulator.net)
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
This program is free software; you can redistribute it and/or modify
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
are required to give you total support for your newly bought product;
|
are required to give you total support for your newly bought product;
|
||||||
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||||||
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
@@ -26,7 +26,7 @@
|
|||||||
|
|
||||||
static bool global_dictionary_init = false;
|
static bool global_dictionary_init = false;
|
||||||
void EQ::InitializeDynamicLookups() {
|
void EQ::InitializeDynamicLookups() {
|
||||||
if (global_dictionary_init)
|
if (global_dictionary_init == true)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
constants::InitializeDynamicLookups();
|
constants::InitializeDynamicLookups();
|
||||||
@@ -47,7 +47,6 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
ClientUnknown::constants::EXPANSION_BIT,
|
ClientUnknown::constants::EXPANSION_BIT,
|
||||||
ClientUnknown::constants::EXPANSIONS_MASK,
|
ClientUnknown::constants::EXPANSIONS_MASK,
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL,
|
|
||||||
ClientUnknown::INULL
|
ClientUnknown::INULL
|
||||||
),
|
),
|
||||||
/*[ClientVersion::Client62] =*/
|
/*[ClientVersion::Client62] =*/
|
||||||
@@ -56,7 +55,6 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
Client62::constants::EXPANSION_BIT,
|
Client62::constants::EXPANSION_BIT,
|
||||||
Client62::constants::EXPANSIONS_MASK,
|
Client62::constants::EXPANSIONS_MASK,
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
Client62::INULL,
|
|
||||||
Client62::INULL
|
Client62::INULL
|
||||||
),
|
),
|
||||||
/*[ClientVersion::Titanium] =*/
|
/*[ClientVersion::Titanium] =*/
|
||||||
@@ -65,8 +63,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
Titanium::constants::EXPANSION_BIT,
|
Titanium::constants::EXPANSION_BIT,
|
||||||
Titanium::constants::EXPANSIONS_MASK,
|
Titanium::constants::EXPANSIONS_MASK,
|
||||||
Titanium::constants::CHARACTER_CREATION_LIMIT,
|
Titanium::constants::CHARACTER_CREATION_LIMIT,
|
||||||
Titanium::constants::SAY_LINK_BODY_SIZE,
|
Titanium::constants::SAY_LINK_BODY_SIZE
|
||||||
Titanium::INULL
|
|
||||||
),
|
),
|
||||||
/*[ClientVersion::SoF] =*/
|
/*[ClientVersion::SoF] =*/
|
||||||
EQ::constants::LookupEntry(
|
EQ::constants::LookupEntry(
|
||||||
@@ -74,8 +71,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
SoF::constants::EXPANSION_BIT,
|
SoF::constants::EXPANSION_BIT,
|
||||||
SoF::constants::EXPANSIONS_MASK,
|
SoF::constants::EXPANSIONS_MASK,
|
||||||
SoF::constants::CHARACTER_CREATION_LIMIT,
|
SoF::constants::CHARACTER_CREATION_LIMIT,
|
||||||
SoF::constants::SAY_LINK_BODY_SIZE,
|
SoF::constants::SAY_LINK_BODY_SIZE
|
||||||
SoF::INULL
|
|
||||||
),
|
),
|
||||||
/*[ClientVersion::SoD] =*/
|
/*[ClientVersion::SoD] =*/
|
||||||
EQ::constants::LookupEntry(
|
EQ::constants::LookupEntry(
|
||||||
@@ -83,8 +79,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
SoD::constants::EXPANSION_BIT,
|
SoD::constants::EXPANSION_BIT,
|
||||||
SoD::constants::EXPANSIONS_MASK,
|
SoD::constants::EXPANSIONS_MASK,
|
||||||
SoD::constants::CHARACTER_CREATION_LIMIT,
|
SoD::constants::CHARACTER_CREATION_LIMIT,
|
||||||
SoD::constants::SAY_LINK_BODY_SIZE,
|
SoD::constants::SAY_LINK_BODY_SIZE
|
||||||
SoD::INULL
|
|
||||||
),
|
),
|
||||||
/*[ClientVersion::UF] =*/
|
/*[ClientVersion::UF] =*/
|
||||||
EQ::constants::LookupEntry(
|
EQ::constants::LookupEntry(
|
||||||
@@ -92,8 +87,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
UF::constants::EXPANSION_BIT,
|
UF::constants::EXPANSION_BIT,
|
||||||
UF::constants::EXPANSIONS_MASK,
|
UF::constants::EXPANSIONS_MASK,
|
||||||
UF::constants::CHARACTER_CREATION_LIMIT,
|
UF::constants::CHARACTER_CREATION_LIMIT,
|
||||||
UF::constants::SAY_LINK_BODY_SIZE,
|
UF::constants::SAY_LINK_BODY_SIZE
|
||||||
UF::INULL
|
|
||||||
),
|
),
|
||||||
/*[ClientVersion::RoF] =*/
|
/*[ClientVersion::RoF] =*/
|
||||||
EQ::constants::LookupEntry(
|
EQ::constants::LookupEntry(
|
||||||
@@ -101,8 +95,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
RoF::constants::EXPANSION_BIT,
|
RoF::constants::EXPANSION_BIT,
|
||||||
RoF::constants::EXPANSIONS_MASK,
|
RoF::constants::EXPANSIONS_MASK,
|
||||||
RoF::constants::CHARACTER_CREATION_LIMIT,
|
RoF::constants::CHARACTER_CREATION_LIMIT,
|
||||||
RoF::constants::SAY_LINK_BODY_SIZE,
|
RoF::constants::SAY_LINK_BODY_SIZE
|
||||||
RoF::INULL
|
|
||||||
),
|
),
|
||||||
/*[ClientVersion::RoF2] =*/
|
/*[ClientVersion::RoF2] =*/
|
||||||
EQ::constants::LookupEntry(
|
EQ::constants::LookupEntry(
|
||||||
@@ -110,8 +103,7 @@ static const EQ::constants::LookupEntry constants_static_lookup_entries[EQ::vers
|
|||||||
RoF2::constants::EXPANSION_BIT,
|
RoF2::constants::EXPANSION_BIT,
|
||||||
RoF2::constants::EXPANSIONS_MASK,
|
RoF2::constants::EXPANSIONS_MASK,
|
||||||
RoF2::constants::CHARACTER_CREATION_LIMIT,
|
RoF2::constants::CHARACTER_CREATION_LIMIT,
|
||||||
RoF2::constants::SAY_LINK_BODY_SIZE,
|
RoF2::constants::SAY_LINK_BODY_SIZE
|
||||||
RoF2::constants::MAX_BAZAAR_TRADERS
|
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -175,7 +167,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
ClientUnknown::INULL, ClientUnknown::INULL, ClientUnknown::INULL,
|
ClientUnknown::INULL, ClientUnknown::INULL, ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL
|
ClientUnknown::INULL
|
||||||
),
|
),
|
||||||
|
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
@@ -183,7 +175,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
ClientUnknown::INULL,
|
ClientUnknown::INULL,
|
||||||
|
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
@@ -202,7 +194,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
Client62::INULL, Client62::INULL, Client62::INULL,
|
Client62::INULL, Client62::INULL, Client62::INULL,
|
||||||
Client62::INULL
|
Client62::INULL
|
||||||
),
|
),
|
||||||
|
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
@@ -210,7 +202,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
Client62::INULL,
|
Client62::INULL,
|
||||||
|
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
@@ -229,7 +221,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
Titanium::INULL, Titanium::INULL, Titanium::INULL,
|
Titanium::INULL, Titanium::INULL, Titanium::INULL,
|
||||||
Titanium::invtype::OTHER_SIZE
|
Titanium::invtype::OTHER_SIZE
|
||||||
),
|
),
|
||||||
|
|
||||||
Titanium::invslot::EQUIPMENT_BITMASK,
|
Titanium::invslot::EQUIPMENT_BITMASK,
|
||||||
Titanium::invslot::GENERAL_BITMASK,
|
Titanium::invslot::GENERAL_BITMASK,
|
||||||
Titanium::invslot::CURSOR_BITMASK,
|
Titanium::invslot::CURSOR_BITMASK,
|
||||||
@@ -237,7 +229,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
Titanium::invslot::CORPSE_BITMASK,
|
Titanium::invslot::CORPSE_BITMASK,
|
||||||
Titanium::invbag::SLOT_COUNT,
|
Titanium::invbag::SLOT_COUNT,
|
||||||
Titanium::invaug::SOCKET_COUNT,
|
Titanium::invaug::SOCKET_COUNT,
|
||||||
|
|
||||||
Titanium::inventory::AllowEmptyBagInBag,
|
Titanium::inventory::AllowEmptyBagInBag,
|
||||||
Titanium::inventory::AllowClickCastFromBag,
|
Titanium::inventory::AllowClickCastFromBag,
|
||||||
Titanium::inventory::ConcatenateInvTypeLimbo,
|
Titanium::inventory::ConcatenateInvTypeLimbo,
|
||||||
@@ -256,7 +248,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
SoF::INULL, SoF::INULL, SoF::INULL,
|
SoF::INULL, SoF::INULL, SoF::INULL,
|
||||||
SoF::invtype::OTHER_SIZE
|
SoF::invtype::OTHER_SIZE
|
||||||
),
|
),
|
||||||
|
|
||||||
SoF::invslot::EQUIPMENT_BITMASK,
|
SoF::invslot::EQUIPMENT_BITMASK,
|
||||||
SoF::invslot::GENERAL_BITMASK,
|
SoF::invslot::GENERAL_BITMASK,
|
||||||
SoF::invslot::CURSOR_BITMASK,
|
SoF::invslot::CURSOR_BITMASK,
|
||||||
@@ -264,7 +256,7 @@ static const EQ::inventory::LookupEntry inventory_static_lookup_entries[EQ::vers
|
|||||||
SoF::invslot::CORPSE_BITMASK,
|
SoF::invslot::CORPSE_BITMASK,
|
||||||
SoF::invbag::SLOT_COUNT,
|
SoF::invbag::SLOT_COUNT,
|
||||||
SoF::invaug::SOCKET_COUNT,
|
SoF::invaug::SOCKET_COUNT,
|
||||||
|
|
||||||
SoF::inventory::AllowEmptyBagInBag,
|
SoF::inventory::AllowEmptyBagInBag,
|
||||||
SoF::inventory::AllowClickCastFromBag,
|
SoF::inventory::AllowClickCastFromBag,
|
||||||
SoF::inventory::ConcatenateInvTypeLimbo,
|
SoF::inventory::ConcatenateInvTypeLimbo,
|
||||||
@@ -771,7 +763,7 @@ void EQ::inventory::InitializeDynamicLookups() {
|
|||||||
// Notes:
|
// Notes:
|
||||||
// - Currently, there are only 3 known expansions that affect inventory-related settings in the clients..
|
// - Currently, there are only 3 known expansions that affect inventory-related settings in the clients..
|
||||||
// -- Expansion::PoR "Prophecy of Ro" - toggles between 24 (set) and 16 (clear) bank slots
|
// -- Expansion::PoR "Prophecy of Ro" - toggles between 24 (set) and 16 (clear) bank slots
|
||||||
// -- Expansion::TBS "The Buried Sea" - toggles slotPowerSource activated (set) and deactivated (clear)
|
// -- Expansion::TBS "The Buried Sea" - toggles slotPowerSource activated (set) and deactivated (clear)
|
||||||
// -- Expansion::HoT "House of Thule" - toggles slotGeneral9/slotGeneral10 activated (set) and deactivated (clear)
|
// -- Expansion::HoT "House of Thule" - toggles slotGeneral9/slotGeneral10 activated (set) and deactivated (clear)
|
||||||
// - Corspe size does not appear to reflect loss of active possessions slots
|
// - Corspe size does not appear to reflect loss of active possessions slots
|
||||||
// - Inspect size does not appear to reflect loss of active equipment slots
|
// - Inspect size does not appear to reflect loss of active equipment slots
|
||||||
@@ -780,7 +772,7 @@ void EQ::inventory::InitializeDynamicLookups() {
|
|||||||
// - General9 and General10 slots are activated by GM flag when expansion bit is (clear)
|
// - General9 and General10 slots are activated by GM flag when expansion bit is (clear)
|
||||||
// - Obviously, the client must support the expansion to allow any (set) or override condition
|
// - Obviously, the client must support the expansion to allow any (set) or override condition
|
||||||
|
|
||||||
const uint32 dynamic_check_mask =
|
const uint32 dynamic_check_mask =
|
||||||
(
|
(
|
||||||
EQ::expansions::bitPoR |
|
EQ::expansions::bitPoR |
|
||||||
EQ::expansions::bitTBS |
|
EQ::expansions::bitTBS |
|
||||||
@@ -1218,10 +1210,10 @@ void EQ::spells::InitializeDynamicLookups() {
|
|||||||
if (spells_dictionary_init == true)
|
if (spells_dictionary_init == true)
|
||||||
return;
|
return;
|
||||||
spells_dictionary_init = true;
|
spells_dictionary_init = true;
|
||||||
|
|
||||||
if (RuleB(World, UseClientBasedExpansionSettings))
|
if (RuleB(World, UseClientBasedExpansionSettings))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// use static references for now
|
// use static references for now
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1247,7 +1239,7 @@ const EQ::spells::LookupEntry* EQ::spells::DynamicGMLookup(versions::ClientVersi
|
|||||||
client_version = versions::ValidateClientVersion(client_version);
|
client_version = versions::ValidateClientVersion(client_version);
|
||||||
if (spells_dynamic_gm_lookup_entries[static_cast<int>(client_version)])
|
if (spells_dynamic_gm_lookup_entries[static_cast<int>(client_version)])
|
||||||
return spells_dynamic_gm_lookup_entries[static_cast<int>(client_version)].get();
|
return spells_dynamic_gm_lookup_entries[static_cast<int>(client_version)].get();
|
||||||
|
|
||||||
return &spells_static_lookup_entries[static_cast<int>(client_version)];
|
return &spells_static_lookup_entries[static_cast<int>(client_version)];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+3
-6
@@ -42,7 +42,6 @@ namespace EQ
|
|||||||
uint32 ExpansionsMask;
|
uint32 ExpansionsMask;
|
||||||
int16 CharacterCreationLimit;
|
int16 CharacterCreationLimit;
|
||||||
size_t SayLinkBodySize;
|
size_t SayLinkBodySize;
|
||||||
uint32 BazaarTraderLimit;
|
|
||||||
|
|
||||||
LookupEntry(const LookupEntry *lookup_entry) { }
|
LookupEntry(const LookupEntry *lookup_entry) { }
|
||||||
LookupEntry(
|
LookupEntry(
|
||||||
@@ -50,15 +49,13 @@ namespace EQ
|
|||||||
uint32 ExpansionBit,
|
uint32 ExpansionBit,
|
||||||
uint32 ExpansionsMask,
|
uint32 ExpansionsMask,
|
||||||
int16 CharacterCreationLimit,
|
int16 CharacterCreationLimit,
|
||||||
size_t SayLinkBodySize,
|
size_t SayLinkBodySize
|
||||||
uint32 BazaarTraderLimit
|
|
||||||
) :
|
) :
|
||||||
Expansion(Expansion),
|
Expansion(Expansion),
|
||||||
ExpansionBit(ExpansionBit),
|
ExpansionBit(ExpansionBit),
|
||||||
ExpansionsMask(ExpansionsMask),
|
ExpansionsMask(ExpansionsMask),
|
||||||
CharacterCreationLimit(CharacterCreationLimit),
|
CharacterCreationLimit(CharacterCreationLimit),
|
||||||
SayLinkBodySize(SayLinkBodySize),
|
SayLinkBodySize(SayLinkBodySize)
|
||||||
BazaarTraderLimit(BazaarTraderLimit)
|
|
||||||
{ }
|
{ }
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -132,7 +129,7 @@ namespace EQ
|
|||||||
|
|
||||||
LookupEntry(const LookupEntry *lookup_entry) { }
|
LookupEntry(const LookupEntry *lookup_entry) { }
|
||||||
LookupEntry(
|
LookupEntry(
|
||||||
const InventoryTypeSize_Struct& InventoryTypeSize,
|
InventoryTypeSize_Struct InventoryTypeSize,
|
||||||
uint64 EquipmentBitmask,
|
uint64 EquipmentBitmask,
|
||||||
uint64 GeneralBitmask,
|
uint64 GeneralBitmask,
|
||||||
uint64 CursorBitmask,
|
uint64 CursorBitmask,
|
||||||
|
|||||||
+138
-1
@@ -236,6 +236,26 @@ uint32 EQApplicationPacket::serialize(uint16 opcode, unsigned char *dest) const
|
|||||||
return size+OpCodeBytes;
|
return size+OpCodeBytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*EQProtocolPacket::EQProtocolPacket(uint16 op, const unsigned char *buf, uint32 len)
|
||||||
|
: BasePacket(buf, len),
|
||||||
|
opcode(op)
|
||||||
|
{
|
||||||
|
|
||||||
|
uint32 offset;
|
||||||
|
opcode=ntohs(*(const uint16 *)buf);
|
||||||
|
offset=2;
|
||||||
|
|
||||||
|
if (len-offset) {
|
||||||
|
pBuffer= new unsigned char[len-offset];
|
||||||
|
memcpy(pBuffer,buf+offset,len-offset);
|
||||||
|
size=len-offset;
|
||||||
|
} else {
|
||||||
|
pBuffer=nullptr;
|
||||||
|
size=0;
|
||||||
|
}
|
||||||
|
OpMgr=&RawOpcodeManager;
|
||||||
|
}*/
|
||||||
|
|
||||||
bool EQProtocolPacket::combine(const EQProtocolPacket *rhs)
|
bool EQProtocolPacket::combine(const EQProtocolPacket *rhs)
|
||||||
{
|
{
|
||||||
bool result=false;
|
bool result=false;
|
||||||
@@ -267,6 +287,74 @@ bool result=false;
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
this is the code to do app-layer combining, instead of protocol layer.
|
||||||
|
this was taken out due to complex interactions with the opcode manager,
|
||||||
|
and will require a bit more thinking (likely moving into EQStream) to
|
||||||
|
get running again... but might be a good thing some day.
|
||||||
|
|
||||||
|
bool EQApplicationPacket::combine(const EQApplicationPacket *rhs)
|
||||||
|
{
|
||||||
|
uint32 newsize=0, offset=0;
|
||||||
|
unsigned char *tmpbuffer=nullptr;
|
||||||
|
|
||||||
|
if (opcode!=OP_AppCombined) {
|
||||||
|
newsize=app_opcode_size+size+(size>254?3:1)+app_opcode_size+rhs->size+(rhs->size>254?3:1);
|
||||||
|
tmpbuffer=new unsigned char [newsize];
|
||||||
|
offset=0;
|
||||||
|
if (size>254) {
|
||||||
|
tmpbuffer[offset++]=0xff;
|
||||||
|
*(uint16 *)(tmpbuffer+offset)=htons(size);
|
||||||
|
offset+=1;
|
||||||
|
} else {
|
||||||
|
tmpbuffer[offset++]=size;
|
||||||
|
}
|
||||||
|
offset+=serialize(tmpbuffer+offset);
|
||||||
|
} else {
|
||||||
|
newsize=size+app_opcode_size+rhs->size+(rhs->size>254?3:1);
|
||||||
|
tmpbuffer=new unsigned char [newsize];
|
||||||
|
memcpy(tmpbuffer,pBuffer,size);
|
||||||
|
offset=size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rhs->size>254) {
|
||||||
|
tmpbuffer[offset++]=0xff;
|
||||||
|
*(uint16 *)(tmpbuffer+offset)=htons(rhs->size);
|
||||||
|
offset+=1;
|
||||||
|
} else {
|
||||||
|
tmpbuffer[offset++]=rhs->size;
|
||||||
|
}
|
||||||
|
offset+=rhs->serialize(tmpbuffer+offset);
|
||||||
|
|
||||||
|
size=offset;
|
||||||
|
opcode=OP_AppCombined;
|
||||||
|
|
||||||
|
delete[] pBuffer;
|
||||||
|
pBuffer=tmpbuffer;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
bool EQProtocolPacket::ValidateCRC(const unsigned char *buffer, int length, uint32 Key)
|
||||||
|
{
|
||||||
|
bool valid=false;
|
||||||
|
// OP_SessionRequest, OP_SessionResponse, OP_OutOfSession are not CRC'd
|
||||||
|
if (buffer[0]==0x00 && (buffer[1]==OP_SessionRequest || buffer[1]==OP_SessionResponse || buffer[1]==OP_OutOfSession)) {
|
||||||
|
valid=true;
|
||||||
|
} else {
|
||||||
|
uint16 comp_crc=CRC16(buffer,length-2,Key);
|
||||||
|
uint16 packet_crc=ntohs(*(const uint16 *)(buffer+length-2));
|
||||||
|
#ifdef EQN_DEBUG
|
||||||
|
if (packet_crc && comp_crc != packet_crc) {
|
||||||
|
std::cout << "CRC mismatch: comp=" << std::hex << comp_crc << ", packet=" << packet_crc << std::dec << std::endl;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
valid = (!packet_crc || comp_crc == packet_crc);
|
||||||
|
}
|
||||||
|
return valid;
|
||||||
|
}
|
||||||
|
|
||||||
uint32 EQProtocolPacket::Decompress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize)
|
uint32 EQProtocolPacket::Decompress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize)
|
||||||
{
|
{
|
||||||
uint32 newlen=0;
|
uint32 newlen=0;
|
||||||
@@ -315,6 +403,55 @@ uint32 flag_offset=1,newlength;
|
|||||||
return newlength;
|
return newlength;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EQProtocolPacket::ChatDecode(unsigned char *buffer, int size, int DecodeKey)
|
||||||
|
{
|
||||||
|
if ((size >= 2) && buffer[1]!=0x01 && buffer[0]!=0x02 && buffer[0]!=0x1d) {
|
||||||
|
int Key=DecodeKey;
|
||||||
|
unsigned char *test=(unsigned char *)malloc(size);
|
||||||
|
buffer+=2;
|
||||||
|
size-=2;
|
||||||
|
|
||||||
|
int i;
|
||||||
|
for (i = 0 ; i+4 <= size ; i+=4)
|
||||||
|
{
|
||||||
|
int pt = (*(int*)&buffer[i])^(Key);
|
||||||
|
Key = (*(int*)&buffer[i]);
|
||||||
|
*(int*)&test[i]=pt;
|
||||||
|
}
|
||||||
|
unsigned char KC=Key&0xFF;
|
||||||
|
for ( ; i < size ; i++)
|
||||||
|
{
|
||||||
|
test[i]=buffer[i]^KC;
|
||||||
|
}
|
||||||
|
memcpy(buffer,test,size);
|
||||||
|
free(test);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EQProtocolPacket::ChatEncode(unsigned char *buffer, int size, int EncodeKey)
|
||||||
|
{
|
||||||
|
if (buffer[1]!=0x01 && buffer[0]!=0x02 && buffer[0]!=0x1d) {
|
||||||
|
int Key=EncodeKey;
|
||||||
|
char *test=(char*)malloc(size);
|
||||||
|
int i;
|
||||||
|
buffer+=2;
|
||||||
|
size-=2;
|
||||||
|
for ( i = 0 ; i+4 <= size ; i+=4)
|
||||||
|
{
|
||||||
|
int pt = (*(int*)&buffer[i])^(Key);
|
||||||
|
Key = pt;
|
||||||
|
*(int*)&test[i]=pt;
|
||||||
|
}
|
||||||
|
unsigned char KC=Key&0xFF;
|
||||||
|
for ( ; i < size ; i++)
|
||||||
|
{
|
||||||
|
test[i]=buffer[i]^KC;
|
||||||
|
}
|
||||||
|
memcpy(buffer,test,size);
|
||||||
|
free(test);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
EQApplicationPacket *EQApplicationPacket::Copy() const {
|
EQApplicationPacket *EQApplicationPacket::Copy() const {
|
||||||
return(new EQApplicationPacket(*this));
|
return(new EQApplicationPacket(*this));
|
||||||
}
|
}
|
||||||
@@ -378,4 +515,4 @@ std::string DumpPacketToString(const EQApplicationPacket* app){
|
|||||||
std::ostringstream out;
|
std::ostringstream out;
|
||||||
out << DumpPacketHexToString(app->pBuffer, app->size);
|
out << DumpPacketHexToString(app->pBuffer, app->size);
|
||||||
return out.str();
|
return out.str();
|
||||||
}
|
}
|
||||||
@@ -80,8 +80,11 @@ public:
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
|
static bool ValidateCRC(const unsigned char *buffer, int length, uint32 Key);
|
||||||
static uint32 Decompress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize);
|
static uint32 Decompress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize);
|
||||||
static uint32 Compress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize);
|
static uint32 Compress(const unsigned char *buffer, const uint32 length, unsigned char *newbuf, uint32 newbufsize);
|
||||||
|
static void ChatDecode(unsigned char *buffer, int size, int DecodeKey);
|
||||||
|
static void ChatEncode(unsigned char *buffer, int size, int EncodeKey);
|
||||||
|
|
||||||
uint16 GetRawOpcode() const { return(opcode); }
|
uint16 GetRawOpcode() const { return(opcode); }
|
||||||
|
|
||||||
|
|||||||
+311
-1320
File diff suppressed because it is too large
Load Diff
@@ -23,7 +23,7 @@
|
|||||||
#include "op_codes.h"
|
#include "op_codes.h"
|
||||||
#include "crc16.h"
|
#include "crc16.h"
|
||||||
#include "platform.h"
|
#include "platform.h"
|
||||||
#include "strings.h"
|
#include "string_util.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
@@ -406,7 +406,7 @@ void EQStream::ProcessPacket(EQProtocolPacket *p)
|
|||||||
if(uint16(SequencedBase + SequencedQueue.size()) != NextOutSeq) {
|
if(uint16(SequencedBase + SequencedQueue.size()) != NextOutSeq) {
|
||||||
LogNetcode(_L "Pre-OOA Invalid Sequenced queue: BS [{}] + SQ [{}] != NOS [{}]" __L, SequencedBase, SequencedQueue.size(), NextOutSeq);
|
LogNetcode(_L "Pre-OOA Invalid Sequenced queue: BS [{}] + SQ [{}] != NOS [{}]" __L, SequencedBase, SequencedQueue.size(), NextOutSeq);
|
||||||
}
|
}
|
||||||
|
|
||||||
//if the packet they got out of order is between our last acked packet and the last sent packet, then its valid.
|
//if the packet they got out of order is between our last acked packet and the last sent packet, then its valid.
|
||||||
if (CompareSequence(SequencedBase,seq) != SeqPast && CompareSequence(NextOutSeq,seq) == SeqPast) {
|
if (CompareSequence(SequencedBase,seq) != SeqPast && CompareSequence(NextOutSeq,seq) == SeqPast) {
|
||||||
Log(Logs::Detail, Logs::Netcode, _L "Received OP_OutOfOrderAck for sequence %d, starting retransmit at the start of our unacked buffer (seq %d, was %d)." __L,
|
Log(Logs::Detail, Logs::Netcode, _L "Received OP_OutOfOrderAck for sequence %d, starting retransmit at the start of our unacked buffer (seq %d, was %d)." __L,
|
||||||
@@ -453,7 +453,7 @@ void EQStream::ProcessPacket(EQProtocolPacket *p)
|
|||||||
(unsigned long)ntohl(ClientStats->packets_received), (unsigned long)ntohl(ClientStats->packets_sent), (unsigned long)ntohl(ClientStats->last_local_delta),
|
(unsigned long)ntohl(ClientStats->packets_received), (unsigned long)ntohl(ClientStats->packets_sent), (unsigned long)ntohl(ClientStats->last_local_delta),
|
||||||
(unsigned long)ntohl(ClientStats->low_delta), (unsigned long)ntohl(ClientStats->average_delta),
|
(unsigned long)ntohl(ClientStats->low_delta), (unsigned long)ntohl(ClientStats->average_delta),
|
||||||
(unsigned long)ntohl(ClientStats->high_delta), (unsigned long)ntohl(ClientStats->last_remote_delta));
|
(unsigned long)ntohl(ClientStats->high_delta), (unsigned long)ntohl(ClientStats->last_remote_delta));
|
||||||
|
|
||||||
AdjustRates(ntohl(ClientStats->average_delta));
|
AdjustRates(ntohl(ClientStats->average_delta));
|
||||||
|
|
||||||
if(GetExecutablePlatform() == ExePlatformWorld || GetExecutablePlatform() == ExePlatformZone) {
|
if(GetExecutablePlatform() == ExePlatformWorld || GetExecutablePlatform() == ExePlatformZone) {
|
||||||
@@ -951,7 +951,7 @@ EQRawApplicationPacket *p=nullptr;
|
|||||||
EmuOpcode emu_op = (*OpMgr)->EQToEmu(p->opcode);
|
EmuOpcode emu_op = (*OpMgr)->EQToEmu(p->opcode);
|
||||||
if (emu_op == OP_Unknown) {
|
if (emu_op == OP_Unknown) {
|
||||||
// Log(Logs::General, Logs::Client_Server_Packet_Unhandled, "Unknown :: [%s - 0x%04x] [Size: %u] %s", OpcodeManager::EmuToName(p->GetOpcode()), p->opcode, p->Size(), DumpPacketToString(p).c_str());
|
// Log(Logs::General, Logs::Client_Server_Packet_Unhandled, "Unknown :: [%s - 0x%04x] [Size: %u] %s", OpcodeManager::EmuToName(p->GetOpcode()), p->opcode, p->Size(), DumpPacketToString(p).c_str());
|
||||||
}
|
}
|
||||||
p->SetOpcode(emu_op);
|
p->SetOpcode(emu_op);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1359,11 +1359,11 @@ void EQStream::AdjustRates(uint32 average_delta)
|
|||||||
DecayRate=DECAYBASE/average_delta;
|
DecayRate=DECAYBASE/average_delta;
|
||||||
if (BytesWritten > RateThreshold)
|
if (BytesWritten > RateThreshold)
|
||||||
BytesWritten = RateThreshold + DecayRate;
|
BytesWritten = RateThreshold + DecayRate;
|
||||||
Log(Logs::Detail, Logs::Netcode, _L "Adjusting data rate to thresh %d, decay %d based on avg delta %d" __L,
|
Log(Logs::Detail, Logs::Netcode, _L "Adjusting data rate to thresh %d, decay %d based on avg delta %d" __L,
|
||||||
RateThreshold, DecayRate, average_delta);
|
RateThreshold, DecayRate, average_delta);
|
||||||
MRate.unlock();
|
MRate.unlock();
|
||||||
} else {
|
} else {
|
||||||
Log(Logs::Detail, Logs::Netcode, _L "Not adjusting data rate because avg delta over max (%d > %d)" __L,
|
Log(Logs::Detail, Logs::Netcode, _L "Not adjusting data rate because avg delta over max (%d > %d)" __L,
|
||||||
average_delta, AVERAGE_DELTA_MAX);
|
average_delta, AVERAGE_DELTA_MAX);
|
||||||
AverageDelta = AVERAGE_DELTA_MAX;
|
AverageDelta = AVERAGE_DELTA_MAX;
|
||||||
}
|
}
|
||||||
@@ -1374,7 +1374,7 @@ void EQStream::AdjustRates(uint32 average_delta)
|
|||||||
BytesWritten = 0;
|
BytesWritten = 0;
|
||||||
RateThreshold=RATEBASE/average_delta;
|
RateThreshold=RATEBASE/average_delta;
|
||||||
DecayRate=DECAYBASE/average_delta;
|
DecayRate=DECAYBASE/average_delta;
|
||||||
Log(Logs::Detail, Logs::Netcode, _L "Adjusting data rate to thresh %d, decay %d based on avg delta %d" __L,
|
Log(Logs::Detail, Logs::Netcode, _L "Adjusting data rate to thresh %d, decay %d based on avg delta %d" __L,
|
||||||
RateThreshold, DecayRate, average_delta);
|
RateThreshold, DecayRate, average_delta);
|
||||||
MRate.unlock();
|
MRate.unlock();
|
||||||
}
|
}
|
||||||
|
|||||||
+6
-11
@@ -218,13 +218,13 @@ class EQStream : public EQStreamInterface {
|
|||||||
|
|
||||||
void init(bool resetSession=true);
|
void init(bool resetSession=true);
|
||||||
public:
|
public:
|
||||||
EQStream() { init(); remote_ip = 0; remote_port = 0; State = UNESTABLISHED;
|
EQStream() { init(); remote_ip = 0; remote_port = 0; State = UNESTABLISHED;
|
||||||
StreamType = UnknownStream; compressed = true; encoded = false; app_opcode_size = 2;
|
StreamType = UnknownStream; compressed = true; encoded = false; app_opcode_size = 2;
|
||||||
bytes_sent = 0; bytes_recv = 0; create_time = Timer::GetTimeSeconds(); sessionAttempts = 0;
|
bytes_sent = 0; bytes_recv = 0; create_time = Timer::GetTimeSeconds(); sessionAttempts = 0;
|
||||||
streamactive = false; }
|
streamactive = false; }
|
||||||
EQStream(sockaddr_in addr) { init(); remote_ip = addr.sin_addr.s_addr;
|
EQStream(sockaddr_in addr) { init(); remote_ip = addr.sin_addr.s_addr;
|
||||||
remote_port = addr.sin_port; State = UNESTABLISHED; StreamType = UnknownStream;
|
remote_port = addr.sin_port; State = UNESTABLISHED; StreamType = UnknownStream;
|
||||||
compressed = true; encoded = false; app_opcode_size = 2; bytes_sent = 0; bytes_recv = 0;
|
compressed = true; encoded = false; app_opcode_size = 2; bytes_sent = 0; bytes_recv = 0;
|
||||||
create_time = Timer::GetTimeSeconds(); }
|
create_time = Timer::GetTimeSeconds(); }
|
||||||
virtual ~EQStream() { RemoveData(); SetState(CLOSED); }
|
virtual ~EQStream() { RemoveData(); SetState(CLOSED); }
|
||||||
void SetMaxLen(uint32 length) { MaxLen=length; }
|
void SetMaxLen(uint32 length) { MaxLen=length; }
|
||||||
@@ -243,11 +243,6 @@ class EQStream : public EQStreamInterface {
|
|||||||
|
|
||||||
virtual void SetOpcodeManager(OpcodeManager **opm) { OpMgr = opm; }
|
virtual void SetOpcodeManager(OpcodeManager **opm) { OpMgr = opm; }
|
||||||
|
|
||||||
virtual OpcodeManager* GetOpcodeManager() const
|
|
||||||
{
|
|
||||||
return (*OpMgr);
|
|
||||||
};
|
|
||||||
|
|
||||||
void CheckTimeout(uint32 now, uint32 timeout=30);
|
void CheckTimeout(uint32 now, uint32 timeout=30);
|
||||||
bool HasOutgoingData();
|
bool HasOutgoingData();
|
||||||
void Process(const unsigned char *data, const uint32 length);
|
void Process(const unsigned char *data, const uint32 length);
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ EQStreamIdentifier::~EQStreamIdentifier() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void EQStreamIdentifier::RegisterPatch(EQStreamInterface::Signature sig, const char *name, OpcodeManager ** opcodes, const StructStrategy *structs) {
|
void EQStreamIdentifier::RegisterPatch(const EQStreamInterface::Signature &sig, const char *name, OpcodeManager ** opcodes, const StructStrategy *structs) {
|
||||||
auto p = new Patch;
|
auto p = new Patch;
|
||||||
p->signature = sig;
|
p->signature = sig;
|
||||||
p->name = name;
|
p->name = name;
|
||||||
@@ -145,7 +145,7 @@ void EQStreamIdentifier::Process() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void EQStreamIdentifier::AddStream(std::shared_ptr<EQStreamInterface> eqs) {
|
void EQStreamIdentifier::AddStream(std::shared_ptr<EQStreamInterface> eqs) {
|
||||||
m_streams.emplace_back(Record(eqs));
|
m_streams.push_back(Record(eqs));
|
||||||
eqs = nullptr;
|
eqs = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ public:
|
|||||||
~EQStreamIdentifier();
|
~EQStreamIdentifier();
|
||||||
|
|
||||||
//registration interface.
|
//registration interface.
|
||||||
void RegisterPatch(EQStreamInterface::Signature sig, const char *name, OpcodeManager ** opcodes, const StructStrategy *structs);
|
void RegisterPatch(const EQStreamInterface::Signature &sig, const char *name, OpcodeManager ** opcodes, const StructStrategy *structs);
|
||||||
|
|
||||||
//main processing interface
|
//main processing interface
|
||||||
void Process();
|
void Process();
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ struct EQStreamManagerInterfaceOptions
|
|||||||
|
|
||||||
//World seems to support both compression and xor zone supports one or the others.
|
//World seems to support both compression and xor zone supports one or the others.
|
||||||
//Enforce one or the other in the convienence construct
|
//Enforce one or the other in the convienence construct
|
||||||
//Login I had trouble getting to recognize compression at all
|
//Login I had trouble getting to recognize compression at all
|
||||||
//but that might be because it was still a bit buggy when i was testing that.
|
//but that might be because it was still a bit buggy when i was testing that.
|
||||||
if (compressed) {
|
if (compressed) {
|
||||||
daybreak_options.encode_passes[0] = EQ::Net::EncodeCompression;
|
daybreak_options.encode_passes[0] = EQ::Net::EncodeCompression;
|
||||||
@@ -100,7 +100,6 @@ public:
|
|||||||
virtual MatchState CheckSignature(const Signature *sig) { return MatchFailed; }
|
virtual MatchState CheckSignature(const Signature *sig) { return MatchFailed; }
|
||||||
virtual EQStreamState GetState() = 0;
|
virtual EQStreamState GetState() = 0;
|
||||||
virtual void SetOpcodeManager(OpcodeManager **opm) = 0;
|
virtual void SetOpcodeManager(OpcodeManager **opm) = 0;
|
||||||
virtual OpcodeManager* GetOpcodeManager() const = 0;
|
|
||||||
virtual const EQ::versions::ClientVersion ClientVersion() const { return EQ::versions::ClientVersion::Unknown; }
|
virtual const EQ::versions::ClientVersion ClientVersion() const { return EQ::versions::ClientVersion::Unknown; }
|
||||||
virtual Stats GetStats() const = 0;
|
virtual Stats GetStats() const = 0;
|
||||||
virtual void ResetStats() = 0;
|
virtual void ResetStats() = 0;
|
||||||
|
|||||||
@@ -38,8 +38,12 @@ void EQStreamProxy::SetOpcodeManager(OpcodeManager **opm)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void EQStreamProxy::QueuePacket(const EQApplicationPacket *p, bool ack_req) {
|
void EQStreamProxy::QueuePacket(const EQApplicationPacket *p, bool ack_req) {
|
||||||
if (p == nullptr) {
|
if(p == nullptr)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (p->GetOpcode() != OP_SpecialMesg) {
|
||||||
|
Log(Logs::General, Logs::PacketServerClient, "[%s - 0x%04x] [Size: %u]", OpcodeManager::EmuToName(p->GetOpcode()), p->GetOpcode(), p->Size());
|
||||||
|
Log(Logs::General, Logs::PacketServerClientWithDump, "[%s - 0x%04x] [Size: %u] %s", OpcodeManager::EmuToName(p->GetOpcode()), p->GetOpcode(), p->Size(), DumpPacketToString(p).c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
EQApplicationPacket *newp = p->Copy();
|
EQApplicationPacket *newp = p->Copy();
|
||||||
@@ -108,8 +112,3 @@ bool EQStreamProxy::CheckState(EQStreamState state) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
OpcodeManager *EQStreamProxy::GetOpcodeManager() const
|
|
||||||
{
|
|
||||||
return (*m_opcodes);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|||||||
@@ -34,15 +34,13 @@ public:
|
|||||||
virtual Stats GetStats() const;
|
virtual Stats GetStats() const;
|
||||||
virtual void ResetStats();
|
virtual void ResetStats();
|
||||||
virtual EQStreamManagerInterface* GetManager() const;
|
virtual EQStreamManagerInterface* GetManager() const;
|
||||||
virtual OpcodeManager* GetOpcodeManager() const;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::shared_ptr<EQStreamInterface> const m_stream; //we own this stream object.
|
std::shared_ptr<EQStreamInterface> const m_stream; //we own this stream object.
|
||||||
const StructStrategy *const m_structs; //we do not own this object.
|
const StructStrategy *const m_structs; //we do not own this object.
|
||||||
//this is a pointer to a pointer to make it less likely that a packet will
|
//this is a pointer to a pointer to make it less likely that a packet will
|
||||||
//reference an invalid opcode manager when they are being reloaded.
|
//reference an invalid opcode manager when they are being reloaded.
|
||||||
OpcodeManager **const m_opcodes;
|
OpcodeManager **const m_opcodes; //we do not own this object.
|
||||||
//we do not own this object.
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /*EQSTREAMPROXY_H_*/
|
#endif /*EQSTREAMPROXY_H_*/
|
||||||
|
|||||||
@@ -23,6 +23,9 @@
|
|||||||
|
|
||||||
EQDB EQDB::s_EQDB;
|
EQDB EQDB::s_EQDB;
|
||||||
|
|
||||||
|
EQDB::EQDB() {
|
||||||
|
}
|
||||||
|
|
||||||
unsigned int EQDB::field_count() {
|
unsigned int EQDB::field_count() {
|
||||||
return mysql_field_count(mysql_ref);
|
return mysql_field_count(mysql_ref);
|
||||||
}
|
}
|
||||||
|
|||||||
+1
-1
@@ -27,7 +27,7 @@
|
|||||||
|
|
||||||
//this is the main object exported to perl.
|
//this is the main object exported to perl.
|
||||||
class EQDB {
|
class EQDB {
|
||||||
EQDB() = default;
|
EQDB();
|
||||||
public:
|
public:
|
||||||
static EQDB *Singleton() { return(&s_EQDB); }
|
static EQDB *Singleton() { return(&s_EQDB); }
|
||||||
|
|
||||||
|
|||||||
+33
-153
@@ -19,13 +19,9 @@
|
|||||||
#include "../common/global_define.h"
|
#include "../common/global_define.h"
|
||||||
#include "eqemu_config.h"
|
#include "eqemu_config.h"
|
||||||
#include "misc_functions.h"
|
#include "misc_functions.h"
|
||||||
#include "strings.h"
|
|
||||||
#include "eqemu_logsys.h"
|
|
||||||
#include "json/json.hpp"
|
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <filesystem>
|
|
||||||
|
|
||||||
std::string EQEmuConfig::ConfigFile = "eqemu_config.json";
|
std::string EQEmuConfig::ConfigFile = "eqemu_config.json";
|
||||||
EQEmuConfig *EQEmuConfig::_config = nullptr;
|
EQEmuConfig *EQEmuConfig::_config = nullptr;
|
||||||
@@ -37,23 +33,17 @@ void EQEmuConfig::parse_config()
|
|||||||
LongName = _root["server"]["world"].get("longname", "").asString();
|
LongName = _root["server"]["world"].get("longname", "").asString();
|
||||||
WorldAddress = _root["server"]["world"].get("address", "").asString();
|
WorldAddress = _root["server"]["world"].get("address", "").asString();
|
||||||
LocalAddress = _root["server"]["world"].get("localaddress", "").asString();
|
LocalAddress = _root["server"]["world"].get("localaddress", "").asString();
|
||||||
MaxClients = Strings::ToInt(_root["server"]["world"].get("maxclients", "-1").asString());
|
MaxClients = atoi(_root["server"]["world"].get("maxclients", "-1").asString().c_str());
|
||||||
SharedKey = _root["server"]["world"].get("key", "").asString();
|
SharedKey = _root["server"]["world"].get("key", "").asString();
|
||||||
LoginCount = 0;
|
LoginCount = 0;
|
||||||
|
|
||||||
if (_root["server"]["world"]["loginserver"].isObject()) {
|
if (_root["server"]["world"]["loginserver"].isObject()) {
|
||||||
LoginHost = _root["server"]["world"]["loginserver"].get("host", "login.eqemulator.net").asString();
|
LoginHost = _root["server"]["world"]["loginserver"].get("host", "login.eqemulator.net").asString();
|
||||||
LoginPort = Strings::ToUnsignedInt(_root["server"]["world"]["loginserver"].get("port", "5998").asString());
|
LoginPort = atoi(_root["server"]["world"]["loginserver"].get("port", "5998").asString().c_str());
|
||||||
LoginLegacy = false;
|
LoginLegacy = false;
|
||||||
if (_root["server"]["world"]["loginserver"].get("legacy", "0").asString() == "1") { LoginLegacy = true; }
|
if (_root["server"]["world"]["loginserver"].get("legacy", "0").asString() == "1") { LoginLegacy = true; }
|
||||||
LoginAccount = _root["server"]["world"]["loginserver"].get("account", "").asString();
|
LoginAccount = _root["server"]["world"]["loginserver"].get("account", "").asString();
|
||||||
LoginPassword = _root["server"]["world"]["loginserver"].get("password", "").asString();
|
LoginPassword = _root["server"]["world"]["loginserver"].get("password", "").asString();
|
||||||
|
|
||||||
// at least today, this is wrong a majority of the time
|
|
||||||
// remove this if eqemulator ever upgrades its loginserver
|
|
||||||
if (LoginHost.find("login.eqemulator.net") != std::string::npos) {
|
|
||||||
LoginLegacy = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
char str[32];
|
char str[32];
|
||||||
@@ -66,60 +56,44 @@ void EQEmuConfig::parse_config()
|
|||||||
|
|
||||||
auto loginconfig = new LoginConfig;
|
auto loginconfig = new LoginConfig;
|
||||||
loginconfig->LoginHost = _root["server"]["world"][str].get("host", "login.eqemulator.net").asString();
|
loginconfig->LoginHost = _root["server"]["world"][str].get("host", "login.eqemulator.net").asString();
|
||||||
loginconfig->LoginPort = Strings::ToUnsignedInt(_root["server"]["world"][str].get("port", "5998").asString());
|
loginconfig->LoginPort = atoi(_root["server"]["world"][str].get("port", "5998").asString().c_str());
|
||||||
loginconfig->LoginAccount = _root["server"]["world"][str].get("account", "").asString();
|
loginconfig->LoginAccount = _root["server"]["world"][str].get("account", "").asString();
|
||||||
loginconfig->LoginPassword = _root["server"]["world"][str].get("password", "").asString();
|
loginconfig->LoginPassword = _root["server"]["world"][str].get("password", "").asString();
|
||||||
|
|
||||||
loginconfig->LoginLegacy = false;
|
loginconfig->LoginLegacy = false;
|
||||||
if (_root["server"]["world"][str].get("legacy", "0").asString() == "1") { loginconfig->LoginLegacy = true; }
|
if (_root["server"]["world"][str].get("legacy", "0").asString() == "1") { loginconfig->LoginLegacy = true; }
|
||||||
|
|
||||||
// at least today, this is wrong a majority of the time
|
|
||||||
// remove this if eqemulator ever upgrades its loginserver
|
|
||||||
if (loginconfig->LoginHost.find("login.eqemulator.net") != std::string::npos) {
|
|
||||||
loginconfig->LoginLegacy = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
loginlist.Insert(loginconfig);
|
loginlist.Insert(loginconfig);
|
||||||
} while (LoginCount < 100);
|
} while (LoginCount < 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//<locked> from xml converts to json as locked: "", so i default to "false".
|
//<locked> from xml converts to json as locked: "", so i default to "false".
|
||||||
//The only way to enable locked is by switching to true, meaning this value is always false until manually set true
|
//The only way to enable locked is by switching to true, meaning this value is always false until manually set true
|
||||||
Locked = false;
|
Locked = false;
|
||||||
if (_root["server"]["world"].get("locked", "false").asString() == "true") { Locked = true; }
|
if (_root["server"]["world"].get("locked", "false").asString() == "true") { Locked = true; }
|
||||||
|
WorldIP = _root["server"]["world"]["tcp"].get("host", "127.0.0.1").asString();
|
||||||
auto_database_updates = false;
|
WorldTCPPort = atoi(_root["server"]["world"]["tcp"].get("port", "9000").asString().c_str());
|
||||||
if (_root["server"].get("auto_database_updates", "true").asString() == "true") {
|
|
||||||
auto_database_updates = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
WorldIP = _root["server"]["world"]["tcp"].get("ip", "127.0.0.1").asString();
|
|
||||||
WorldTCPPort = Strings::ToUnsignedInt(_root["server"]["world"]["tcp"].get("port", "9000").asString());
|
|
||||||
|
|
||||||
TelnetIP = _root["server"]["world"]["telnet"].get("ip", "127.0.0.1").asString();
|
TelnetIP = _root["server"]["world"]["telnet"].get("ip", "127.0.0.1").asString();
|
||||||
TelnetTCPPort = Strings::ToUnsignedInt(_root["server"]["world"]["telnet"].get("port", "9001").asString());
|
TelnetTCPPort = atoi(_root["server"]["world"]["telnet"].get("port", "9001").asString().c_str());
|
||||||
TelnetEnabled = false;
|
TelnetEnabled = false;
|
||||||
if (_root["server"]["world"]["telnet"].get("enabled", "false").asString() == "true") { TelnetEnabled = true; }
|
if (_root["server"]["world"]["telnet"].get("enabled", "false").asString() == "true") { TelnetEnabled = true; }
|
||||||
|
|
||||||
WorldHTTPMimeFile = _root["server"]["world"]["http"].get("mimefile", "mime.types").asString();
|
WorldHTTPMimeFile = _root["server"]["world"]["http"].get("mimefile", "mime.types").asString();
|
||||||
WorldHTTPPort = Strings::ToUnsignedInt(_root["server"]["world"]["http"].get("port", "9080").asString());
|
WorldHTTPPort = atoi(_root["server"]["world"]["http"].get("port", "9080").asString().c_str());
|
||||||
WorldHTTPEnabled = false;
|
WorldHTTPEnabled = false;
|
||||||
|
|
||||||
if (_root["server"]["world"]["http"].get("enabled", "false").asString() == "true") {
|
if (_root["server"]["world"]["http"].get("enabled", "false").asString() == "true") {
|
||||||
WorldHTTPEnabled = true;
|
WorldHTTPEnabled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_root["server"].get("disable_config_checks", "false").asString() == "true") {
|
/**
|
||||||
DisableConfigChecks = true;
|
* UCS
|
||||||
}
|
*/
|
||||||
|
ChatHost = _root["server"]["chatserver"].get("host", "eqchat.eqemulator.net").asString();
|
||||||
|
ChatPort = atoi(_root["server"]["chatserver"].get("port", "7778").asString().c_str());
|
||||||
|
MailHost = _root["server"]["mailserver"].get("host", "eqmail.eqemulator.net").asString();
|
||||||
CheckUcsConfigConversion();
|
MailPort = atoi(_root["server"]["mailserver"].get("port", "7778").asString().c_str());
|
||||||
|
|
||||||
m_ucs_host = _root["server"]["ucs"].get("host", "eqchat.eqemulator.net").asString();
|
|
||||||
m_ucs_port = Strings::ToUnsignedInt(_root["server"]["ucs"].get("port", "7778").asString());
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Database
|
* Database
|
||||||
@@ -127,7 +101,7 @@ void EQEmuConfig::parse_config()
|
|||||||
DatabaseUsername = _root["server"]["database"].get("username", "eq").asString();
|
DatabaseUsername = _root["server"]["database"].get("username", "eq").asString();
|
||||||
DatabasePassword = _root["server"]["database"].get("password", "eq").asString();
|
DatabasePassword = _root["server"]["database"].get("password", "eq").asString();
|
||||||
DatabaseHost = _root["server"]["database"].get("host", "localhost").asString();
|
DatabaseHost = _root["server"]["database"].get("host", "localhost").asString();
|
||||||
DatabasePort = Strings::ToUnsignedInt(_root["server"]["database"].get("port", "3306").asString());
|
DatabasePort = atoi(_root["server"]["database"].get("port", "3306").asString().c_str());
|
||||||
DatabaseDB = _root["server"]["database"].get("db", "eq").asString();
|
DatabaseDB = _root["server"]["database"].get("db", "eq").asString();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -136,14 +110,14 @@ void EQEmuConfig::parse_config()
|
|||||||
ContentDbUsername = _root["server"]["content_database"].get("username", "").asString();
|
ContentDbUsername = _root["server"]["content_database"].get("username", "").asString();
|
||||||
ContentDbPassword = _root["server"]["content_database"].get("password", "").asString();
|
ContentDbPassword = _root["server"]["content_database"].get("password", "").asString();
|
||||||
ContentDbHost = _root["server"]["content_database"].get("host", "").asString();
|
ContentDbHost = _root["server"]["content_database"].get("host", "").asString();
|
||||||
ContentDbPort = Strings::ToUnsignedInt(_root["server"]["content_database"].get("port", 0).asString());
|
ContentDbPort = atoi(_root["server"]["content_database"].get("port", 0).asString().c_str());
|
||||||
ContentDbName = _root["server"]["content_database"].get("db", "").asString();
|
ContentDbName = _root["server"]["content_database"].get("db", "").asString();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* QS
|
* QS
|
||||||
*/
|
*/
|
||||||
QSDatabaseHost = _root["server"]["qsdatabase"].get("host", "localhost").asString();
|
QSDatabaseHost = _root["server"]["qsdatabase"].get("host", "localhost").asString();
|
||||||
QSDatabasePort = Strings::ToUnsignedInt(_root["server"]["qsdatabase"].get("port", "3306").asString());
|
QSDatabasePort = atoi(_root["server"]["qsdatabase"].get("port", "3306").asString().c_str());
|
||||||
QSDatabaseUsername = _root["server"]["qsdatabase"].get("username", "eq").asString();
|
QSDatabaseUsername = _root["server"]["qsdatabase"].get("username", "eq").asString();
|
||||||
QSDatabasePassword = _root["server"]["qsdatabase"].get("password", "eq").asString();
|
QSDatabasePassword = _root["server"]["qsdatabase"].get("password", "eq").asString();
|
||||||
QSDatabaseDB = _root["server"]["qsdatabase"].get("db", "eq").asString();
|
QSDatabaseDB = _root["server"]["qsdatabase"].get("db", "eq").asString();
|
||||||
@@ -151,9 +125,9 @@ void EQEmuConfig::parse_config()
|
|||||||
/**
|
/**
|
||||||
* Zones
|
* Zones
|
||||||
*/
|
*/
|
||||||
DefaultStatus = Strings::ToUnsignedInt(_root["server"]["zones"].get("defaultstatus", 0).asString());
|
DefaultStatus = atoi(_root["server"]["zones"].get("defaultstatus", 0).asString().c_str());
|
||||||
ZonePortLow = Strings::ToUnsignedInt(_root["server"]["zones"]["ports"].get("low", "7000").asString());
|
ZonePortLow = atoi(_root["server"]["zones"]["ports"].get("low", "7000").asString().c_str());
|
||||||
ZonePortHigh = Strings::ToUnsignedInt(_root["server"]["zones"]["ports"].get("high", "7999").asString());
|
ZonePortHigh = atoi(_root["server"]["zones"]["ports"].get("high", "7999").asString().c_str());
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Files
|
* Files
|
||||||
@@ -171,7 +145,6 @@ void EQEmuConfig::parse_config()
|
|||||||
PluginDir = _root["server"]["directories"].get("plugins", "plugins/").asString();
|
PluginDir = _root["server"]["directories"].get("plugins", "plugins/").asString();
|
||||||
LuaModuleDir = _root["server"]["directories"].get("lua_modules", "lua_modules/").asString();
|
LuaModuleDir = _root["server"]["directories"].get("lua_modules", "lua_modules/").asString();
|
||||||
PatchDir = _root["server"]["directories"].get("patches", "./").asString();
|
PatchDir = _root["server"]["directories"].get("patches", "./").asString();
|
||||||
OpcodeDir = _root["server"]["directories"].get("opcodes", "./").asString();
|
|
||||||
SharedMemDir = _root["server"]["directories"].get("shared_memory", "shared/").asString();
|
SharedMemDir = _root["server"]["directories"].get("shared_memory", "shared/").asString();
|
||||||
LogDir = _root["server"]["directories"].get("logs", "logs/").asString();
|
LogDir = _root["server"]["directories"].get("logs", "logs/").asString();
|
||||||
|
|
||||||
@@ -184,10 +157,10 @@ void EQEmuConfig::parse_config()
|
|||||||
/**
|
/**
|
||||||
* Launcher
|
* Launcher
|
||||||
*/
|
*/
|
||||||
RestartWait = Strings::ToInt(_root["server"]["launcher"]["timers"].get("restart", "10000").asString());
|
RestartWait = atoi(_root["server"]["launcher"]["timers"].get("restart", "10000").asString().c_str());
|
||||||
TerminateWait = Strings::ToInt(_root["server"]["launcher"]["timers"].get("reterminate", "10000").asString());
|
TerminateWait = atoi(_root["server"]["launcher"]["timers"].get("reterminate", "10000").asString().c_str());
|
||||||
InitialBootWait = Strings::ToInt(_root["server"]["launcher"]["timers"].get("initial", "20000").asString());
|
InitialBootWait = atoi(_root["server"]["launcher"]["timers"].get("initial", "20000").asString().c_str());
|
||||||
ZoneBootInterval = Strings::ToInt(_root["server"]["launcher"]["timers"].get("interval", "2000").asString());
|
ZoneBootInterval = atoi(_root["server"]["launcher"]["timers"].get("interval", "2000").asString().c_str());
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
ZoneExe = _root["server"]["launcher"].get("exe", "zone.exe").asString();
|
ZoneExe = _root["server"]["launcher"].get("exe", "zone.exe").asString();
|
||||||
#else
|
#else
|
||||||
@@ -249,16 +222,16 @@ std::string EQEmuConfig::GetByName(const std::string &var_name) const
|
|||||||
return (WorldHTTPEnabled ? "true" : "false");
|
return (WorldHTTPEnabled ? "true" : "false");
|
||||||
}
|
}
|
||||||
if (var_name == "ChatHost") {
|
if (var_name == "ChatHost") {
|
||||||
return (m_ucs_host);
|
return (ChatHost);
|
||||||
}
|
}
|
||||||
if (var_name == "ChatPort") {
|
if (var_name == "ChatPort") {
|
||||||
return (itoa(m_ucs_port));
|
return (itoa(ChatPort));
|
||||||
}
|
}
|
||||||
if (var_name == "MailHost") {
|
if (var_name == "MailHost") {
|
||||||
return (m_ucs_host);
|
return (MailHost);
|
||||||
}
|
}
|
||||||
if (var_name == "MailPort") {
|
if (var_name == "MailPort") {
|
||||||
return (itoa(m_ucs_port));
|
return (itoa(MailPort));
|
||||||
}
|
}
|
||||||
if (var_name == "DatabaseHost") {
|
if (var_name == "DatabaseHost") {
|
||||||
return (DatabaseHost);
|
return (DatabaseHost);
|
||||||
@@ -365,8 +338,10 @@ void EQEmuConfig::Dump() const
|
|||||||
std::cout << "WorldHTTPPort = " << WorldHTTPPort << std::endl;
|
std::cout << "WorldHTTPPort = " << WorldHTTPPort << std::endl;
|
||||||
std::cout << "WorldHTTPMimeFile = " << WorldHTTPMimeFile << std::endl;
|
std::cout << "WorldHTTPMimeFile = " << WorldHTTPMimeFile << std::endl;
|
||||||
std::cout << "WorldHTTPEnabled = " << WorldHTTPEnabled << std::endl;
|
std::cout << "WorldHTTPEnabled = " << WorldHTTPEnabled << std::endl;
|
||||||
std::cout << "UCSHost = " << m_ucs_host << std::endl;
|
std::cout << "ChatHost = " << ChatHost << std::endl;
|
||||||
std::cout << "UCSPort = " << m_ucs_port << std::endl;
|
std::cout << "ChatPort = " << ChatPort << std::endl;
|
||||||
|
std::cout << "MailHost = " << MailHost << std::endl;
|
||||||
|
std::cout << "MailPort = " << MailPort << std::endl;
|
||||||
std::cout << "DatabaseHost = " << DatabaseHost << std::endl;
|
std::cout << "DatabaseHost = " << DatabaseHost << std::endl;
|
||||||
std::cout << "DatabaseUsername = " << DatabaseUsername << std::endl;
|
std::cout << "DatabaseUsername = " << DatabaseUsername << std::endl;
|
||||||
std::cout << "DatabasePassword = " << DatabasePassword << std::endl;
|
std::cout << "DatabasePassword = " << DatabasePassword << std::endl;
|
||||||
@@ -393,98 +368,3 @@ void EQEmuConfig::Dump() const
|
|||||||
std::cout << "DefaultStatus = " << (int) DefaultStatus << std::endl;
|
std::cout << "DefaultStatus = " << (int) DefaultStatus << std::endl;
|
||||||
// std::cout << "DynamicCount = " << DynamicCount << std::endl;
|
// std::cout << "DynamicCount = " << DynamicCount << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string &EQEmuConfig::GetUCSHost() const
|
|
||||||
{
|
|
||||||
return m_ucs_host;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16 EQEmuConfig::GetUCSPort() const
|
|
||||||
{
|
|
||||||
return m_ucs_port;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EQEmuConfig::CheckUcsConfigConversion()
|
|
||||||
{
|
|
||||||
std::string chat_host = _root["server"]["chatserver"].get("host", "").asString();
|
|
||||||
uint32 chat_port = Strings::ToUnsignedInt(_root["server"]["chatserver"].get("port", "0").asString());
|
|
||||||
std::string mail_host = _root["server"]["mailserver"].get("host", "").asString();
|
|
||||||
uint32 mail_port = Strings::ToUnsignedInt(_root["server"]["mailserver"].get("port", "0").asString());
|
|
||||||
std::string ucs_host = _root["server"]["ucs"].get("host", "").asString();
|
|
||||||
|
|
||||||
// automatic ucs legacy configuration migration
|
|
||||||
// if old configuration values are set, let's backup the existing configuration
|
|
||||||
// and migrate to to use the new fields and write the new config
|
|
||||||
if ((!chat_host.empty() || !mail_host.empty()) && ucs_host.empty()) {
|
|
||||||
LogInfo("Migrating old [eqemu_config] UCS configuration to new configuration");
|
|
||||||
|
|
||||||
std::string config_file_path = std::filesystem::path{
|
|
||||||
path.GetServerPath() + "/eqemu_config.json"
|
|
||||||
}.string();
|
|
||||||
|
|
||||||
std::string config_file_bak_path = std::filesystem::path{
|
|
||||||
path.GetServerPath() + "/eqemu_config.ucs-migrate-json.bak"
|
|
||||||
}.string();
|
|
||||||
|
|
||||||
// copy eqemu_config.json to eqemu_config.json.bak
|
|
||||||
std::ifstream src(config_file_path, std::ios::binary);
|
|
||||||
std::ofstream dst(config_file_bak_path, std::ios::binary);
|
|
||||||
dst << src.rdbuf();
|
|
||||||
src.close();
|
|
||||||
|
|
||||||
LogInfo("Old configuration backed up to [{}]", config_file_bak_path);
|
|
||||||
|
|
||||||
// read eqemu_config.json, transplant new fields and write to eqemu_config.json
|
|
||||||
Json::Value root;
|
|
||||||
Json::Reader reader;
|
|
||||||
std::ifstream file(config_file_path);
|
|
||||||
if (!reader.parse(file, root)) {
|
|
||||||
LogError("Failed to parse configuration file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
|
|
||||||
// get old fields
|
|
||||||
std::string host = !chat_host.empty() ? chat_host : mail_host;
|
|
||||||
if (host.empty()) {
|
|
||||||
host = "eqchat.eqemulator.net";
|
|
||||||
}
|
|
||||||
std::string port = chat_port > 0 ? std::to_string(chat_port) : std::to_string(mail_port);
|
|
||||||
if (port.empty()) {
|
|
||||||
port = "7778";
|
|
||||||
}
|
|
||||||
|
|
||||||
// set new fields
|
|
||||||
root["server"]["ucs"]["host"] = host;
|
|
||||||
root["server"]["ucs"]["port"] = port;
|
|
||||||
|
|
||||||
// unset old fields
|
|
||||||
root["server"].removeMember("chatserver");
|
|
||||||
root["server"].removeMember("mailserver");
|
|
||||||
|
|
||||||
// get Json::Value raw string
|
|
||||||
std::string config = root.toStyledString();
|
|
||||||
|
|
||||||
// format using more modern json library
|
|
||||||
nlohmann::json data = nlohmann::json::parse(config);
|
|
||||||
|
|
||||||
// write to file
|
|
||||||
std::ofstream o(config_file_path);
|
|
||||||
o << std::setw(1) << data << std::endl;
|
|
||||||
o.close();
|
|
||||||
|
|
||||||
// write new config
|
|
||||||
LogInfo("New configuration written to [{}]", config_file_path);
|
|
||||||
LogInfo("Migration complete, please review the new configuration file");
|
|
||||||
|
|
||||||
// reload config internally
|
|
||||||
try {
|
|
||||||
std::ifstream fconfig(config_file_path, std::ifstream::binary);
|
|
||||||
fconfig >> _config->_root;
|
|
||||||
_config->parse_config();
|
|
||||||
}
|
|
||||||
catch (std::exception &) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
+22
-29
@@ -20,9 +20,7 @@
|
|||||||
|
|
||||||
#include "json/json.h"
|
#include "json/json.h"
|
||||||
#include "linked_list.h"
|
#include "linked_list.h"
|
||||||
#include "path_manager.h"
|
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <fmt/format.h>
|
|
||||||
|
|
||||||
struct LoginConfig {
|
struct LoginConfig {
|
||||||
std::string LoginHost;
|
std::string LoginHost;
|
||||||
@@ -60,7 +58,14 @@ class EQEmuConfig
|
|||||||
uint16 WorldHTTPPort;
|
uint16 WorldHTTPPort;
|
||||||
std::string WorldHTTPMimeFile;
|
std::string WorldHTTPMimeFile;
|
||||||
std::string SharedKey;
|
std::string SharedKey;
|
||||||
bool DisableConfigChecks;
|
|
||||||
|
// From <chatserver/>
|
||||||
|
std::string ChatHost;
|
||||||
|
uint16 ChatPort;
|
||||||
|
|
||||||
|
// From <mailserver/>
|
||||||
|
std::string MailHost;
|
||||||
|
uint16 MailPort;
|
||||||
|
|
||||||
// From <database/>
|
// From <database/>
|
||||||
std::string DatabaseHost;
|
std::string DatabaseHost;
|
||||||
@@ -95,7 +100,6 @@ class EQEmuConfig
|
|||||||
std::string PluginDir;
|
std::string PluginDir;
|
||||||
std::string LuaModuleDir;
|
std::string LuaModuleDir;
|
||||||
std::string PatchDir;
|
std::string PatchDir;
|
||||||
std::string OpcodeDir;
|
|
||||||
std::string SharedMemDir;
|
std::string SharedMemDir;
|
||||||
std::string LogDir;
|
std::string LogDir;
|
||||||
|
|
||||||
@@ -113,20 +117,12 @@ class EQEmuConfig
|
|||||||
uint16 ZonePortHigh;
|
uint16 ZonePortHigh;
|
||||||
uint8 DefaultStatus;
|
uint8 DefaultStatus;
|
||||||
|
|
||||||
bool auto_database_updates;
|
|
||||||
|
|
||||||
const std::string &GetUCSHost() const;
|
|
||||||
uint16 GetUCSPort() const;
|
|
||||||
|
|
||||||
// uint16 DynamicCount;
|
// uint16 DynamicCount;
|
||||||
|
|
||||||
// map<string,uint16> StaticZones;
|
// map<string,uint16> StaticZones;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
std::string m_ucs_host;
|
|
||||||
uint16 m_ucs_port;
|
|
||||||
|
|
||||||
static EQEmuConfig *_config;
|
static EQEmuConfig *_config;
|
||||||
Json::Value _root;
|
Json::Value _root;
|
||||||
static std::string ConfigFile;
|
static std::string ConfigFile;
|
||||||
@@ -134,12 +130,12 @@ class EQEmuConfig
|
|||||||
void parse_config();
|
void parse_config();
|
||||||
|
|
||||||
EQEmuConfig()
|
EQEmuConfig()
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
virtual ~EQEmuConfig() {}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
virtual ~EQEmuConfig() {}
|
|
||||||
|
|
||||||
// Produce a const singleton
|
// Produce a const singleton
|
||||||
static const EQEmuConfig *get()
|
static const EQEmuConfig *get()
|
||||||
@@ -148,44 +144,41 @@ class EQEmuConfig
|
|||||||
return (_config);
|
return (_config);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Allow the use to set the conf file to be used.
|
||||||
|
static void SetConfigFile(std::string file)
|
||||||
|
{
|
||||||
|
EQEmuConfig::ConfigFile = file;
|
||||||
|
}
|
||||||
|
|
||||||
// Load the config
|
// Load the config
|
||||||
static bool LoadConfig(const std::string& path = "")
|
static bool LoadConfig()
|
||||||
{
|
{
|
||||||
if (_config != nullptr) {
|
if (_config != nullptr) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
_config = new EQEmuConfig;
|
_config = new EQEmuConfig;
|
||||||
|
|
||||||
return parseFile(path);
|
return parseFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load config file and parse data
|
// Load config file and parse data
|
||||||
static bool parseFile(const std::string& file_path = ".")
|
static bool parseFile() {
|
||||||
{
|
|
||||||
if (_config == nullptr) {
|
if (_config == nullptr) {
|
||||||
return LoadConfig(file_path);
|
return LoadConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string file = fmt::format(
|
std::ifstream fconfig(EQEmuConfig::ConfigFile, std::ifstream::binary);
|
||||||
"{}/{}",
|
|
||||||
(file_path.empty() ? path.GetServerPath() : file_path),
|
|
||||||
EQEmuConfig::ConfigFile
|
|
||||||
);
|
|
||||||
|
|
||||||
std::ifstream fconfig(file, std::ifstream::binary);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
fconfig >> _config->_root;
|
fconfig >> _config->_root;
|
||||||
_config->parse_config();
|
_config->parse_config();
|
||||||
}
|
}
|
||||||
catch (std::exception &) {
|
catch (std::exception &) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Dump() const;
|
void Dump() const;
|
||||||
void CheckUcsConfigConversion();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
+305
-532
File diff suppressed because it is too large
Load Diff
+129
-178
@@ -23,9 +23,8 @@
|
|||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <cstdio>
|
#include <stdio.h>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#ifdef utf16_to_utf8
|
#ifdef utf16_to_utf8
|
||||||
@@ -38,8 +37,9 @@
|
|||||||
|
|
||||||
namespace Logs {
|
namespace Logs {
|
||||||
enum DebugLevel {
|
enum DebugLevel {
|
||||||
General = 1, // 1 - Low-Level general debugging, useful info on single line
|
General = 1, /* 1 - Low-Level general debugging, useful info on single line */
|
||||||
Detail // 2 - Use this for very chatty logging you want to leave in but don't want on by default
|
Moderate, /* 2 - Informational based, used in functions, when particular things load */
|
||||||
|
Detail /* 3 - Use this for extreme detail in logging, usually in extreme debugging in the stack or interprocess communication */
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -53,7 +53,7 @@ namespace Logs {
|
|||||||
AI,
|
AI,
|
||||||
Aggro,
|
Aggro,
|
||||||
Attack,
|
Attack,
|
||||||
DeprecatedCS, // deprecated
|
PacketClientServer,
|
||||||
Combat,
|
Combat,
|
||||||
Commands,
|
Commands,
|
||||||
Crash,
|
Crash,
|
||||||
@@ -64,36 +64,36 @@ namespace Logs {
|
|||||||
Inventory,
|
Inventory,
|
||||||
Launcher,
|
Launcher,
|
||||||
Netcode,
|
Netcode,
|
||||||
Normal, // deprecated
|
Normal,
|
||||||
Object,
|
Object,
|
||||||
Pathing,
|
Pathing,
|
||||||
QSServer, // deprecated
|
QSServer,
|
||||||
Quests,
|
Quests,
|
||||||
Rules,
|
Rules,
|
||||||
Skills,
|
Skills,
|
||||||
Spawns,
|
Spawns,
|
||||||
Spells,
|
Spells,
|
||||||
Status, // deprecated
|
Status,
|
||||||
TCPConnection,
|
TCPConnection,
|
||||||
Tasks,
|
Tasks,
|
||||||
Tradeskills,
|
Tradeskills,
|
||||||
Trading,
|
Trading,
|
||||||
Tribute,
|
Tribute,
|
||||||
UCSServer, // deprecated
|
UCSServer,
|
||||||
WebInterfaceServer, // deprecated
|
WebInterfaceServer,
|
||||||
WorldServer, // deprecated
|
WorldServer,
|
||||||
ZoneServer, // deprecated
|
ZoneServer,
|
||||||
MySQLError,
|
MySQLError,
|
||||||
MySQLQuery,
|
MySQLQuery,
|
||||||
Mercenaries,
|
Mercenaries,
|
||||||
QuestDebug,
|
QuestDebug,
|
||||||
DeprecatedSC, // deprecated
|
PacketServerClient,
|
||||||
DeprecatedCSU, // deprecated
|
PacketClientServerUnhandled,
|
||||||
DeprecatedSCD, // deprecated
|
PacketServerClientWithDump,
|
||||||
DeprecatedCSD, // deprecated
|
PacketClientServerWithDump,
|
||||||
Loginserver, // deprecated
|
Loginserver,
|
||||||
ClientLogin,
|
ClientLogin,
|
||||||
HeadlessClient, // deprecated
|
HeadlessClient,
|
||||||
HPUpdate,
|
HPUpdate,
|
||||||
FixZ,
|
FixZ,
|
||||||
Food,
|
Food,
|
||||||
@@ -103,10 +103,10 @@ namespace Logs {
|
|||||||
MobAppearance,
|
MobAppearance,
|
||||||
Info,
|
Info,
|
||||||
Warning,
|
Warning,
|
||||||
Critical, // deprecated
|
Critical,
|
||||||
Emergency, // deprecated
|
Emergency,
|
||||||
Alert, // deprecated
|
Alert,
|
||||||
Notice, // deprecated
|
Notice,
|
||||||
AIScanClose,
|
AIScanClose,
|
||||||
AIYellForHelp,
|
AIYellForHelp,
|
||||||
AICastBeneficialClose,
|
AICastBeneficialClose,
|
||||||
@@ -121,43 +121,19 @@ namespace Logs {
|
|||||||
Expeditions,
|
Expeditions,
|
||||||
DynamicZones,
|
DynamicZones,
|
||||||
Scheduler,
|
Scheduler,
|
||||||
Cheat,
|
|
||||||
ClientList,
|
|
||||||
DiaWind,
|
|
||||||
HTTP,
|
|
||||||
Saylink,
|
|
||||||
ChecksumVerification,
|
|
||||||
CombatRecord,
|
|
||||||
Hate,
|
|
||||||
Discord,
|
|
||||||
Faction,
|
|
||||||
PacketServerClient,
|
|
||||||
PacketClientServer,
|
|
||||||
PacketServerToServer,
|
|
||||||
Bugs,
|
|
||||||
QuestErrors,
|
|
||||||
PlayerEvents,
|
|
||||||
DataBuckets,
|
|
||||||
Zoning,
|
|
||||||
EqTime,
|
|
||||||
Corpses,
|
|
||||||
XTargets,
|
|
||||||
EvolveItem,
|
|
||||||
PositionUpdate,
|
|
||||||
KSM,
|
|
||||||
MaxCategoryID /* Don't Remove this */
|
MaxCategoryID /* Don't Remove this */
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If you add to this, make sure you update LogCategory
|
* If you add to this, make sure you update LogCategory
|
||||||
*/
|
*/
|
||||||
static const char *LogCategoryName[LogCategory::MaxCategoryID] = {
|
static const char* LogCategoryName[LogCategory::MaxCategoryID] = {
|
||||||
"",
|
"",
|
||||||
"AA",
|
"AA",
|
||||||
"AI",
|
"AI",
|
||||||
"Aggro",
|
"Aggro",
|
||||||
"Attack",
|
"Attack",
|
||||||
"Deprecated",
|
"Packet :: Client -> Server",
|
||||||
"Combat",
|
"Combat",
|
||||||
"Commands",
|
"Commands",
|
||||||
"Crash",
|
"Crash",
|
||||||
@@ -168,52 +144,52 @@ namespace Logs {
|
|||||||
"Inventory",
|
"Inventory",
|
||||||
"Launcher",
|
"Launcher",
|
||||||
"Netcode",
|
"Netcode",
|
||||||
"Normal (Deprecated)",
|
"Normal",
|
||||||
"Object",
|
"Object",
|
||||||
"Pathing",
|
"Pathing",
|
||||||
"QS Server (Deprecated)",
|
"QS Server",
|
||||||
"Quests",
|
"Quests",
|
||||||
"Rules",
|
"Rules",
|
||||||
"Skills",
|
"Skills",
|
||||||
"Spawns",
|
"Spawns",
|
||||||
"Spells",
|
"Spells",
|
||||||
"Status (Deprecated)",
|
"Status",
|
||||||
"TCP Connection",
|
"TCP Connection",
|
||||||
"Tasks",
|
"Tasks",
|
||||||
"Tradeskills",
|
"Tradeskills",
|
||||||
"Trading",
|
"Trading",
|
||||||
"Tribute",
|
"Tribute",
|
||||||
"UCS Server (Deprecated)",
|
"UCS Server",
|
||||||
"Web Interface (Deprecated)",
|
"WebInterface Server",
|
||||||
"World Server (Deprecated)",
|
"World Server",
|
||||||
"Zone Server (Deprecated)",
|
"Zone Server",
|
||||||
"QueryErr",
|
"MySQL Error",
|
||||||
"Query",
|
"MySQL Query",
|
||||||
"Mercenaries",
|
"Mercenaries",
|
||||||
"Quest Debug",
|
"Quest Debug",
|
||||||
"Legacy Packet Logging (Deprecated)",
|
"Packet :: Server -> Client",
|
||||||
"Legacy Packet Logging (Deprecated)",
|
"Packet :: Client -> Server Unhandled",
|
||||||
"Legacy Packet Logging (Deprecated)",
|
"Packet :: Server -> Client (Dump)",
|
||||||
"Legacy Packet Logging (Deprecated)",
|
"Packet :: Client -> Server (Dump)",
|
||||||
"Login Server (Deprecated)",
|
"Login Server",
|
||||||
"Client Login",
|
"Client Login",
|
||||||
"Headless Client (Deprecated)",
|
"Headless Client",
|
||||||
"HP Update",
|
"HP Update",
|
||||||
"FixZ",
|
"FixZ",
|
||||||
"Food",
|
"Food",
|
||||||
"Traps",
|
"Traps",
|
||||||
"NPC Roam Box",
|
"NPC Roam Box",
|
||||||
"NPC Scaling",
|
"NPC Scaling",
|
||||||
"MobAppearance",
|
"Mob Appearance",
|
||||||
"Info",
|
"Info",
|
||||||
"Warning",
|
"Warning",
|
||||||
"Critical (Deprecated)",
|
"Critical",
|
||||||
"Emergency (Deprecated)",
|
"Emergency",
|
||||||
"Alert (Deprecated)",
|
"Alert",
|
||||||
"Notice (Deprecated)",
|
"Notice",
|
||||||
"AI Scan",
|
"AI Scan Close",
|
||||||
"AI Yell",
|
"AI Yell For Help",
|
||||||
"AI CastBeneficial",
|
"AI Cast Beneficial Close",
|
||||||
"AOE Cast",
|
"AOE Cast",
|
||||||
"Entity Management",
|
"Entity Management",
|
||||||
"Flee",
|
"Flee",
|
||||||
@@ -225,39 +201,11 @@ namespace Logs {
|
|||||||
"Expeditions",
|
"Expeditions",
|
||||||
"DynamicZones",
|
"DynamicZones",
|
||||||
"Scheduler",
|
"Scheduler",
|
||||||
"Cheat",
|
|
||||||
"ClientList",
|
|
||||||
"DialogueWindow",
|
|
||||||
"HTTP",
|
|
||||||
"Saylink",
|
|
||||||
"ChecksumVer",
|
|
||||||
"CombatRecord",
|
|
||||||
"Hate",
|
|
||||||
"Discord",
|
|
||||||
"Faction",
|
|
||||||
"Packet S->C",
|
|
||||||
"Packet C->S",
|
|
||||||
"Packet S->S",
|
|
||||||
"Bugs",
|
|
||||||
"QuestErrors",
|
|
||||||
"PlayerEvents",
|
|
||||||
"DataBuckets",
|
|
||||||
"Zoning",
|
|
||||||
"EqTime",
|
|
||||||
"Corpses",
|
|
||||||
"XTargets",
|
|
||||||
"EvolveItem",
|
|
||||||
"PositionUpdate",
|
|
||||||
"KSM" // Kernel Samepage Merging
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
#include "eqemu_logsys_log_aliases.h"
|
#include "eqemu_logsys_log_aliases.h"
|
||||||
|
|
||||||
class Database;
|
|
||||||
|
|
||||||
constexpr uint16 MAX_DISCORD_WEBHOOK_ID = 300;
|
|
||||||
|
|
||||||
class EQEmuLogSys {
|
class EQEmuLogSys {
|
||||||
public:
|
public:
|
||||||
EQEmuLogSys();
|
EQEmuLogSys();
|
||||||
@@ -268,8 +216,7 @@ public:
|
|||||||
* This should be handled on deconstructor but to be safe we use it anyways.
|
* This should be handled on deconstructor but to be safe we use it anyways.
|
||||||
*/
|
*/
|
||||||
void CloseFileLogs();
|
void CloseFileLogs();
|
||||||
EQEmuLogSys *LoadLogSettingsDefaults();
|
void LoadLogSettingsDefaults();
|
||||||
EQEmuLogSys *LoadLogDatabaseSettings();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param directory_name
|
* @param directory_name
|
||||||
@@ -299,7 +246,7 @@ public:
|
|||||||
* Used in file logs to prepend a timestamp entry for logs
|
* Used in file logs to prepend a timestamp entry for logs
|
||||||
* @param time_stamp
|
* @param time_stamp
|
||||||
*/
|
*/
|
||||||
void SetCurrentTimeStamp(char *time_stamp);
|
void SetCurrentTimeStamp(char* time_stamp);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param log_name
|
* @param log_name
|
||||||
@@ -320,104 +267,108 @@ public:
|
|||||||
uint8 log_to_file;
|
uint8 log_to_file;
|
||||||
uint8 log_to_console;
|
uint8 log_to_console;
|
||||||
uint8 log_to_gmsay;
|
uint8 log_to_gmsay;
|
||||||
uint8 log_to_discord;
|
|
||||||
int discord_webhook_id;
|
|
||||||
uint8 is_category_enabled; /* When any log output in a category > 0, set this to 1 as (Enabled) */
|
uint8 is_category_enabled; /* When any log output in a category > 0, set this to 1 as (Enabled) */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct OriginationInfo {
|
|
||||||
std::string zone_short_name;
|
|
||||||
std::string zone_long_name;
|
|
||||||
int instance_id;
|
|
||||||
};
|
|
||||||
|
|
||||||
OriginationInfo origination_info{};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Internally used memory reference for all log settings per category
|
* Internally used memory reference for all log settings per category
|
||||||
* These are loaded via DB and have defaults loaded in LoadLogSettingsDefaults
|
* These are loaded via DB and have defaults loaded in LoadLogSettingsDefaults
|
||||||
* Database loaded via LogSys.SetDatabase(&database)->LoadLogDatabaseSettings();
|
* Database loaded via Database::LoadLogSettings(log_settings)
|
||||||
*/
|
*/
|
||||||
LogSettings log_settings[Logs::LogCategory::MaxCategoryID]{};
|
LogSettings log_settings[Logs::LogCategory::MaxCategoryID]{};
|
||||||
|
|
||||||
// temporary bucket to re-load after silencing
|
bool file_logs_enabled = false;
|
||||||
LogSettings pre_silence_settings[Logs::LogCategory::MaxCategoryID]{};
|
|
||||||
|
|
||||||
struct LogEnabled {
|
/**
|
||||||
bool log_to_file_enabled;
|
* Sets Executable platform (Zone/World/UCS) etc.
|
||||||
bool log_to_console_enabled;
|
*/
|
||||||
bool log_to_gmsay_enabled;
|
int log_platform = 0;
|
||||||
bool log_to_discord_enabled;
|
|
||||||
bool log_enabled;
|
|
||||||
};
|
|
||||||
|
|
||||||
LogEnabled GetLogsEnabled(const Logs::DebugLevel &debug_level, const uint16 &log_category);
|
/**
|
||||||
bool IsLogEnabled(const Logs::DebugLevel &debug_level, const uint16 &log_category);
|
* File name used in writing logs
|
||||||
|
*/
|
||||||
|
std::string platform_file_name;
|
||||||
|
|
||||||
struct DiscordWebhooks {
|
/**
|
||||||
int id;
|
* GMSay Client Message colors mapped by category
|
||||||
std::string webhook_name;
|
*
|
||||||
std::string webhook_url;
|
* @param log_category
|
||||||
};
|
* @return
|
||||||
|
*/
|
||||||
const DiscordWebhooks *GetDiscordWebhooks() const;
|
|
||||||
|
|
||||||
// gmsay
|
|
||||||
uint16 GetGMSayColorFromCategory(uint16 log_category);
|
uint16 GetGMSayColorFromCategory(uint16 log_category);
|
||||||
|
|
||||||
EQEmuLogSys *SetGMSayHandler(const std::function<void(uint16 log_type, const char *func, const std::string &)>& f)
|
/**
|
||||||
{
|
* @param f
|
||||||
m_on_log_gmsay_hook = f;
|
*/
|
||||||
return this;
|
void SetGMSayHandler(std::function<void(uint16 log_type, const std::string&)> f) { on_log_gmsay_hook = f; }
|
||||||
}
|
|
||||||
|
|
||||||
EQEmuLogSys *SetDiscordHandler(std::function<void(uint16 log_category, int webhook_id, const std::string &)> f)
|
/**
|
||||||
{
|
* @param f
|
||||||
m_on_log_discord_hook = f;
|
*/
|
||||||
return this;
|
void SetConsoleHandler(std::function<void(uint16 debug_level, uint16 log_type, const std::string&)> f) { on_log_console_hook = f; }
|
||||||
}
|
|
||||||
|
|
||||||
// console
|
/**
|
||||||
void SetConsoleHandler(
|
* Silence console logging
|
||||||
std::function<void(
|
*/
|
||||||
uint16 log_type,
|
|
||||||
const std::string &
|
|
||||||
)> f
|
|
||||||
) { m_on_log_console_hook = f; }
|
|
||||||
void SilenceConsoleLogging();
|
void SilenceConsoleLogging();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turn on all console logging
|
||||||
|
*/
|
||||||
void EnableConsoleLogging();
|
void EnableConsoleLogging();
|
||||||
|
|
||||||
// database
|
|
||||||
EQEmuLogSys *SetDatabase(Database *db);
|
|
||||||
|
|
||||||
[[nodiscard]] const std::string &GetLogPath() const;
|
|
||||||
EQEmuLogSys * SetLogPath(const std::string &log_path);
|
|
||||||
|
|
||||||
void DisableMySQLErrorLogs();
|
|
||||||
void EnableMySQLErrorLogs();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
// reference to database
|
/**
|
||||||
Database *m_database;
|
* Callback pointer to zone process for hooking logs to zone using GMSay
|
||||||
std::function<void(uint16 log_category, const char *func, const std::string &)> m_on_log_gmsay_hook;
|
*/
|
||||||
std::function<void(uint16 log_category, int webhook_id, const std::string &)> m_on_log_discord_hook;
|
std::function<void(uint16 log_category, const std::string&)> on_log_gmsay_hook;
|
||||||
std::function<void(uint16 log_category, const std::string &)> m_on_log_console_hook;
|
std::function<void(uint16 debug_level, uint16 log_category, const std::string&)> on_log_console_hook;
|
||||||
DiscordWebhooks m_discord_webhooks[MAX_DISCORD_WEBHOOK_ID]{};
|
|
||||||
bool m_file_logs_enabled = false;
|
|
||||||
int m_log_platform = 0;
|
|
||||||
std::string m_platform_file_name;
|
|
||||||
std::string m_log_path;
|
|
||||||
|
|
||||||
void ProcessConsoleMessage(
|
/**
|
||||||
uint16 log_category,
|
* Formats log messages like '[Category] This is a log message'
|
||||||
const std::string &message,
|
*/
|
||||||
const char *file,
|
std::string FormatOutMessageString(uint16 log_category, const std::string &in_message);
|
||||||
const char *func,
|
|
||||||
int line
|
/**
|
||||||
);
|
* Linux console color messages mapped by category
|
||||||
void ProcessLogWrite(uint16 log_category, const std::string &message);
|
*
|
||||||
void InjectTablesIfNotExist();
|
* @param log_category
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
std::string GetLinuxConsoleColorFromCategory(uint16 log_category);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Windows console color messages mapped by category
|
||||||
|
*/
|
||||||
|
uint16 GetWindowsConsoleColorFromCategory(uint16 log_category);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param debug_level
|
||||||
|
* @param log_category
|
||||||
|
* @param message
|
||||||
|
*/
|
||||||
|
void ProcessConsoleMessage(uint16 debug_level, uint16 log_category, const std::string &message);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param debug_level
|
||||||
|
* @param log_category
|
||||||
|
* @param message
|
||||||
|
*/
|
||||||
|
void ProcessGMSay(uint16 debug_level, uint16 log_category, const std::string &message);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param debug_level
|
||||||
|
* @param log_category
|
||||||
|
* @param message
|
||||||
|
*/
|
||||||
|
void ProcessLogWrite(uint16 debug_level, uint16 log_category, const std::string &message);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param log_category
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
bool IsRfc5424LogCategory(uint16 log_category);
|
||||||
};
|
};
|
||||||
|
|
||||||
extern EQEmuLogSys LogSys;
|
extern EQEmuLogSys LogSys;
|
||||||
|
|||||||
+563
-423
File diff suppressed because it is too large
Load Diff
+12
-52
@@ -46,16 +46,20 @@ EQTime::EQTime()
|
|||||||
timezone = 0;
|
timezone = 0;
|
||||||
memset(&eqTime, 0, sizeof(eqTime));
|
memset(&eqTime, 0, sizeof(eqTime));
|
||||||
//Defaults for time
|
//Defaults for time
|
||||||
TimeOfDay_Struct t{};
|
TimeOfDay_Struct start;
|
||||||
t.day = 1;
|
start.day = 1;
|
||||||
t.hour = 9;
|
start.hour = 9;
|
||||||
t.minute = 0;
|
start.minute = 0;
|
||||||
t.month = 1;
|
start.month = 1;
|
||||||
t.year = 3100;
|
start.year = 3100;
|
||||||
//Set default time zone
|
//Set default time zone
|
||||||
timezone = 0;
|
timezone = 0;
|
||||||
//Start EQTimer
|
//Start EQTimer
|
||||||
SetCurrentEQTimeOfDay(t, time(nullptr));
|
SetCurrentEQTimeOfDay(start, time(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
EQTime::~EQTime()
|
||||||
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
//getEQTimeOfDay - Reads timeConvert and writes the result to eqTimeOfDay
|
//getEQTimeOfDay - Reads timeConvert and writes the result to eqTimeOfDay
|
||||||
@@ -199,48 +203,4 @@ void EQTime::ToString(TimeOfDay_Struct *t, std::string &str) {
|
|||||||
t->month, t->day, t->year, t->hour, t->minute);
|
t->month, t->day, t->year, t->hour, t->minute);
|
||||||
buf[127] = '\0';
|
buf[127] = '\0';
|
||||||
str = buf;
|
str = buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool EQTime::IsDayTime() {
|
|
||||||
TimeOfDay_Struct tod{}; //Day time is 5am to 6:59pm (14 hours in-game)
|
|
||||||
GetCurrentEQTimeOfDay(&tod); //TODO: what if it fails and returns zero?
|
|
||||||
|
|
||||||
if (tod.hour >= 5 || tod.hour < 19) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool EQTime::IsNightTime() {
|
|
||||||
TimeOfDay_Struct tod{}; //Night time is 7pm to 4:59am (10 hours in-game)
|
|
||||||
GetCurrentEQTimeOfDay(&tod); //TODO: what if it fails and returns zero?
|
|
||||||
|
|
||||||
if (tod.hour >= 19 || tod.hour < 5) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool EQTime::IsInbetweenTime(uint8 min_time, uint8 max_time) {
|
|
||||||
TimeOfDay_Struct tod{};
|
|
||||||
GetCurrentEQTimeOfDay(&tod);
|
|
||||||
|
|
||||||
if (min_time == 0 || max_time == 0 || min_time > 24 || max_time > 24) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (max_time < min_time) {
|
|
||||||
if ((tod.hour >= min_time && tod.hour > max_time) || (tod.hour < min_time && tod.hour <= max_time)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (tod.hour >= min_time && tod.hour <= max_time) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
+1
-4
@@ -18,7 +18,7 @@ public:
|
|||||||
//Constructor/destructor
|
//Constructor/destructor
|
||||||
EQTime(TimeOfDay_Struct start_eq, time_t start_real);
|
EQTime(TimeOfDay_Struct start_eq, time_t start_real);
|
||||||
EQTime();
|
EQTime();
|
||||||
~EQTime() = default;
|
~EQTime();
|
||||||
|
|
||||||
//Get functions
|
//Get functions
|
||||||
int GetCurrentEQTimeOfDay( TimeOfDay_Struct *eqTimeOfDay ) { return(GetCurrentEQTimeOfDay(time(nullptr), eqTimeOfDay)); }
|
int GetCurrentEQTimeOfDay( TimeOfDay_Struct *eqTimeOfDay ) { return(GetCurrentEQTimeOfDay(time(nullptr), eqTimeOfDay)); }
|
||||||
@@ -28,9 +28,6 @@ public:
|
|||||||
uint32 getEQTimeZone() { return timezone; }
|
uint32 getEQTimeZone() { return timezone; }
|
||||||
uint32 getEQTimeZoneHr() { return timezone/60; }
|
uint32 getEQTimeZoneHr() { return timezone/60; }
|
||||||
uint32 getEQTimeZoneMin() { return timezone%60; }
|
uint32 getEQTimeZoneMin() { return timezone%60; }
|
||||||
bool IsDayTime();
|
|
||||||
bool IsNightTime();
|
|
||||||
bool IsInbetweenTime(uint8 min_time, uint8 max_time);
|
|
||||||
|
|
||||||
//Set functions
|
//Set functions
|
||||||
int SetCurrentEQTimeOfDay(TimeOfDay_Struct start_eq, time_t start_real);
|
int SetCurrentEQTimeOfDay(TimeOfDay_Struct start_eq, time_t start_real);
|
||||||
|
|||||||
@@ -0,0 +1,59 @@
|
|||||||
|
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Library General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Library General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Library General Public
|
||||||
|
License along with this library; if not, write to the Free
|
||||||
|
Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
|
||||||
|
MA 02111-1307, USA */
|
||||||
|
|
||||||
|
/* Error messages for mysql clients */
|
||||||
|
/* error messages for the demon is in share/language/errmsg.sys */
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
void init_client_errs(void);
|
||||||
|
extern const char *client_errors[]; /* Error messages */
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define CR_MIN_ERROR 2000 /* For easier client code */
|
||||||
|
#define CR_MAX_ERROR 2999
|
||||||
|
#if defined(OS2) && defined( MYSQL_SERVER)
|
||||||
|
#define CER(X) client_errors[(X)-CR_MIN_ERROR]
|
||||||
|
#else
|
||||||
|
#define ER(X) client_errors[(X)-CR_MIN_ERROR]
|
||||||
|
#endif
|
||||||
|
#define CLIENT_ERRMAP 2 /* Errormap used by my_error() */
|
||||||
|
|
||||||
|
#define CR_UNKNOWN_ERROR 2000
|
||||||
|
#define CR_SOCKET_CREATE_ERROR 2001
|
||||||
|
#define CR_CONNECTION_ERROR 2002
|
||||||
|
#define CR_CONN_HOST_ERROR 2003
|
||||||
|
#define CR_IPSOCK_ERROR 2004
|
||||||
|
#define CR_UNKNOWN_HOST 2005
|
||||||
|
#define CR_SERVER_GONE_ERROR 2006
|
||||||
|
#define CR_VERSION_ERROR 2007
|
||||||
|
#define CR_OUT_OF_MEMORY 2008
|
||||||
|
#define CR_WRONG_HOST_INFO 2009
|
||||||
|
#define CR_LOCALHOST_CONNECTION 2010
|
||||||
|
#define CR_TCP_CONNECTION 2011
|
||||||
|
#define CR_SERVER_HANDSHAKE_ERR 2012
|
||||||
|
#define CR_SERVER_LOST 2013
|
||||||
|
#define CR_COMMANDS_OUT_OF_SYNC 2014
|
||||||
|
#define CR_NAMEDPIPE_CONNECTION 2015
|
||||||
|
#define CR_NAMEDPIPEWAIT_ERROR 2016
|
||||||
|
#define CR_NAMEDPIPEOPEN_ERROR 2017
|
||||||
|
#define CR_NAMEDPIPESETSTATE_ERROR 2018
|
||||||
|
#define CR_CANT_READ_CHARSET 2019
|
||||||
|
#define CR_NET_PACKET_TOO_LARGE 2020
|
||||||
+4
-4
@@ -1,14 +1,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <any>
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <exception>
|
#include <exception>
|
||||||
#include "event_loop.h"
|
#include "event_loop.h"
|
||||||
|
#include "../any.h"
|
||||||
|
|
||||||
namespace EQ {
|
namespace EQ {
|
||||||
class Task
|
class Task
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
typedef std::function<void(const std::any&)> ResolveFn;
|
typedef std::function<void(const EQ::Any&)> ResolveFn;
|
||||||
typedef std::function<void(const std::exception&)> RejectFn;
|
typedef std::function<void(const std::exception&)> RejectFn;
|
||||||
typedef std::function<void()> FinallyFn;
|
typedef std::function<void()> FinallyFn;
|
||||||
typedef std::function<void(ResolveFn, RejectFn)> TaskFn;
|
typedef std::function<void(ResolveFn, RejectFn)> TaskFn;
|
||||||
@@ -19,7 +19,7 @@ namespace EQ {
|
|||||||
RejectFn on_catch;
|
RejectFn on_catch;
|
||||||
FinallyFn on_finally;
|
FinallyFn on_finally;
|
||||||
bool has_result;
|
bool has_result;
|
||||||
std::any result;
|
EQ::Any result;
|
||||||
bool has_error;
|
bool has_error;
|
||||||
std::exception error;
|
std::exception error;
|
||||||
};
|
};
|
||||||
@@ -63,7 +63,7 @@ namespace EQ {
|
|||||||
uv_queue_work(EventLoop::Get().Handle(), m_work, [](uv_work_t* req) {
|
uv_queue_work(EventLoop::Get().Handle(), m_work, [](uv_work_t* req) {
|
||||||
TaskBaton *baton = (TaskBaton*)req->data;
|
TaskBaton *baton = (TaskBaton*)req->data;
|
||||||
|
|
||||||
baton->fn([baton](const std::any& result) {
|
baton->fn([baton](const EQ::Any& result) {
|
||||||
baton->has_error = false;
|
baton->has_error = false;
|
||||||
baton->has_result = true;
|
baton->has_result = true;
|
||||||
baton->result = result;
|
baton->result = result;
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ namespace EQ
|
|||||||
_running = true;
|
_running = true;
|
||||||
|
|
||||||
for (size_t i = 0; i < threads; ++i) {
|
for (size_t i = 0; i < threads; ++i) {
|
||||||
_threads.emplace_back(std::thread(std::bind(&TaskScheduler::ProcessWork, this)));
|
_threads.push_back(std::thread(std::bind(&TaskScheduler::ProcessWork, this)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,8 +60,8 @@ namespace EQ
|
|||||||
}
|
}
|
||||||
|
|
||||||
template<typename Fn, typename... Args>
|
template<typename Fn, typename... Args>
|
||||||
auto Enqueue(Fn&& fn, Args&&... args) -> std::future<typename std::invoke_result<Fn, Args...>::type> {
|
auto Enqueue(Fn&& fn, Args&&... args) -> std::future<typename std::result_of<Fn(Args...)>::type> {
|
||||||
using return_type = typename std::invoke_result<Fn, Args...>::type;
|
using return_type = typename std::result_of<Fn(Args...)>::type;
|
||||||
|
|
||||||
auto task = std::make_shared<std::packaged_task<return_type()>>(
|
auto task = std::make_shared<std::packaged_task<return_type()>>(
|
||||||
std::bind(std::forward<Fn>(fn), std::forward<Args>(args)...)
|
std::bind(std::forward<Fn>(fn), std::forward<Args>(args)...)
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,214 +0,0 @@
|
|||||||
#ifndef EQEMU_PLAYER_EVENT_DISCORD_FORMATTER_H
|
|
||||||
#define EQEMU_PLAYER_EVENT_DISCORD_FORMATTER_H
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include "player_events.h"
|
|
||||||
#include "../repositories/base/base_player_event_logs_repository.h"
|
|
||||||
#include <cereal/archives/json.hpp>
|
|
||||||
#include <cereal/types/vector.hpp>
|
|
||||||
|
|
||||||
struct DiscordField {
|
|
||||||
std::string name;
|
|
||||||
std::string value;
|
|
||||||
bool is_inline;
|
|
||||||
|
|
||||||
// cereal
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive &ar)
|
|
||||||
{
|
|
||||||
ar(
|
|
||||||
CEREAL_NVP(name),
|
|
||||||
CEREAL_NVP(value),
|
|
||||||
cereal::make_nvp("inline", is_inline)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct DiscordAuthor {
|
|
||||||
std::string name;
|
|
||||||
std::string icon_url;
|
|
||||||
std::string url;
|
|
||||||
|
|
||||||
// cereal
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive &ar)
|
|
||||||
{
|
|
||||||
ar(
|
|
||||||
CEREAL_NVP(name),
|
|
||||||
CEREAL_NVP(icon_url),
|
|
||||||
CEREAL_NVP(url)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct DiscordEmbed {
|
|
||||||
std::vector<DiscordField> fields;
|
|
||||||
std::string title;
|
|
||||||
std::string description;
|
|
||||||
std::string timestamp;
|
|
||||||
DiscordAuthor author;
|
|
||||||
|
|
||||||
|
|
||||||
// cereal
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive &ar)
|
|
||||||
{
|
|
||||||
ar(
|
|
||||||
CEREAL_NVP(fields),
|
|
||||||
CEREAL_NVP(title),
|
|
||||||
CEREAL_NVP(description),
|
|
||||||
CEREAL_NVP(timestamp),
|
|
||||||
CEREAL_NVP(author)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct DiscordWebhook {
|
|
||||||
std::vector<DiscordEmbed> embeds;
|
|
||||||
std::string content;
|
|
||||||
std::string avatar_url;
|
|
||||||
|
|
||||||
// cereal
|
|
||||||
template<class Archive>
|
|
||||||
void serialize(Archive &ar)
|
|
||||||
{
|
|
||||||
ar(
|
|
||||||
CEREAL_NVP(embeds),
|
|
||||||
CEREAL_NVP(avatar_url),
|
|
||||||
CEREAL_NVP(content)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
class PlayerEventDiscordFormatter {
|
|
||||||
public:
|
|
||||||
static std::string GetCurrentTimestamp();
|
|
||||||
static std::string FormatEventSay(const PlayerEvent::PlayerEventContainer &c, const PlayerEvent::SayEvent &e);
|
|
||||||
static std::string
|
|
||||||
FormatGMCommand(const PlayerEvent::PlayerEventContainer &c, const PlayerEvent::GMCommandEvent &e);
|
|
||||||
static void BuildDiscordField(
|
|
||||||
std::vector<DiscordField> *f,
|
|
||||||
const std::string &name,
|
|
||||||
const std::string &value,
|
|
||||||
bool is_inline = true
|
|
||||||
);
|
|
||||||
static void BuildBaseEmbed(
|
|
||||||
std::vector<DiscordEmbed> *e,
|
|
||||||
const std::vector<DiscordField> &f,
|
|
||||||
PlayerEvent::PlayerEventContainer c
|
|
||||||
);
|
|
||||||
static std::string FormatWithNodata(const PlayerEvent::PlayerEventContainer &c);
|
|
||||||
|
|
||||||
static std::string FormatAAGainedEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::AAGainedEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatAAPurchasedEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::AAPurchasedEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatDeathEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::DeathEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatFishSuccessEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::FishSuccessEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatForageSuccessEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::ForageSuccessEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatDestroyItemEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::DestroyItemEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatDiscoverItemEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::DiscoverItemEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatDroppedItemEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::DroppedItemEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatLevelGainedEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::LevelGainedEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatLevelLostEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::LevelLostEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatLootItemEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::LootItemEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatGroundSpawnPickupEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::GroundSpawnPickupEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatMerchantPurchaseEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::MerchantPurchaseEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatMerchantSellEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::MerchantSellEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatNPCHandinEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::HandinEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatSkillUpEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::SkillUpEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTaskAcceptEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TaskAcceptEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTaskCompleteEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TaskCompleteEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTaskUpdateEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TaskUpdateEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTradeEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TradeEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTraderPurchaseEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TraderPurchaseEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatTraderSellEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::TraderSellEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatResurrectAcceptEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::ResurrectAcceptEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatSplitMoneyEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::SplitMoneyEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatCombineEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::CombineEvent &e
|
|
||||||
);
|
|
||||||
static std::string FormatZoningEvent(
|
|
||||||
const PlayerEvent::PlayerEventContainer &c,
|
|
||||||
const PlayerEvent::ZoningEvent &e
|
|
||||||
);
|
|
||||||
static DiscordWebhook BuildDiscordWebhook(
|
|
||||||
const PlayerEvent::PlayerEventContainer &p,
|
|
||||||
std::vector<DiscordEmbed> &embeds
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#endif //EQEMU_PLAYER_EVENT_DISCORD_FORMATTER_H
|
|
||||||
@@ -1,715 +0,0 @@
|
|||||||
#include <cereal/archives/json.hpp>
|
|
||||||
#include "player_event_logs.h"
|
|
||||||
#include "player_event_discord_formatter.h"
|
|
||||||
#include "../platform.h"
|
|
||||||
#include "../rulesys.h"
|
|
||||||
|
|
||||||
const uint32 PROCESS_RETENTION_TRUNCATION_TIMER_INTERVAL = 60 * 60 * 1000; // 1 hour
|
|
||||||
|
|
||||||
// general initialization routine
|
|
||||||
void PlayerEventLogs::Init()
|
|
||||||
{
|
|
||||||
m_process_batch_events_timer.SetTimer(RuleI(Logging, BatchPlayerEventProcessIntervalSeconds) * 1000);
|
|
||||||
m_process_retention_truncation_timer.SetTimer(PROCESS_RETENTION_TRUNCATION_TIMER_INTERVAL);
|
|
||||||
|
|
||||||
ValidateDatabaseConnection();
|
|
||||||
|
|
||||||
// initialize settings array
|
|
||||||
for (int i = PlayerEvent::GM_COMMAND; i != PlayerEvent::MAX; i++) {
|
|
||||||
m_settings[i].id = i;
|
|
||||||
m_settings[i].event_name = PlayerEvent::EventName[i];
|
|
||||||
m_settings[i].event_enabled = 1;
|
|
||||||
m_settings[i].retention_days = 0;
|
|
||||||
m_settings[i].discord_webhook_id = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
SetSettingsDefaults();
|
|
||||||
|
|
||||||
// initialize settings from database
|
|
||||||
auto s = PlayerEventLogSettingsRepository::All(*m_database);
|
|
||||||
std::vector<int> db{};
|
|
||||||
db.reserve(s.size());
|
|
||||||
for (auto &e: s) {
|
|
||||||
if (e.id >= PlayerEvent::MAX) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
m_settings[e.id] = e;
|
|
||||||
db.emplace_back(e.id);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<PlayerEventLogSettingsRepository::PlayerEventLogSettings> settings_to_insert{};
|
|
||||||
|
|
||||||
// insert entries that don't exist in database
|
|
||||||
for (int i = PlayerEvent::GM_COMMAND; i != PlayerEvent::MAX; i++) {
|
|
||||||
bool is_in_database = std::find(db.begin(), db.end(), i) != db.end();
|
|
||||||
bool is_deprecated = Strings::Contains(PlayerEvent::EventName[i], "Deprecated");
|
|
||||||
bool is_implemented = !Strings::Contains(PlayerEvent::EventName[i], "Unimplemented");
|
|
||||||
|
|
||||||
// remove when deprecated
|
|
||||||
if (is_deprecated && is_in_database) {
|
|
||||||
LogInfo("[Deprecated] Removing PlayerEvent [{}] ({})", PlayerEvent::EventName[i], i);
|
|
||||||
PlayerEventLogSettingsRepository::DeleteWhere(*m_database, fmt::format("id = {}", i));
|
|
||||||
}
|
|
||||||
// remove when unimplemented if present
|
|
||||||
if (!is_implemented && is_in_database) {
|
|
||||||
LogInfo("[Unimplemented] Removing PlayerEvent [{}] ({})", PlayerEvent::EventName[i], i);
|
|
||||||
PlayerEventLogSettingsRepository::DeleteWhere(*m_database, fmt::format("id = {}", i));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool is_missing_in_database = std::find(db.begin(), db.end(), i) == db.end();
|
|
||||||
if (is_missing_in_database && is_implemented && !is_deprecated) {
|
|
||||||
LogInfo("[New] PlayerEvent [{}] ({})", PlayerEvent::EventName[i], i);
|
|
||||||
|
|
||||||
auto c = PlayerEventLogSettingsRepository::NewEntity();
|
|
||||||
c.id = i;
|
|
||||||
c.event_name = PlayerEvent::EventName[i];
|
|
||||||
c.event_enabled = m_settings[i].event_enabled;
|
|
||||||
c.retention_days = m_settings[i].retention_days;
|
|
||||||
settings_to_insert.emplace_back(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!settings_to_insert.empty()) {
|
|
||||||
PlayerEventLogSettingsRepository::ReplaceMany(*m_database, settings_to_insert);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool processing_in_world = !RuleB(Logging, PlayerEventsQSProcess) && IsWorld();
|
|
||||||
bool processing_in_qs = RuleB(Logging, PlayerEventsQSProcess) && IsQueryServ();
|
|
||||||
|
|
||||||
// on initial boot process truncation
|
|
||||||
if (processing_in_world || processing_in_qs) {
|
|
||||||
ProcessRetentionTruncation();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the database object, during initialization
|
|
||||||
PlayerEventLogs *PlayerEventLogs::SetDatabase(Database *db)
|
|
||||||
{
|
|
||||||
m_database = db;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// validates whether the connection is valid or not, used in initialization
|
|
||||||
bool PlayerEventLogs::ValidateDatabaseConnection()
|
|
||||||
{
|
|
||||||
if (!m_database) {
|
|
||||||
LogError("No database connection");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// determines if the passed in event is enabled or not
|
|
||||||
// this is used to gate logic or events from firing off
|
|
||||||
// this is used prior to building the events, we don't want to
|
|
||||||
// build the events, send them through the stack in a function call
|
|
||||||
// only to discard them immediately afterwards, very wasteful on resources
|
|
||||||
// the quest api currently does this
|
|
||||||
bool PlayerEventLogs::IsEventEnabled(PlayerEvent::EventType event)
|
|
||||||
{
|
|
||||||
return m_settings[event].event_enabled ? m_settings[event].event_enabled : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// this processes any current player events on the queue
|
|
||||||
void PlayerEventLogs::ProcessBatchQueue()
|
|
||||||
{
|
|
||||||
m_batch_queue_lock.lock();
|
|
||||||
if (m_record_batch_queue.empty()) {
|
|
||||||
m_batch_queue_lock.unlock();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
BenchTimer benchmark;
|
|
||||||
|
|
||||||
// flush many
|
|
||||||
PlayerEventLogsRepository::InsertMany(*m_database, m_record_batch_queue);
|
|
||||||
LogPlayerEventsDetail(
|
|
||||||
"Processing batch player event log queue of [{}] took [{}]",
|
|
||||||
m_record_batch_queue.size(),
|
|
||||||
benchmark.elapsed()
|
|
||||||
);
|
|
||||||
|
|
||||||
// empty
|
|
||||||
m_record_batch_queue = {};
|
|
||||||
m_batch_queue_lock.unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
// adds a player event to the queue
|
|
||||||
void PlayerEventLogs::AddToQueue(const PlayerEventLogsRepository::PlayerEventLogs &log)
|
|
||||||
{
|
|
||||||
m_batch_queue_lock.lock();
|
|
||||||
m_record_batch_queue.emplace_back(log);
|
|
||||||
m_batch_queue_lock.unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
// fills common event data in the SendEvent function
|
|
||||||
void PlayerEventLogs::FillPlayerEvent(
|
|
||||||
const PlayerEvent::PlayerEvent &p,
|
|
||||||
PlayerEventLogsRepository::PlayerEventLogs &n
|
|
||||||
)
|
|
||||||
{
|
|
||||||
n.account_id = p.account_id;
|
|
||||||
n.character_id = p.character_id;
|
|
||||||
n.zone_id = p.zone_id;
|
|
||||||
n.instance_id = p.instance_id;
|
|
||||||
n.x = p.x;
|
|
||||||
n.y = p.y;
|
|
||||||
n.z = p.z;
|
|
||||||
n.heading = p.heading;
|
|
||||||
}
|
|
||||||
|
|
||||||
// builds the dynamic packet used to ship the player event over the wire
|
|
||||||
// supports serializing the struct so it can be rebuilt on the other end
|
|
||||||
std::unique_ptr<ServerPacket>
|
|
||||||
PlayerEventLogs::BuildPlayerEventPacket(const PlayerEvent::PlayerEventContainer &e)
|
|
||||||
{
|
|
||||||
EQ::Net::DynamicPacket dyn_pack;
|
|
||||||
dyn_pack.PutSerialize(0, e);
|
|
||||||
auto pack_size = sizeof(ServerSendPlayerEvent_Struct) + dyn_pack.Length();
|
|
||||||
auto pack = std::make_unique<ServerPacket>(ServerOP_PlayerEvent, static_cast<uint32_t>(pack_size));
|
|
||||||
auto buf = reinterpret_cast<ServerSendPlayerEvent_Struct *>(pack->pBuffer);
|
|
||||||
buf->cereal_size = static_cast<uint32_t>(dyn_pack.Length());
|
|
||||||
memcpy(buf->cereal_data, dyn_pack.Data(), dyn_pack.Length());
|
|
||||||
|
|
||||||
return pack;
|
|
||||||
}
|
|
||||||
|
|
||||||
const PlayerEventLogSettingsRepository::PlayerEventLogSettings *PlayerEventLogs::GetSettings() const
|
|
||||||
{
|
|
||||||
return m_settings;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool PlayerEventLogs::IsEventDiscordEnabled(int32_t event_type_id)
|
|
||||||
{
|
|
||||||
// out of bounds check
|
|
||||||
if (event_type_id >= PlayerEvent::EventType::MAX) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// make sure webhook id is set
|
|
||||||
if (m_settings[event_type_id].discord_webhook_id == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ensure there is a matching webhook to begin with
|
|
||||||
if (!LogSys.GetDiscordWebhooks()[m_settings[event_type_id].discord_webhook_id].webhook_url.empty()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string PlayerEventLogs::GetDiscordWebhookUrlFromEventType(int32_t event_type_id)
|
|
||||||
{
|
|
||||||
// out of bounds check
|
|
||||||
if (event_type_id >= PlayerEvent::EventType::MAX) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
// make sure webhook id is set
|
|
||||||
if (m_settings[event_type_id].discord_webhook_id == 0) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
// ensure there is a matching webhook to begin with
|
|
||||||
if (!LogSys.GetDiscordWebhooks()[m_settings[event_type_id].discord_webhook_id].webhook_url.empty()) {
|
|
||||||
return LogSys.GetDiscordWebhooks()[m_settings[event_type_id].discord_webhook_id].webhook_url;
|
|
||||||
}
|
|
||||||
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
// GM_COMMAND | [x] Implemented Formatter
|
|
||||||
// ZONING | [x] Implemented Formatter
|
|
||||||
// AA_GAIN | [x] Implemented Formatter
|
|
||||||
// AA_PURCHASE | [x] Implemented Formatter
|
|
||||||
// FORAGE_SUCCESS | [x] Implemented Formatter
|
|
||||||
// FORAGE_FAILURE | [x] Implemented Formatter
|
|
||||||
// FISH_SUCCESS | [x] Implemented Formatter
|
|
||||||
// FISH_FAILURE | [x] Implemented Formatter
|
|
||||||
// ITEM_DESTROY | [x] Implemented Formatter
|
|
||||||
// WENT_ONLINE | [x] Implemented Formatter
|
|
||||||
// WENT_OFFLINE | [x] Implemented Formatter
|
|
||||||
// LEVEL_GAIN | [x] Implemented Formatter
|
|
||||||
// LEVEL_LOSS | [x] Implemented Formatter
|
|
||||||
// LOOT_ITEM | [x] Implemented Formatter
|
|
||||||
// MERCHANT_PURCHASE | [x] Implemented Formatter
|
|
||||||
// MERCHANT_SELL | [x] Implemented Formatter
|
|
||||||
// GROUP_JOIN | [] Implemented Formatter
|
|
||||||
// GROUP_LEAVE | [] Implemented Formatter
|
|
||||||
// RAID_JOIN | [] Implemented Formatter
|
|
||||||
// RAID_LEAVE | [] Implemented Formatter
|
|
||||||
// GROUNDSPAWN_PICKUP | [x] Implemented Formatter
|
|
||||||
// NPC_HANDIN | [x] Implemented Formatter
|
|
||||||
// SKILL_UP | [x] Implemented Formatter
|
|
||||||
// TASK_ACCEPT | [x] Implemented Formatter
|
|
||||||
// TASK_UPDATE | [x] Implemented Formatter
|
|
||||||
// TASK_COMPLETE | [x] Implemented Formatter
|
|
||||||
// TRADE | [] Implemented Formatter
|
|
||||||
// GIVE_ITEM | [] Implemented Formatter
|
|
||||||
// SAY | [x] Implemented Formatter
|
|
||||||
// REZ_ACCEPTED | [x] Implemented Formatter
|
|
||||||
// DEATH | [x] Implemented Formatter
|
|
||||||
// COMBINE_FAILURE | [x] Implemented Formatter
|
|
||||||
// COMBINE_SUCCESS | [x] Implemented Formatter
|
|
||||||
// DROPPED_ITEM | [x] Implemented Formatter
|
|
||||||
// SPLIT_MONEY | [x] Implemented Formatter
|
|
||||||
// DZ_JOIN | [] Implemented Formatter
|
|
||||||
// DZ_LEAVE | [] Implemented Formatter
|
|
||||||
// TRADER_PURCHASE | [x] Implemented Formatter
|
|
||||||
// TRADER_SELL | [x] Implemented Formatter
|
|
||||||
// BANDOLIER_CREATE | [] Implemented Formatter
|
|
||||||
// BANDOLIER_SWAP | [] Implemented Formatter
|
|
||||||
// DISCOVER_ITEM | [X] Implemented Formatter
|
|
||||||
|
|
||||||
std::string PlayerEventLogs::GetDiscordPayloadFromEvent(const PlayerEvent::PlayerEventContainer &e)
|
|
||||||
{
|
|
||||||
std::string payload;
|
|
||||||
switch (e.player_event_log.event_type_id) {
|
|
||||||
case PlayerEvent::AA_GAIN: {
|
|
||||||
PlayerEvent::AAGainedEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatAAGainedEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::AA_PURCHASE: {
|
|
||||||
PlayerEvent::AAPurchasedEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatAAPurchasedEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::COMBINE_FAILURE:
|
|
||||||
case PlayerEvent::COMBINE_SUCCESS: {
|
|
||||||
PlayerEvent::CombineEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatCombineEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::DEATH: {
|
|
||||||
PlayerEvent::DeathEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatDeathEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::DISCOVER_ITEM: {
|
|
||||||
PlayerEvent::DiscoverItemEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatDiscoverItemEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::DROPPED_ITEM: {
|
|
||||||
PlayerEvent::DroppedItemEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatDroppedItemEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::FISH_FAILURE:
|
|
||||||
case PlayerEvent::FORAGE_FAILURE:
|
|
||||||
case PlayerEvent::WENT_ONLINE:
|
|
||||||
case PlayerEvent::WENT_OFFLINE: {
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatWithNodata(e);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::FISH_SUCCESS: {
|
|
||||||
PlayerEvent::FishSuccessEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatFishSuccessEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::FORAGE_SUCCESS: {
|
|
||||||
PlayerEvent::ForageSuccessEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatForageSuccessEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::ITEM_DESTROY: {
|
|
||||||
PlayerEvent::DestroyItemEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatDestroyItemEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::LEVEL_GAIN: {
|
|
||||||
PlayerEvent::LevelGainedEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatLevelGainedEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::LEVEL_LOSS: {
|
|
||||||
PlayerEvent::LevelLostEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatLevelLostEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::LOOT_ITEM: {
|
|
||||||
PlayerEvent::LootItemEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatLootItemEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::GROUNDSPAWN_PICKUP: {
|
|
||||||
PlayerEvent::GroundSpawnPickupEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatGroundSpawnPickupEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::NPC_HANDIN: {
|
|
||||||
PlayerEvent::HandinEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatNPCHandinEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::SAY: {
|
|
||||||
PlayerEvent::SayEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatEventSay(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::GM_COMMAND: {
|
|
||||||
PlayerEvent::GMCommandEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatGMCommand(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::SKILL_UP: {
|
|
||||||
PlayerEvent::SkillUpEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatSkillUpEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::SPLIT_MONEY: {
|
|
||||||
PlayerEvent::SplitMoneyEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatSplitMoneyEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TASK_ACCEPT: {
|
|
||||||
PlayerEvent::TaskAcceptEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTaskAcceptEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TASK_COMPLETE: {
|
|
||||||
PlayerEvent::TaskCompleteEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTaskCompleteEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TASK_UPDATE: {
|
|
||||||
PlayerEvent::TaskUpdateEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTaskUpdateEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TRADE: {
|
|
||||||
PlayerEvent::TradeEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTradeEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TRADER_PURCHASE: {
|
|
||||||
PlayerEvent::TraderPurchaseEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTraderPurchaseEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::TRADER_SELL: {
|
|
||||||
PlayerEvent::TraderSellEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatTraderSellEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::REZ_ACCEPTED: {
|
|
||||||
PlayerEvent::ResurrectAcceptEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatResurrectAcceptEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::MERCHANT_PURCHASE: {
|
|
||||||
PlayerEvent::MerchantPurchaseEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatMerchantPurchaseEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::MERCHANT_SELL: {
|
|
||||||
PlayerEvent::MerchantSellEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatMerchantSellEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PlayerEvent::ZONING: {
|
|
||||||
PlayerEvent::ZoningEvent n{};
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
ss << e.player_event_log.event_data;
|
|
||||||
cereal::JSONInputArchive ar(ss);
|
|
||||||
n.serialize(ar);
|
|
||||||
}
|
|
||||||
|
|
||||||
payload = PlayerEventDiscordFormatter::FormatZoningEvent(e, n);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default: {
|
|
||||||
LogInfo(
|
|
||||||
"Player event [{}] ({}) Discord formatter not implemented",
|
|
||||||
e.player_event_log.event_type_name,
|
|
||||||
e.player_event_log.event_type_id
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return payload;
|
|
||||||
}
|
|
||||||
|
|
||||||
// general process function, used in world or QS depending on rule Logging:PlayerEventsQSProcess
|
|
||||||
void PlayerEventLogs::Process()
|
|
||||||
{
|
|
||||||
if (m_process_batch_events_timer.Check() || m_record_batch_queue.size() >= RuleI(Logging, BatchPlayerEventProcessChunkSize)) {
|
|
||||||
ProcessBatchQueue();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_process_retention_truncation_timer.Check()) {
|
|
||||||
ProcessRetentionTruncation();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void PlayerEventLogs::ProcessRetentionTruncation()
|
|
||||||
{
|
|
||||||
LogPlayerEvents("Running truncation");
|
|
||||||
|
|
||||||
for (int i = PlayerEvent::GM_COMMAND; i != PlayerEvent::MAX; i++) {
|
|
||||||
if (m_settings[i].retention_days > 0) {
|
|
||||||
int deleted_count = PlayerEventLogsRepository::DeleteWhere(
|
|
||||||
*m_database,
|
|
||||||
fmt::format(
|
|
||||||
"event_type_id = {} AND created_at < (NOW() - INTERVAL {} DAY)",
|
|
||||||
i,
|
|
||||||
m_settings[i].retention_days
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
if (deleted_count > 0) {
|
|
||||||
LogInfo(
|
|
||||||
"Truncated [{}] events of type [{}] ({}) older than [{}] days",
|
|
||||||
deleted_count,
|
|
||||||
PlayerEvent::EventName[i],
|
|
||||||
i,
|
|
||||||
m_settings[i].retention_days
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void PlayerEventLogs::ReloadSettings()
|
|
||||||
{
|
|
||||||
for (auto &e: PlayerEventLogSettingsRepository::All(*m_database)) {
|
|
||||||
if (e.id >= PlayerEvent::MAX || e.id < 0) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_settings[e.id] = e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const int32_t RETENTION_DAYS_DEFAULT = 7;
|
|
||||||
|
|
||||||
void PlayerEventLogs::SetSettingsDefaults()
|
|
||||||
{
|
|
||||||
m_settings[PlayerEvent::GM_COMMAND].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::ZONING].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::AA_GAIN].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::AA_PURCHASE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::FORAGE_SUCCESS].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::FORAGE_FAILURE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::FISH_SUCCESS].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::FISH_FAILURE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::ITEM_DESTROY].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::WENT_ONLINE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::WENT_OFFLINE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::LEVEL_GAIN].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::LEVEL_LOSS].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::LOOT_ITEM].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::MERCHANT_PURCHASE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::MERCHANT_SELL].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::GROUP_JOIN].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::GROUP_LEAVE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::RAID_JOIN].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::RAID_LEAVE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::GROUNDSPAWN_PICKUP].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::NPC_HANDIN].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::SKILL_UP].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::TASK_ACCEPT].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::TASK_UPDATE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::TASK_COMPLETE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::TRADE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::GIVE_ITEM].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::SAY].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::REZ_ACCEPTED].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::DEATH].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::COMBINE_FAILURE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::COMBINE_SUCCESS].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::DROPPED_ITEM].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::SPLIT_MONEY].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::DZ_JOIN].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::DZ_LEAVE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::TRADER_PURCHASE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::TRADER_SELL].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::BANDOLIER_CREATE].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::BANDOLIER_SWAP].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::DISCOVER_ITEM].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::POSSIBLE_HACK].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::KILLED_NPC].event_enabled = 0;
|
|
||||||
m_settings[PlayerEvent::KILLED_NAMED_NPC].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::KILLED_RAID_NPC].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::ITEM_CREATION].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::GUILD_TRIBUTE_DONATE_ITEM].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::GUILD_TRIBUTE_DONATE_PLAT].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::PARCEL_SEND].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::PARCEL_RETRIEVE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::PARCEL_DELETE].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::BARTER_TRANSACTION].event_enabled = 1;
|
|
||||||
m_settings[PlayerEvent::EVOLVE_ITEM].event_enabled = 1;
|
|
||||||
|
|
||||||
for (int i = PlayerEvent::GM_COMMAND; i != PlayerEvent::MAX; i++) {
|
|
||||||
m_settings[i].retention_days = RETENTION_DAYS_DEFAULT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
#ifndef EQEMU_PLAYER_EVENT_LOGS_H
|
|
||||||
#define EQEMU_PLAYER_EVENT_LOGS_H
|
|
||||||
|
|
||||||
#include "../repositories/player_event_log_settings_repository.h"
|
|
||||||
#include "player_events.h"
|
|
||||||
#include "../servertalk.h"
|
|
||||||
#include "../repositories/player_event_logs_repository.h"
|
|
||||||
#include "../timer.h"
|
|
||||||
#include "../json/json_archive_single_line.h"
|
|
||||||
#include <cereal/archives/json.hpp>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
class PlayerEventLogs {
|
|
||||||
public:
|
|
||||||
void Init();
|
|
||||||
void ReloadSettings();
|
|
||||||
PlayerEventLogs *SetDatabase(Database *db);
|
|
||||||
bool ValidateDatabaseConnection();
|
|
||||||
bool IsEventEnabled(PlayerEvent::EventType event);
|
|
||||||
|
|
||||||
void Process();
|
|
||||||
|
|
||||||
// batch queue
|
|
||||||
void AddToQueue(const PlayerEventLogsRepository::PlayerEventLogs &logs);
|
|
||||||
|
|
||||||
// main event record generic function
|
|
||||||
// can ingest any struct event types
|
|
||||||
template<typename T>
|
|
||||||
std::unique_ptr<ServerPacket> RecordEvent(
|
|
||||||
PlayerEvent::EventType t,
|
|
||||||
const PlayerEvent::PlayerEvent &p,
|
|
||||||
T e
|
|
||||||
)
|
|
||||||
{
|
|
||||||
auto n = PlayerEventLogsRepository::NewEntity();
|
|
||||||
FillPlayerEvent(p, n);
|
|
||||||
n.event_type_id = t;
|
|
||||||
|
|
||||||
std::stringstream ss;
|
|
||||||
{
|
|
||||||
cereal::JSONOutputArchiveSingleLine ar(ss);
|
|
||||||
e.serialize(ar);
|
|
||||||
}
|
|
||||||
|
|
||||||
n.event_type_name = PlayerEvent::EventName[t];
|
|
||||||
n.event_data = Strings::Contains(ss.str(), "noop") ? "{}" : ss.str();
|
|
||||||
n.created_at = std::time(nullptr);
|
|
||||||
|
|
||||||
auto c = PlayerEvent::PlayerEventContainer{
|
|
||||||
.player_event = p,
|
|
||||||
.player_event_log = n
|
|
||||||
};
|
|
||||||
|
|
||||||
return BuildPlayerEventPacket(c);
|
|
||||||
}
|
|
||||||
|
|
||||||
[[nodiscard]] const PlayerEventLogSettingsRepository::PlayerEventLogSettings *GetSettings() const;
|
|
||||||
bool IsEventDiscordEnabled(int32_t event_type_id);
|
|
||||||
std::string GetDiscordWebhookUrlFromEventType(int32_t event_type_id);
|
|
||||||
|
|
||||||
static std::string GetDiscordPayloadFromEvent(const PlayerEvent::PlayerEventContainer &e);
|
|
||||||
private:
|
|
||||||
Database *m_database; // reference to database
|
|
||||||
PlayerEventLogSettingsRepository::PlayerEventLogSettings m_settings[PlayerEvent::EventType::MAX]{};
|
|
||||||
|
|
||||||
// batch queue is used to record events in batch
|
|
||||||
std::vector<PlayerEventLogsRepository::PlayerEventLogs> m_record_batch_queue{};
|
|
||||||
static void FillPlayerEvent(const PlayerEvent::PlayerEvent &p, PlayerEventLogsRepository::PlayerEventLogs &n);
|
|
||||||
static std::unique_ptr<ServerPacket>
|
|
||||||
BuildPlayerEventPacket(const PlayerEvent::PlayerEventContainer &e);
|
|
||||||
|
|
||||||
// timers
|
|
||||||
Timer m_process_batch_events_timer; // events processing timer
|
|
||||||
Timer m_process_retention_truncation_timer; // timer for truncating events based on retention settings
|
|
||||||
|
|
||||||
// processing
|
|
||||||
std::mutex m_batch_queue_lock{};
|
|
||||||
void ProcessBatchQueue();
|
|
||||||
void ProcessRetentionTruncation();
|
|
||||||
void SetSettingsDefaults();
|
|
||||||
};
|
|
||||||
|
|
||||||
extern PlayerEventLogs player_event_logs;
|
|
||||||
|
|
||||||
#endif //EQEMU_PLAYER_EVENT_LOGS_H
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,303 +0,0 @@
|
|||||||
#include "evolving_items.h"
|
|
||||||
#include "item_instance.h"
|
|
||||||
#include "events/player_event_logs.h"
|
|
||||||
#include "repositories/character_evolving_items_repository.h"
|
|
||||||
|
|
||||||
EvolvingItemsManager::EvolvingItemsManager()
|
|
||||||
{
|
|
||||||
m_db = nullptr;
|
|
||||||
m_content_db = nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EvolvingItemsManager::LoadEvolvingItems() const
|
|
||||||
{
|
|
||||||
auto const &results = ItemsEvolvingDetailsRepository::All(*m_content_db);
|
|
||||||
|
|
||||||
if (results.empty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::ranges::transform(
|
|
||||||
results.begin(),
|
|
||||||
results.end(),
|
|
||||||
std::inserter(
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache(),
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().end()
|
|
||||||
),
|
|
||||||
[](const ItemsEvolvingDetailsRepository::ItemsEvolvingDetails &x) {
|
|
||||||
return std::make_pair(x.item_id, x);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void EvolvingItemsManager::SetDatabase(Database *db)
|
|
||||||
{
|
|
||||||
m_db = db;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EvolvingItemsManager::SetContentDatabase(Database *db)
|
|
||||||
{
|
|
||||||
m_content_db = db;
|
|
||||||
}
|
|
||||||
|
|
||||||
double EvolvingItemsManager::CalculateProgression(const uint64 current_amount, const uint32 item_id)
|
|
||||||
{
|
|
||||||
if (!evolving_items_manager.GetEvolvingItemsCache().contains(item_id)) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return evolving_items_manager.GetEvolvingItemsCache().at(item_id).required_amount > 0
|
|
||||||
? static_cast<double>(current_amount)
|
|
||||||
/ static_cast<double>(evolving_items_manager.GetEvolvingItemsCache().at(item_id).required_amount) * 100
|
|
||||||
: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EvolvingItemsManager::DoLootChecks(const uint32 char_id, const uint16 slot_id, const EQ::ItemInstance &inst) const
|
|
||||||
{
|
|
||||||
inst.SetEvolveEquipped(false);
|
|
||||||
if (inst.IsEvolving() && slot_id <= EQ::invslot::EQUIPMENT_END && slot_id >= EQ::invslot::EQUIPMENT_BEGIN) {
|
|
||||||
inst.SetEvolveEquipped(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!inst.IsEvolving()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!inst.GetEvolveUniqueID()) {
|
|
||||||
auto e = CharacterEvolvingItemsRepository::NewEntity();
|
|
||||||
|
|
||||||
e.character_id = char_id;
|
|
||||||
e.item_id = inst.GetID();
|
|
||||||
e.equipped = inst.GetEvolveEquipped();
|
|
||||||
e.final_item_id = evolving_items_manager.GetFinalItemID(inst);
|
|
||||||
|
|
||||||
auto r = CharacterEvolvingItemsRepository::InsertOne(*m_db, e);
|
|
||||||
e.id = r.id;
|
|
||||||
|
|
||||||
inst.SetEvolveUniqueID(e.id);
|
|
||||||
inst.SetEvolveCharID(e.character_id);
|
|
||||||
inst.SetEvolveItemID(e.item_id);
|
|
||||||
inst.SetEvolveFinalItemID(e.final_item_id);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
CharacterEvolvingItemsRepository::SetEquipped(*m_db, inst.GetEvolveUniqueID(), inst.GetEvolveEquipped());
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 EvolvingItemsManager::GetFinalItemID(const EQ::ItemInstance &inst) const
|
|
||||||
{
|
|
||||||
const auto start_iterator = std::ranges::find_if(
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().cbegin(),
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().cend(),
|
|
||||||
[&](const std::pair<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> &a) {
|
|
||||||
return a.second.item_evo_id == inst.GetEvolveLoreID();
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (start_iterator == std::end(evolving_items_manager.GetEvolvingItemsCache())) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto final_id = std::ranges::max_element(
|
|
||||||
start_iterator,
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().cend(),
|
|
||||||
[&](
|
|
||||||
const std::pair<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> &a,
|
|
||||||
const std::pair<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> &b
|
|
||||||
) {
|
|
||||||
return a.second.item_evo_id == b.second.item_evo_id &&
|
|
||||||
a.second.item_evolve_level < b.second.item_evolve_level;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
return final_id->first;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 EvolvingItemsManager::GetNextEvolveItemID(const EQ::ItemInstance &inst) const
|
|
||||||
{
|
|
||||||
int8 const current_level = inst.GetEvolveLvl();
|
|
||||||
|
|
||||||
const auto iterator = std::ranges::find_if(
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().cbegin(),
|
|
||||||
evolving_items_manager.GetEvolvingItemsCache().cend(),
|
|
||||||
[&](const std::pair<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> &a) {
|
|
||||||
return a.second.item_evo_id == inst.GetEvolveLoreID() &&
|
|
||||||
a.second.item_evolve_level == current_level + 1;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (iterator == std::end(evolving_items_manager.GetEvolvingItemsCache())) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return iterator->first;
|
|
||||||
}
|
|
||||||
|
|
||||||
ItemsEvolvingDetailsRepository::ItemsEvolvingDetails EvolvingItemsManager::GetEvolveItemDetails(const uint64 unique_id)
|
|
||||||
{
|
|
||||||
if (GetEvolvingItemsCache().contains(unique_id)) {
|
|
||||||
return GetEvolvingItemsCache().at(unique_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ItemsEvolvingDetailsRepository::NewEntity();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> EvolvingItemsManager::GetEvolveIDItems(
|
|
||||||
const uint32 evolve_id
|
|
||||||
)
|
|
||||||
{
|
|
||||||
std::vector<ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> e{};
|
|
||||||
|
|
||||||
for (auto const &[key, value]: GetEvolvingItemsCache()) {
|
|
||||||
if (value.item_evo_id == evolve_id) {
|
|
||||||
e.push_back(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::ranges::sort(
|
|
||||||
e.begin(),
|
|
||||||
e.end(),
|
|
||||||
[&](
|
|
||||||
ItemsEvolvingDetailsRepository::ItemsEvolvingDetails const &a,
|
|
||||||
ItemsEvolvingDetailsRepository::ItemsEvolvingDetails const &b
|
|
||||||
) {
|
|
||||||
return a.item_evolve_level < b.item_evolve_level;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
return e;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint64 EvolvingItemsManager::GetTotalEarnedXP(const EQ::ItemInstance &inst)
|
|
||||||
{
|
|
||||||
if (!inst) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint64 xp = inst.GetEvolveCurrentAmount();
|
|
||||||
auto evolve_id_item_cache = GetEvolveIDItems(inst.GetEvolveLoreID());
|
|
||||||
auto current_level = inst.GetEvolveLvl();
|
|
||||||
|
|
||||||
for (auto const &i: evolve_id_item_cache) {
|
|
||||||
if (i.item_evolve_level < current_level) {
|
|
||||||
xp += i.required_amount;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return xp;
|
|
||||||
}
|
|
||||||
|
|
||||||
EvolveGetNextItem EvolvingItemsManager::GetNextItemByXP(const EQ::ItemInstance &inst_in, const int64 in_xp)
|
|
||||||
{
|
|
||||||
EvolveGetNextItem ets{};
|
|
||||||
const auto evolve_items = GetEvolveIDItems(inst_in.GetEvolveLoreID());
|
|
||||||
uint32 max_transfer_level = 0;
|
|
||||||
int64 xp = in_xp;
|
|
||||||
|
|
||||||
for (auto const &e: evolve_items) {
|
|
||||||
if (e.item_evolve_level < inst_in.GetEvolveLvl()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
int64 have = 0;
|
|
||||||
if (e.item_evolve_level == inst_in.GetEvolveLvl()) {
|
|
||||||
have = inst_in.GetEvolveCurrentAmount();
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto required = e.required_amount;
|
|
||||||
const int64 need = required - have;
|
|
||||||
const int64 balance = xp - need;
|
|
||||||
|
|
||||||
if (balance <= 0) {
|
|
||||||
ets.new_current_amount = have + xp;
|
|
||||||
ets.new_item_id = e.item_id;
|
|
||||||
ets.from_current_amount = 0;
|
|
||||||
ets.max_transfer_level = max_transfer_level;
|
|
||||||
return ets;
|
|
||||||
}
|
|
||||||
|
|
||||||
xp = balance;
|
|
||||||
max_transfer_level += 1;
|
|
||||||
|
|
||||||
ets.new_current_amount = required;
|
|
||||||
ets.new_item_id = e.item_id;
|
|
||||||
ets.from_current_amount = balance - required;
|
|
||||||
ets.max_transfer_level = max_transfer_level;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ets;
|
|
||||||
}
|
|
||||||
|
|
||||||
EvolveTransfer EvolvingItemsManager::DetermineTransferResults(
|
|
||||||
const EQ::ItemInstance &inst_from,
|
|
||||||
const EQ::ItemInstance &inst_to
|
|
||||||
)
|
|
||||||
{
|
|
||||||
EvolveTransfer ets{};
|
|
||||||
|
|
||||||
auto evolving_details_inst_from = evolving_items_manager.GetEvolveItemDetails(inst_from.GetID());
|
|
||||||
auto evolving_details_inst_to = evolving_items_manager.GetEvolveItemDetails(inst_to.GetID());
|
|
||||||
|
|
||||||
if (!evolving_details_inst_from.id || !evolving_details_inst_to.id) {
|
|
||||||
return ets;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (evolving_details_inst_from.type == evolving_details_inst_to.type) {
|
|
||||||
uint32 compatibility = 0;
|
|
||||||
uint64 xp = 0;
|
|
||||||
if (evolving_details_inst_from.sub_type == evolving_details_inst_to.sub_type) {
|
|
||||||
compatibility = 100;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
compatibility = 30;
|
|
||||||
}
|
|
||||||
|
|
||||||
xp = evolving_items_manager.GetTotalEarnedXP(inst_from) * compatibility / 100;
|
|
||||||
auto results = evolving_items_manager.GetNextItemByXP(inst_to, xp);
|
|
||||||
|
|
||||||
ets.item_from_id = evolving_items_manager.GetFirstItemInLoreGroup(inst_from.GetEvolveLoreID());
|
|
||||||
ets.item_from_current_amount = results.from_current_amount;
|
|
||||||
ets.item_to_id = results.new_item_id;
|
|
||||||
ets.item_to_current_amount = results.new_current_amount;
|
|
||||||
ets.compatibility = compatibility;
|
|
||||||
ets.max_transfer_level = results.max_transfer_level;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ets;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 EvolvingItemsManager::GetFirstItemInLoreGroup(const uint32 lore_id)
|
|
||||||
{
|
|
||||||
for (auto const &[key, value]: GetEvolvingItemsCache()) {
|
|
||||||
if (value.item_evo_id == lore_id && value.item_evolve_level == 1) {
|
|
||||||
return key;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32 EvolvingItemsManager::GetFirstItemInLoreGroupByItemID(const uint32 item_id)
|
|
||||||
{
|
|
||||||
for (auto const &[key, value]: GetEvolvingItemsCache()) {
|
|
||||||
if (value.item_id == item_id) {
|
|
||||||
for (auto const &[key2, value2]: GetEvolvingItemsCache()) {
|
|
||||||
if (value2.item_evo_id == value.item_evo_id && value2.item_evolve_level == 1) {
|
|
||||||
return key;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EvolvingItemsManager::LoadPlayerEvent(const EQ::ItemInstance &inst, PlayerEvent::EvolveItem &e)
|
|
||||||
{
|
|
||||||
e.item_id = inst.GetID();
|
|
||||||
e.item_name = inst.GetItem() ? inst.GetItem()->Name : std::string();
|
|
||||||
e.level = inst.GetEvolveLvl();
|
|
||||||
e.progression = inst.GetEvolveProgression();
|
|
||||||
e.unique_id = inst.GetEvolveUniqueID();
|
|
||||||
}
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
#ifndef EVOLVING_H
|
|
||||||
#define EVOLVING_H
|
|
||||||
|
|
||||||
#include "shareddb.h"
|
|
||||||
#include "events/player_events.h"
|
|
||||||
#include "repositories/items_evolving_details_repository.h"
|
|
||||||
|
|
||||||
namespace EQ {
|
|
||||||
class ItemInstance;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace EvolvingItems {
|
|
||||||
namespace Actions {
|
|
||||||
constexpr int8 UPDATE_ITEMS = 0;
|
|
||||||
constexpr int8 TRANSFER_WINDOW_OPEN = 1;
|
|
||||||
constexpr int8 TRANSFER_WINDOW_DETAILS = 2;
|
|
||||||
constexpr int8 TRANSFER_XP = 3;
|
|
||||||
constexpr int8 FINAL_RESULT = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Types {
|
|
||||||
constexpr int8 AMOUNT_OF_EXP = 1;
|
|
||||||
constexpr int8 NUMBER_OF_KILLS = 2;
|
|
||||||
constexpr int8 SPECIFIC_MOB_RACE = 3;
|
|
||||||
constexpr int8 SPECIFIC_ZONE_ID = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace SubTypes {
|
|
||||||
constexpr int8 ALL_EXP = 0;
|
|
||||||
constexpr int8 SOLO_EXP = 1;
|
|
||||||
constexpr int8 GROUP_EXP = 2;
|
|
||||||
constexpr int8 RAID_EXP = 3;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class EvolvingItemsManager
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
EvolvingItemsManager();
|
|
||||||
void SetDatabase(Database *db);
|
|
||||||
void SetContentDatabase(Database *db);
|
|
||||||
|
|
||||||
void LoadEvolvingItems() const;
|
|
||||||
void DoLootChecks(uint32 char_id, uint16 slot_id, const EQ::ItemInstance &inst) const;
|
|
||||||
uint32 GetFinalItemID(const EQ::ItemInstance &inst) const;
|
|
||||||
uint32 GetNextEvolveItemID(const EQ::ItemInstance &inst) const;
|
|
||||||
uint32 GetFirstItemInLoreGroup(uint32 lore_id);
|
|
||||||
uint32 GetFirstItemInLoreGroupByItemID(uint32 item_id);
|
|
||||||
uint64 GetTotalEarnedXP(const EQ::ItemInstance &inst);
|
|
||||||
static double CalculateProgression(uint64 current_amount, uint32 item_id);
|
|
||||||
static void LoadPlayerEvent(const EQ::ItemInstance &inst, PlayerEvent::EvolveItem &e);
|
|
||||||
|
|
||||||
ItemsEvolvingDetailsRepository::ItemsEvolvingDetails GetEvolveItemDetails(uint64 id);
|
|
||||||
EvolveTransfer DetermineTransferResults(const EQ::ItemInstance& inst_from, const EQ::ItemInstance& inst_to);
|
|
||||||
EvolveGetNextItem GetNextItemByXP(const EQ::ItemInstance &inst_in, int64 in_xp);
|
|
||||||
std::map<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails>& GetEvolvingItemsCache() { return evolving_items_cache; }
|
|
||||||
std::vector<ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> GetEvolveIDItems(uint32 evolve_id);
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::map<uint32, ItemsEvolvingDetailsRepository::ItemsEvolvingDetails> evolving_items_cache;
|
|
||||||
Database * m_db;
|
|
||||||
Database * m_content_db;
|
|
||||||
};
|
|
||||||
|
|
||||||
extern EvolvingItemsManager evolving_items_manager;
|
|
||||||
|
|
||||||
#endif //EVOLVING_H
|
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
#include "expedition_base.h"
|
||||||
|
#include "repositories/expeditions_repository.h"
|
||||||
|
|
||||||
|
ExpeditionBase::ExpeditionBase(uint32_t id, const std::string& uuid,
|
||||||
|
const std::string& expedition_name, const DynamicZoneMember& leader
|
||||||
|
) :
|
||||||
|
m_id(id),
|
||||||
|
m_uuid(uuid),
|
||||||
|
m_expedition_name(expedition_name),
|
||||||
|
m_leader(leader)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExpeditionBase::LoadRepositoryResult(ExpeditionsRepository::ExpeditionWithLeader&& entry)
|
||||||
|
{
|
||||||
|
m_id = entry.id;
|
||||||
|
m_uuid = std::move(entry.uuid);
|
||||||
|
m_expedition_name = std::move(entry.expedition_name);
|
||||||
|
m_add_replay_on_join = entry.add_replay_on_join;
|
||||||
|
m_is_locked = entry.is_locked;
|
||||||
|
m_leader.id = entry.leader_id;
|
||||||
|
m_leader.name = std::move(entry.leader_name);
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user