Compare commits
No commits in common. "master" and "1.3.0" have entirely different histories.
319 changed files with 200792 additions and 16865 deletions
6
.gitignore
vendored
6
.gitignore
vendored
|
@ -1,11 +1,7 @@
|
||||||
.idea
|
|
||||||
/target
|
/target
|
||||||
/hotpatch.toml
|
/hotpatch.toml
|
||||||
/configserver.toml
|
/configserver.toml
|
||||||
/loginserver.toml
|
/loginserver.toml
|
||||||
/gateway.toml
|
/gateway.toml
|
||||||
/gameserver.toml
|
/gameserver.toml
|
||||||
/wicked-waifus-protocol-internal/generated
|
/shorekeeper-protocol/generated
|
||||||
/wicked-waifus-protocol/generated
|
|
||||||
/data/assets/config-server
|
|
||||||
/data/assets/game-data
|
|
||||||
|
|
0
.gitmodules
vendored
0
.gitmodules
vendored
1938
Cargo.lock
generated
1938
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
29
Cargo.toml
29
Cargo.toml
|
@ -1,5 +1,5 @@
|
||||||
[workspace]
|
[workspace]
|
||||||
members = ["wicked-waifus-asset-updater", "wicked-waifus-commons", "wicked-waifus-config-server", "wicked-waifus-hotpatch-server", "wicked-waifus-login-server", "wicked-waifus-gateway-server", "wicked-waifus-gateway-server/kcp", "wicked-waifus-database", "wicked-waifus-http", "wicked-waifus-protokey", "wicked-waifus-protocol-internal", "wicked-waifus-game-server", "wicked-waifus-network", "wicked-waifus-data"]
|
members = ["common", "config-server", "hotpatch-server", "login-server", "gateway-server", "gateway-server/kcp", "shorekeeper-database", "shorekeeper-http", "shorekeeper-protokey", "shorekeeper-protocol", "shorekeeper-protocol/shorekeeper-protocol-derive", "game-server", "shorekeeper-network", "shorekeeper-data"]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
|
|
||||||
[workspace.package]
|
[workspace.package]
|
||||||
|
@ -8,7 +8,6 @@ version = "0.1.0"
|
||||||
[workspace.dependencies]
|
[workspace.dependencies]
|
||||||
# Framework
|
# Framework
|
||||||
tokio = { version = "1.39.3", features = ["full"] }
|
tokio = { version = "1.39.3", features = ["full"] }
|
||||||
tower-http = { version = "0.6.1", features = ["fs", "trace"] }
|
|
||||||
axum = "0.7.5"
|
axum = "0.7.5"
|
||||||
axum-server = "0.7.1"
|
axum-server = "0.7.1"
|
||||||
zeromq = { version = "0.4.0", default-features = false, features = ["tokio-runtime", "tcp-transport"] }
|
zeromq = { version = "0.4.0", default-features = false, features = ["tokio-runtime", "tcp-transport"] }
|
||||||
|
@ -20,14 +19,12 @@ sqlx = { version = "0.8.2", features = ["postgres", "runtime-tokio-rustls"] }
|
||||||
aes = "0.8.4"
|
aes = "0.8.4"
|
||||||
cbc = { version = "0.1.2", features = ["alloc"] }
|
cbc = { version = "0.1.2", features = ["alloc"] }
|
||||||
cipher = "0.4.4"
|
cipher = "0.4.4"
|
||||||
crc32fast = "1.4.2"
|
|
||||||
rand = "0.8.5"
|
rand = "0.8.5"
|
||||||
rsa = { version = "0.9.6", features = ["pem"] }
|
rsa = { version = "0.9.6", features = ["pem"] }
|
||||||
|
|
||||||
# Serialization
|
# Serialization
|
||||||
serde = { version = "1.0.209", features = ["derive"] }
|
serde = { version = "1.0.209", features = ["derive"] }
|
||||||
serde_json = "1.0.128"
|
serde_json = "1.0.128"
|
||||||
serde_repr = "0.1.19"
|
|
||||||
toml = "0.8.19"
|
toml = "0.8.19"
|
||||||
prost = "0.13.2"
|
prost = "0.13.2"
|
||||||
prost-build = "0.13.2"
|
prost-build = "0.13.2"
|
||||||
|
@ -40,24 +37,22 @@ rbase64 = "2.0.3"
|
||||||
dashmap = "6.1.0"
|
dashmap = "6.1.0"
|
||||||
hex = "0.4.3"
|
hex = "0.4.3"
|
||||||
byteorder = "1.5.0"
|
byteorder = "1.5.0"
|
||||||
|
crc32fast = "1.4.2"
|
||||||
|
|
||||||
# Tracing
|
# Tracing
|
||||||
tracing = "0.1.40"
|
tracing = "0.1.40"
|
||||||
tracing-subscriber = { version = "0.3.18", features = ["env-filter"] }
|
tracing-subscriber = "0.3.18"
|
||||||
|
|
||||||
# Internal
|
# Internal
|
||||||
kcp = { path = "wicked-waifus-gateway-server/kcp" }
|
kcp = { path = "gateway-server/kcp" }
|
||||||
wicked-waifus-asset-updater = { path = "wicked-waifus-asset-updater" }
|
common = { path = "common/" }
|
||||||
wicked-waifus-commons = { path = "wicked-waifus-commons" }
|
shorekeeper-http = { path = "shorekeeper-http/" }
|
||||||
wicked-waifus-http = { path = "wicked-waifus-http" }
|
shorekeeper-data = { path = "shorekeeper-data/" }
|
||||||
wicked-waifus-data = { path = "wicked-waifus-data" }
|
shorekeeper-database = { path = "shorekeeper-database/" }
|
||||||
wicked-waifus-database = { path = "wicked-waifus-database" }
|
shorekeeper-network = { path = "shorekeeper-network/" }
|
||||||
wicked-waifus-network = { path = "wicked-waifus-network" }
|
shorekeeper-protocol = { path = "shorekeeper-protocol/" }
|
||||||
wicked-waifus-protocol-internal = { path = "wicked-waifus-protocol-internal" }
|
shorekeeper-protocol-derive = { path = "shorekeeper-protocol/shorekeeper-protocol-derive" }
|
||||||
wicked-waifus-protokey = { path = "wicked-waifus-protokey" }
|
shorekeeper-protokey = { path = "shorekeeper-protokey/" }
|
||||||
|
|
||||||
wicked-waifus-protocol = { git = "https://git.xeondev.com/wickedwaifus/wicked-waifus-proto" }
|
|
||||||
wicked-waifus-protocol-derive = { git = "https://git.xeondev.com/wickedwaifus/wicked-waifus-proto" }
|
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
strip = true # Automatically strip symbols from the binary.
|
strip = true # Automatically strip symbols from the binary.
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
FROM rust:1.82-alpine3.20
|
FROM rust:1.81-alpine3.20
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
|
@ -2,5 +2,5 @@ FROM alpine:3.20
|
||||||
ARG MICROSERVICE
|
ARG MICROSERVICE
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=wicked-waifus-builder:2.1.0-SNAPSHOT /app/target/release/$MICROSERVICE ./service
|
COPY --from=shorekeeper-builder:1.3.0-SNAPSHOT /app/target/release/$MICROSERVICE ./service
|
||||||
CMD ["./service"]
|
CMD ["./service"]
|
35
README.md
35
README.md
|
@ -1,12 +1,9 @@
|
||||||
# Wicked Waifus
|
# Shorekeeper
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
## About
|
## About
|
||||||
**Wicked Waifus is an open-source Wuthering Waves server emulator written in Rust**.
|
**Shorekeeper is an open-source Wuthering Waves server emulator written in Rust**. The goal of this project is to ensure a clean, easy-to-understand code environment. Shorekeeper uses **tokio** for asynchronous networking operations, **axum** as http framework and **ZeroMQ** for communication between servers. It also implements **performant and extensible ECS** for emulation of the game world.
|
||||||
The goal of this project is to ensure a clean, easy-to-understand code environment.
|
|
||||||
Wicked Waifus uses **tokio** for asynchronous networking operations, **axum** as http framework and **ZeroMQ** for communication between servers.
|
|
||||||
It also implements **performant and extensible ECS** for emulation of the game world.
|
|
||||||
|
|
||||||
## Getting started
|
## Getting started
|
||||||
#### Requirements
|
#### Requirements
|
||||||
|
@ -18,13 +15,13 @@ It also implements **performant and extensible ECS** for emulation of the game w
|
||||||
##### a) building from sources
|
##### a) building from sources
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
git clone --recursive https://git.xeondev.com/wickedwaifus/wicked-waifus-rs.git
|
git clone https://git.xeondev.com/Shorekeeper/Shorekeeper.git
|
||||||
cd wicked-waifus-rs
|
cd Shorekeeper
|
||||||
cargo run --bin wicked-waifus-config-server
|
cargo run --bin config-server
|
||||||
cargo run --bin wicked-waifus-hotpatch-server
|
cargo run --bin hotpatch-server
|
||||||
cargo run --bin wicked-waifus-login-server
|
cargo run --bin login-server
|
||||||
cargo run --bin wicked-waifus-gateway-server
|
cargo run --bin gateway-server
|
||||||
cargo run --bin wicked-waifus-game-server
|
cargo run --bin game-server
|
||||||
```
|
```
|
||||||
|
|
||||||
##### b) building from sources(docker edition)
|
##### b) building from sources(docker edition)
|
||||||
|
@ -43,9 +40,9 @@ docker compose up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
##### c) using pre-built binaries
|
##### c) using pre-built binaries
|
||||||
Navigate to the [Releases](https://git.xeondev.com/wickedwaifus/wicked-waifus-rs/releases)
|
Navigate to the [Releases](https://git.xeondev.com/Shorekeeper/Shorekeeper/releases)
|
||||||
page and download the latest release for your platform.<br>
|
page and download the latest release for your platform.<br>
|
||||||
Launch all servers: `wicked-waifus-config-server`, `wicked-waifus-hotpatch-server`, `wicked-waifus-login-server`, `wicked-waifus-gateway-server`, `wicked-waifus-game-server`
|
Launch all servers: `config-server`, `hotpatch-server`, `login-server`, `gateway-server`, `game-server`
|
||||||
|
|
||||||
##### NOTE: you don't have to install Rust and Protoc if you're going to use pre-built binaries, although the preferred way is building from sources.<br>We don't provide any support for pre-built binaries.
|
##### NOTE: you don't have to install Rust and Protoc if you're going to use pre-built binaries, although the preferred way is building from sources.<br>We don't provide any support for pre-built binaries.
|
||||||
|
|
||||||
|
@ -60,15 +57,15 @@ You have to specify credentials for **PostgreSQL**<br>
|
||||||
host = "localhost:5432"
|
host = "localhost:5432"
|
||||||
user_name = "postgres"
|
user_name = "postgres"
|
||||||
password = ""
|
password = ""
|
||||||
db_name = "wicked_waifus_db"
|
db_name = "shorekeeper"
|
||||||
```
|
```
|
||||||
##### NOTE: don't forget to create database with specified `db_name` (default: `wicked_waifus_db`). For example, you can do so with PgAdmin.
|
##### NOTE: don't forget to create database with specified `db_name` (default: `shorekeeper`). For example, you can do so with PgAdmin.
|
||||||
|
|
||||||
#### Data
|
#### Data
|
||||||
The data files: Logic JSON collections (`data/assets/game-data/BinData`) and config/hotpatch indexes (`data/assets/config-server`, `data/assets/hotpatch-server`) are included in this repository. Keep in mind that you need to have the `data` subdirectory in current working directory.
|
The data files: Logic JSON collections (`assets/logic/json`) and config/hotpatch indexes (`assets/config`, `assets/hotpatch`) are included in this repository. Keep in mind that you need to have the `assets` subdirectory in current working directory.
|
||||||
|
|
||||||
#### Connecting
|
#### Connecting
|
||||||
You have to download client of Wuthering Waves Beta 2.1, apply the [wicked-waifus-win-patch](https://git.xeondev.com/wickedwaifus/wicked-waifus-win-patch/releases) and add necessary `.pak` files, which you can get here: [wicked-waifus-pak](https://git.xeondev.com/wickedwaifus/wicked-waifus-pak)
|
You have to download client of Wuthering Waves Beta 1.3, apply the [shorekeeper-patch](https://git.xeondev.com/xeon/shorekeeper-patch/releases) and add necessary `.pak` files, which you can get here: [shorekeeper-pak](https://git.xeondev.com/Shorekeeper/shorekeeper-pak)
|
||||||
|
|
||||||
### Troubleshooting
|
### Troubleshooting
|
||||||
[Visit our discord](https://discord.gg/reversedrooms) if you have any questions/issues
|
[Visit our discord](https://discord.gg/reversedrooms) if you have any questions/issues
|
||||||
|
|
52
assets/config/index.json
Normal file
52
assets/config/index.json
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
{
|
||||||
|
"default": {
|
||||||
|
"CdnUrl": [
|
||||||
|
{
|
||||||
|
"url": "http://127.0.0.1:10002/prod/client/",
|
||||||
|
"weight": "2323"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SecondaryUrl": [],
|
||||||
|
"PriceRatio": 1,
|
||||||
|
"SpeedRatio": 1,
|
||||||
|
"GachaUrl": {
|
||||||
|
"GachaRecord": "http://127.0.0.1:10001/gacha/record",
|
||||||
|
"GachaPoolDetail": "http://127.0.0.1:10001/gacha/detail"
|
||||||
|
},
|
||||||
|
"LogReport": {
|
||||||
|
"name": "pioneer-upload-log-1319073642",
|
||||||
|
"region": "dev-reversedrooms"
|
||||||
|
},
|
||||||
|
"PackageUpdateDescUrl": {
|
||||||
|
"MainUrl": "http://127.0.0.1:10001/force_update/UpdateDesc.html",
|
||||||
|
"SubUrl": "http://127.0.0.1:10001/force_update/UpdateDesc.html"
|
||||||
|
},
|
||||||
|
"PackageUpdateUrl": {
|
||||||
|
"MainUrl": "http://127.0.0.1:10001/force_update/UpdateJs.html",
|
||||||
|
"SubUrl": "http://127.0.0.1:10001/force_update/UpdateJs.html"
|
||||||
|
},
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "3e2e647670b7498fa645eb9574f78c2c",
|
||||||
|
"URL": "http://127.0.0.1:10001/TDCfg"
|
||||||
|
},
|
||||||
|
"GmOpen": false,
|
||||||
|
"IosAuditFirstDownloadTip": false,
|
||||||
|
"NoticeUrl": "http://127.0.0.1:10001/notice",
|
||||||
|
"MixUri": "rODM5DcqOhYsIOtsEuZWNGFa2guZgl57",
|
||||||
|
"ResUri": "rODM5DcqOhYsIOtsEuZWNGFa2guZgl57",
|
||||||
|
"LoginServers": [
|
||||||
|
{
|
||||||
|
"id": "f9e0fc655c1931bc03ad976e9fc14473",
|
||||||
|
"ip": "http://127.0.0.1:5500",
|
||||||
|
"name": "Shorekeeper"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PrivateServers": {
|
||||||
|
"enable": false,
|
||||||
|
"serverUrl": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"p1": {
|
||||||
|
"IosAuditFirstDownloadTip": false
|
||||||
|
}
|
||||||
|
}
|
64
assets/config/samples/cn_beta_1.3.0.json
Normal file
64
assets/config/samples/cn_beta_1.3.0.json
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
{
|
||||||
|
"default": {
|
||||||
|
"CdnUrl": [
|
||||||
|
{
|
||||||
|
"url": "https://cdn-huoshan-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "2323"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-qiniu-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-qcloud-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "5443"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-aliyun-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "0"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SecondaryUrl": [],
|
||||||
|
"PriceRatio": 1,
|
||||||
|
"SpeedRatio": 1,
|
||||||
|
"GachaUrl": {
|
||||||
|
"GachaRecord": "http://127.0.0.1:10001/gacha/record",
|
||||||
|
"GachaPoolDetail": "http://127.0.0.1:10001/gacha/detail"
|
||||||
|
},
|
||||||
|
"LogReport": {
|
||||||
|
"name": "pioneer-upload-log-1319073642",
|
||||||
|
"region": "dev-reversedrooms"
|
||||||
|
},
|
||||||
|
"PackageUpdateDescUrl": {
|
||||||
|
"MainUrl": "http://127.0.0.1:10001/force_update/UpdateDesc.html",
|
||||||
|
"SubUrl": "http://127.0.0.1:10001/force_update/UpdateDesc.html"
|
||||||
|
},
|
||||||
|
"PackageUpdateUrl": {
|
||||||
|
"MainUrl": "http://127.0.0.1:10001/force_update/UpdateJs.html",
|
||||||
|
"SubUrl": "http://127.0.0.1:10001/force_update/UpdateJs.html"
|
||||||
|
},
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "3e2e647670b7498fa645eb9574f78c2c",
|
||||||
|
"URL": "http://127.0.0.1:10001/TDCfg"
|
||||||
|
},
|
||||||
|
"GmOpen": false,
|
||||||
|
"IosAuditFirstDownloadTip": false,
|
||||||
|
"NoticeUrl": "http://127.0.0.1:10001/notice",
|
||||||
|
"MixUri": "rODM5DcqOhYsIOtsEuZWNGFa2guZgl57",
|
||||||
|
"ResUri": "rODM5DcqOhYsIOtsEuZWNGFa2guZgl57",
|
||||||
|
"LoginServers": [
|
||||||
|
{
|
||||||
|
"id": "f9e0fc655c1931bc03ad976e9fc14473",
|
||||||
|
"ip": "http://127.0.0.1:5500",
|
||||||
|
"name": "Shorekeeper"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PrivateServers": {
|
||||||
|
"enable": false,
|
||||||
|
"serverUrl": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"p1": {
|
||||||
|
"IosAuditFirstDownloadTip": false
|
||||||
|
}
|
||||||
|
}
|
64
assets/config/samples/cn_live_1.3.0.json
Normal file
64
assets/config/samples/cn_live_1.3.0.json
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
{
|
||||||
|
"default": {
|
||||||
|
"CdnUrl": [
|
||||||
|
{
|
||||||
|
"url": "https://cdn-huoshan-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-qiniu-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "4181"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-qcloud-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "2937"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-aliyun-cn-mc.aki-game.com/prod/client/",
|
||||||
|
"weight": "0"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SecondaryUrl": [],
|
||||||
|
"PriceRatio": 1,
|
||||||
|
"SpeedRatio": 1,
|
||||||
|
"GachaUrl": {
|
||||||
|
"GachaRecord": "https://aki-gm-resources.aki-game.com",
|
||||||
|
"GachaPoolDetail": "https://aki-gm-resources.aki-game.com"
|
||||||
|
},
|
||||||
|
"LogReport": {
|
||||||
|
"name": "aki-upload-log-1319073642",
|
||||||
|
"region": "ap-shanghai"
|
||||||
|
},
|
||||||
|
"PackageUpdateDescUrl": {
|
||||||
|
"MainUrl": "https://aki-gm-resources-back.aki-game.com/force_update/UpdateDesc.html",
|
||||||
|
"SubUrl": "https://aki-gm-resources-back-qiniu.aki-game.com/force_update/UpdateDesc.html"
|
||||||
|
},
|
||||||
|
"PackageUpdateUrl": {
|
||||||
|
"MainUrl": "https://aki-gm-resources-back.aki-game.com/force_update/UpdateJs.html",
|
||||||
|
"SubUrl": "https://aki-gm-resources-back-qiniu.aki-game.com/force_update/UpdateJs.html"
|
||||||
|
},
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "7be70c9c181f4fa8bc013e9b89d2b63b",
|
||||||
|
"URL": "http://cn-datareceiver.aki-game.com"
|
||||||
|
},
|
||||||
|
"GmOpen": false,
|
||||||
|
"IosAuditFirstDownloadTip": false,
|
||||||
|
"NoticeUrl": "https://aki-gm-resources-back.aki-game.com",
|
||||||
|
"MixUri": "e66lKLx7RnUf9QlkMCGaW0jdyjRV1weo",
|
||||||
|
"ResUri": "e66lKLx7RnUf9QlkMCGaW0jdyjRV1weo",
|
||||||
|
"LoginServers": [
|
||||||
|
{
|
||||||
|
"id": "76402e5b20be2c39f095a152090afddc",
|
||||||
|
"ip": "https://login-sh.aki-game.com",
|
||||||
|
"name": "鸣潮"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PrivateServers": {
|
||||||
|
"enable": false,
|
||||||
|
"serverUrl": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"p1": {
|
||||||
|
"IosAuditFirstDownloadTip": false
|
||||||
|
}
|
||||||
|
}
|
115
assets/config/samples/os_live_1.3.0.json
Normal file
115
assets/config/samples/os_live_1.3.0.json
Normal file
|
@ -0,0 +1,115 @@
|
||||||
|
{
|
||||||
|
"default": {
|
||||||
|
"CdnUrl": [
|
||||||
|
{
|
||||||
|
"url": "https://cdn-aws-hw-mc.aki-game.net/prod/client/",
|
||||||
|
"weight": "829"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-qcloud-hw-mc.aki-game.net/prod/client/",
|
||||||
|
"weight": "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-akamai-hw-mc.aki-game.net/prod/client/",
|
||||||
|
"weight": "262"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://cdn-huoshan-hw-mc.aki-game.net/prod/client/",
|
||||||
|
"weight": "829"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SecondaryUrl": [],
|
||||||
|
"PriceRatio": 1,
|
||||||
|
"SpeedRatio": 1,
|
||||||
|
"GachaUrl": {
|
||||||
|
"GachaRecord": "https://aki-gm-resources-oversea.aki-game.net",
|
||||||
|
"GachaPoolDetail": "https://aki-gm-resources-oversea.aki-game.net"
|
||||||
|
},
|
||||||
|
"LogReport": {
|
||||||
|
"name": "hw-aki-upload-log-1319073642",
|
||||||
|
"region": "ap-singapore"
|
||||||
|
},
|
||||||
|
"PackageUpdateDescUrl": {
|
||||||
|
"MainUrl": "https://aki-gm-resources-back.aki-game.net/force_update/UpdateDesc.html",
|
||||||
|
"SubUrl": "https://aki-gm-resources-back-aws.aki-game.net/force_update/UpdateDesc.html"
|
||||||
|
},
|
||||||
|
"PackageUpdateUrl": {
|
||||||
|
"MainUrl": "https://aki-gm-resources-back.aki-game.net/force_update/UpdateJs.html",
|
||||||
|
"SubUrl": "https://aki-gm-resources-back-aws.aki-game.net/force_update/UpdateJs.html"
|
||||||
|
},
|
||||||
|
"GARUrl": "https://gar-service.aki-game.net",
|
||||||
|
"TDCfg": {
|
||||||
|
"URL": "http://us-datareceiver.aki-game.net",
|
||||||
|
"AppID": "bf3f44edc6cf43c582e347ba660876c0"
|
||||||
|
},
|
||||||
|
"GmOpen": false,
|
||||||
|
"IosAuditFirstDownloadTip": false,
|
||||||
|
"NoticeUrl": "https://aki-gm-resources-back.aki-game.net",
|
||||||
|
"MixUri": "u4S3uXaif7gNKnNGnbZ6WnW67vWFmQgd",
|
||||||
|
"ResUri": "u4S3uXaif7gNKnNGnbZ6WnW67vWFmQgd",
|
||||||
|
"LoginServers": [
|
||||||
|
{
|
||||||
|
"PingUrl": "49.51.79.239",
|
||||||
|
"Region": "America",
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "12c7c870b57541caa2093e275212f083",
|
||||||
|
"URL": "http://us-datareceiver.aki-game.net"
|
||||||
|
},
|
||||||
|
"id": "591d6af3a3090d8ea00d8f86cf6d7501",
|
||||||
|
"ip": "https://login-us.aki-game.net",
|
||||||
|
"name": "America"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PingUrl": "124.156.236.245",
|
||||||
|
"Region": "Asia",
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "dc2e8c02609546e89d78c2eb2974becb",
|
||||||
|
"URL": "http://jp-datareceiver.aki-game.net"
|
||||||
|
},
|
||||||
|
"id": "86d52186155b148b5c138ceb41be9650",
|
||||||
|
"ip": "https://login-jp.aki-game.net",
|
||||||
|
"name": "Asia"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PingUrl": "49.51.129.88",
|
||||||
|
"Region": "Europe",
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "0fb77e4801db41a7a1fb1a3431625fbd",
|
||||||
|
"URL": "http://eu-datareceiver.aki-game.net"
|
||||||
|
},
|
||||||
|
"id": "6eb2a235b30d05efd77bedb5cf60999e",
|
||||||
|
"ip": "https://login-eu.aki-game.net",
|
||||||
|
"name": "Europe"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PingUrl": "43.129.150.38",
|
||||||
|
"Region": "HMT",
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "5ccd297c928d405a9e6b18a122f4d48d",
|
||||||
|
"URL": "http://hk-datareceiver.aki-game.net"
|
||||||
|
},
|
||||||
|
"id": "919752ae5ea09c1ced910dd668a63ffb",
|
||||||
|
"ip": "https://login-hk.aki-game.net",
|
||||||
|
"name": "HMT(HK, MO, TW)"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PingUrl": "101.33.100.22",
|
||||||
|
"Region": "SEA",
|
||||||
|
"TDCfg": {
|
||||||
|
"AppID": "dffd1143291b4877a8866fb24c623038",
|
||||||
|
"URL": "http://as-datareceiver.aki-game.net"
|
||||||
|
},
|
||||||
|
"id": "10cd7254d57e58ae560b15d51e34b4c8",
|
||||||
|
"ip": "https://login-sg.aki-game.net",
|
||||||
|
"name": "SEA"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PrivateServers": {
|
||||||
|
"enable": false,
|
||||||
|
"serverUrl": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"p1": {
|
||||||
|
"IosAuditFirstDownloadTip": false
|
||||||
|
}
|
||||||
|
}
|
60
assets/hotpatch/prod/Windows/config.json
Normal file
60
assets/hotpatch/prod/Windows/config.json
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
{
|
||||||
|
"PackageVersion": "1.3.0",
|
||||||
|
"LauncherVersion": "1.3.9",
|
||||||
|
"ResourceVersion": "1.3.9",
|
||||||
|
"LauncherIndexSha1": {
|
||||||
|
"1.3.1": "90FDF17EA0B4015D43C344CB7229E76AB32549DD",
|
||||||
|
"1.3.2": "C9A587AB1FA6CA57CD23E0FB3F0103BFDCAA8E37",
|
||||||
|
"1.3.3": "1C7AF02F13DBE69637DB43039E2FFB8C9AD9A04B",
|
||||||
|
"1.3.4": "DA50F315041E216568A7713074C6475F6AB4530E",
|
||||||
|
"1.3.5": "EA9C6F6D5E920F47F96D8F8BC366A4CED62A0346",
|
||||||
|
"1.3.6": "8CA7E6573A52B16CFAA29E996D389918B6829E7A",
|
||||||
|
"1.3.7": "FCAAED58E5983027A82F52C350418CCE7BD531D2",
|
||||||
|
"1.3.8": "91D6231B3F4C9A6605B79E23D0C02F9790DD6BCF",
|
||||||
|
"1.3.9": "B36BD648AB2A637A4E087B7B115A6CCBDAEBDF9A"
|
||||||
|
},
|
||||||
|
"ResourceIndexSha1": {
|
||||||
|
"1.3.1": "2D635E549EB6F99659571D72741B62249473A77A",
|
||||||
|
"1.3.2": "C5814A80EA3E7D80D4CFBCD884D1FD158BF0AD9D",
|
||||||
|
"1.3.3": "1E0F05333B09A9215B4AA5C437BFC7DC4014E348",
|
||||||
|
"1.3.4": "6155D492540A99ECF0DA06D2B7EEBFE36231FBC2",
|
||||||
|
"1.3.5": "1E60C8F60CA1AAA9955441B4F4265C8288B95F33",
|
||||||
|
"1.3.6": "AA10A8DD1025D5033E291060C686B816513ADCAD",
|
||||||
|
"1.3.7": "A9881305EBD3DC5A6892D49BDAF540F56EE56232",
|
||||||
|
"1.3.8": "261CA25DAD6877DF3C57DA39947130867FCC09CE",
|
||||||
|
"1.3.9": "88A9E40631FC1C11A91A61CB3F4BE8C13C5E2BD3"
|
||||||
|
},
|
||||||
|
"ChangeList": "2333675",
|
||||||
|
"CompatibleChangeLists": [],
|
||||||
|
"Versions": [
|
||||||
|
{
|
||||||
|
"Name": "en",
|
||||||
|
"Version": "1.3.0",
|
||||||
|
"IndexSha1": {
|
||||||
|
"1.3.0": "6FB5B66EF8B3EECBBBEBE74A82BC23E3FC35450B"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ja",
|
||||||
|
"Version": "1.3.0",
|
||||||
|
"IndexSha1": {
|
||||||
|
"1.3.0": "E4DA1960DB36CE8166C042AD8B9AF98C1A9119F3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ko",
|
||||||
|
"Version": "1.3.0",
|
||||||
|
"IndexSha1": {
|
||||||
|
"1.3.0": "498B379E95FC617385CCD832B8C359FA5AC220CE"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "zh",
|
||||||
|
"Version": "1.3.0",
|
||||||
|
"IndexSha1": {
|
||||||
|
"1.3.0": "CC58C357A80E7B3846264918197FC3ECAA1FE190"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"UpdateTime": 1725869509
|
||||||
|
}
|
122739
assets/logic/json/BaseProperty.json
Normal file
122739
assets/logic/json/BaseProperty.json
Normal file
File diff suppressed because it is too large
Load diff
246
assets/logic/json/ExploreTools.json
Normal file
246
assets/logic/json/ExploreTools.json
Normal file
|
@ -0,0 +1,246 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1004,
|
||||||
|
"Name": "ExploreTools_1004_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1004_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT3.SP_IconT3",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT3.SP_IconT3",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT3.SP_IconT3",
|
||||||
|
"SortId": 3,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1001,
|
||||||
|
"Name": "ExploreTools_1001_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1001_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT6.SP_IconT6",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT6.SP_IconT6",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT6.SP_IconT6",
|
||||||
|
"SortId": 1,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1003,
|
||||||
|
"Name": "ExploreTools_1003_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1003_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT1.SP_IconT1",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT1.SP_IconT1",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT1.SP_IconT1",
|
||||||
|
"SortId": 4,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1005,
|
||||||
|
"Name": "ExploreTools_1005_Name",
|
||||||
|
"SkillType": 2,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1005_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT2.SP_IconT2",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT2.SP_IconT2",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT2.SP_IconT2",
|
||||||
|
"SortId": 5,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1006,
|
||||||
|
"Name": "ExploreTools_1006_Name",
|
||||||
|
"SkillType": 2,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1006_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT5.SP_IconT5",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT5.SP_IconT5",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT5.SP_IconT5",
|
||||||
|
"SortId": 6,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1007,
|
||||||
|
"Name": "ExploreTools_1007_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1007_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT29.SP_IconT29",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT29.SP_IconT29",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT29.SP_IconT29",
|
||||||
|
"SortId": 7,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1009,
|
||||||
|
"Name": "ExploreTools_1009_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1009_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"SortId": 9,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": true,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 3001,
|
||||||
|
"Name": "ExploreTools_1009_Name",
|
||||||
|
"SkillType": 3,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1009_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"SortId": 10,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 3002,
|
||||||
|
"Name": "ExploreTools_1009_Name",
|
||||||
|
"SkillType": 3,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1009_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT9.SP_IconT9",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/UiSkillswitching/Atlas/SP_SkillswitchingItemNone.SP_SkillswitchingItemNone",
|
||||||
|
"SortId": 10,
|
||||||
|
"AutoFill": false,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1010,
|
||||||
|
"Name": "ExploreTools_1010_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1010_CurrentSkillInfo",
|
||||||
|
"HelpId": 30,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT30.SP_IconT30",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT30.SP_IconT30",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT30.SP_IconT30",
|
||||||
|
"SortId": 11,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {
|
||||||
|
"10808": 1
|
||||||
|
},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1011,
|
||||||
|
"Name": "ExploreTools_1011_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1011_CurrentSkillInfo",
|
||||||
|
"HelpId": 28,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT32.SP_IconT32",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT32.SP_IconT32",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT32.SP_IconT32",
|
||||||
|
"SortId": 12,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {
|
||||||
|
"10807": 1
|
||||||
|
},
|
||||||
|
"Authorization": {
|
||||||
|
"1": 10805,
|
||||||
|
"900": 10805
|
||||||
|
},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1012,
|
||||||
|
"Name": "ExploreTools_1012_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1012_CurrentSkillInfo",
|
||||||
|
"HelpId": 29,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT31.SP_IconT31",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT31.SP_IconT31",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT31.SP_IconT31",
|
||||||
|
"SortId": 13,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {
|
||||||
|
"1": 10806,
|
||||||
|
"900": 10806
|
||||||
|
},
|
||||||
|
"SummonConfigId": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PhantomSkillId": 1013,
|
||||||
|
"Name": "ExploreTools_1013_Name",
|
||||||
|
"SkillType": 1,
|
||||||
|
"CurrentSkillInfo": "ExploreTools_1013_CurrentSkillInfo",
|
||||||
|
"HelpId": 0,
|
||||||
|
"Icon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT34.SP_IconT34",
|
||||||
|
"BackGround": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT34.SP_IconT34",
|
||||||
|
"BattleViewIcon": "/Game/Aki/UI/UIResources/Common/Atlas/SkillIcon/SkillIconNor/SP_IconT34.SP_IconT34",
|
||||||
|
"SortId": 14,
|
||||||
|
"AutoFill": true,
|
||||||
|
"ShowUnlock": false,
|
||||||
|
"SkillGroupId": 1,
|
||||||
|
"IsUseInPhantomTeam": false,
|
||||||
|
"Cost": {},
|
||||||
|
"Authorization": {},
|
||||||
|
"SummonConfigId": 24000022
|
||||||
|
}
|
||||||
|
]
|
1034
assets/logic/json/FunctionCondition.json
Normal file
1034
assets/logic/json/FunctionCondition.json
Normal file
File diff suppressed because it is too large
Load diff
44971
assets/logic/json/InstanceDungeon.json
Normal file
44971
assets/logic/json/InstanceDungeon.json
Normal file
File diff suppressed because it is too large
Load diff
3874
assets/logic/json/RoleInfo.json
Normal file
3874
assets/logic/json/RoleInfo.json
Normal file
File diff suppressed because it is too large
Load diff
5680
assets/logic/json/WeaponConf.json
Normal file
5680
assets/logic/json/WeaponConf.json
Normal file
File diff suppressed because it is too large
Load diff
16
builder.bat
16
builder.bat
|
@ -1,12 +1,12 @@
|
||||||
docker build -t wicked-waifus-builder:2.1.0-SNAPSHOT -f Dockerfile-builder .
|
docker build -t shorekeeper-builder:1.3.0-SNAPSHOT -f Dockerfile-builder .
|
||||||
|
|
||||||
docker build -t wicked-waifus-config-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-config-server -f Dockerfile-service .
|
docker build -t shorekeeper-config-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=config-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-hotpatch-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-server -f Dockerfile-service .
|
docker build -t shorekeeper-hotpatch-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=hotpatch-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-login-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-login-server -f Dockerfile-service .
|
docker build -t shorekeeper-login-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=login-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-gateway-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-gateway-server -f Dockerfile-service .
|
docker build -t shorekeeper-gateway-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=gateway-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-game-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-game-server -f Dockerfile-service .
|
docker build -t shorekeeper-game-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=game-server -f Dockerfile-service .
|
||||||
|
|
||||||
docker rmi wicked-waifus-builder:2.1.0-SNAPSHOT
|
docker rmi shorekeeper-builder:1.3.0-SNAPSHOT
|
||||||
|
|
||||||
: Persistence for the application
|
: Persistence for the application
|
||||||
: docker volume create wicked-waifus-postgres-vol
|
docker volume create shorekeeper-postgres-vol
|
16
builder.sh
16
builder.sh
|
@ -1,12 +1,12 @@
|
||||||
docker build -t wicked-waifus-builder:2.1.0-SNAPSHOT -f Dockerfile-builder .
|
docker build -t shorekeeper-builder:1.3.0-SNAPSHOT -f Dockerfile-builder .
|
||||||
|
|
||||||
docker build -t wicked-waifus-config-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-config-server -f Dockerfile-service .
|
docker build -t shorekeeper-config-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=config-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-hotpatch-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-hotpatch-server -f Dockerfile-service .
|
docker build -t shorekeeper-hotpatch-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=hotpatch-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-login-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-login-server -f Dockerfile-service .
|
docker build -t shorekeeper-login-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=login-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-gateway-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-gateway-server -f Dockerfile-service .
|
docker build -t shorekeeper-gateway-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=gateway-server -f Dockerfile-service .
|
||||||
docker build -t wicked-waifus-game-server:2.1.0-SNAPSHOT --build-arg MICROSERVICE=wicked-waifus-game-server -f Dockerfile-service .
|
docker build -t shorekeeper-game-server:1.3.0-SNAPSHOT --build-arg MICROSERVICE=game-server -f Dockerfile-service .
|
||||||
|
|
||||||
docker rmi wicked-waifus-builder:2.1.0-SNAPSHOT
|
docker rmi shorekeeper-builder:1.3.0-SNAPSHOT
|
||||||
|
|
||||||
# Persistence for the application
|
# Persistence for the application
|
||||||
# docker volume create wicked-waifus-postgres-vol
|
docker volume create shorekeeper-postgres-vol
|
|
@ -1,5 +1,5 @@
|
||||||
[package]
|
[package]
|
||||||
name = "wicked-waifus-commons"
|
name = "common"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
version.workspace = true
|
version.workspace = true
|
||||||
|
|
8
common/src/logging.rs
Normal file
8
common/src/logging.rs
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
use tracing::Level;
|
||||||
|
|
||||||
|
pub fn init(max_level: Level) {
|
||||||
|
tracing_subscriber::fmt()
|
||||||
|
.with_max_level(max_level)
|
||||||
|
.with_target(false)
|
||||||
|
.init();
|
||||||
|
}
|
3
common/src/splash.rs
Normal file
3
common/src/splash.rs
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
pub fn print_splash() {
|
||||||
|
println!(" _____ __ __ \n / ___// /_ ____ ________ / /_____ ___ ____ ___ _____\n \\__ \\/ __ \\/ __ \\/ ___/ _ \\/ //_/ _ \\/ _ \\/ __ \\/ _ \\/ ___/\n ___/ / / / / /_/ / / / __/ ,< / __/ __/ /_/ / __/ / \n/____/_/ /_/\\____/_/ \\___/_/|_|\\___/\\___/ .___/\\___/_/ \n /_/ ");
|
||||||
|
}
|
|
@ -1,17 +1,5 @@
|
||||||
use std::time::{SystemTime, UNIX_EPOCH};
|
use std::time::{SystemTime, UNIX_EPOCH};
|
||||||
|
|
||||||
const SECONDS_PER_MINUTE: u64 = 60;
|
|
||||||
const SECONDS_PER_HOUR: u64 = 60 * SECONDS_PER_MINUTE;
|
|
||||||
const SECONDS_PER_DAY: u64 = 24 * SECONDS_PER_HOUR;
|
|
||||||
|
|
||||||
pub fn unix_days() -> i32 {
|
|
||||||
(SystemTime::now()
|
|
||||||
.duration_since(UNIX_EPOCH)
|
|
||||||
.unwrap()
|
|
||||||
.as_secs()
|
|
||||||
/ SECONDS_PER_DAY) as i32
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn unix_timestamp() -> u64 {
|
pub fn unix_timestamp() -> u64 {
|
||||||
SystemTime::now()
|
SystemTime::now()
|
||||||
.duration_since(UNIX_EPOCH)
|
.duration_since(UNIX_EPOCH)
|
|
@ -1,11 +1,12 @@
|
||||||
[package]
|
[package]
|
||||||
name = "wicked-waifus-hotpatch-server"
|
name = "config-server"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
version.workspace = true
|
version.workspace = true
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
# Framework
|
# Framework
|
||||||
tokio.workspace = true
|
tokio.workspace = true
|
||||||
|
shorekeeper-http.workspace = true
|
||||||
|
|
||||||
# Serialization
|
# Serialization
|
||||||
serde.workspace = true
|
serde.workspace = true
|
||||||
|
@ -18,5 +19,4 @@ anyhow.workspace = true
|
||||||
tracing.workspace = true
|
tracing.workspace = true
|
||||||
|
|
||||||
# Internal
|
# Internal
|
||||||
wicked-waifus-commons.workspace = true
|
common.workspace = true
|
||||||
wicked-waifus-http.workspace = true
|
|
6
config-server/configserver.default.toml
Normal file
6
config-server/configserver.default.toml
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
[network]
|
||||||
|
http_addr = "0.0.0.0:10001"
|
||||||
|
|
||||||
|
[encryption]
|
||||||
|
key = "t+AEu5SGdpz06tomonajLMau9AJgmyTvVhz9VtGf1+0="
|
||||||
|
iv = "fprc5lBWADQB7tim0R2JxQ=="
|
45
config-server/src/main.rs
Normal file
45
config-server/src/main.rs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
use std::fs;
|
||||||
|
use std::sync::LazyLock;
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
|
||||||
|
use common::config_util::{self, TomlConfig};
|
||||||
|
use serde::Deserialize;
|
||||||
|
use shorekeeper_http::{
|
||||||
|
config::{AesSettings, NetworkSettings},
|
||||||
|
Application,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
pub struct ServerConfig {
|
||||||
|
pub network: NetworkSettings,
|
||||||
|
pub encryption: AesSettings,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TomlConfig for ServerConfig {
|
||||||
|
const DEFAULT_TOML: &str = include_str!("../configserver.default.toml");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() -> Result<()> {
|
||||||
|
static CONFIG: LazyLock<ServerConfig> =
|
||||||
|
LazyLock::new(|| config_util::load_or_create("configserver.toml"));
|
||||||
|
|
||||||
|
::common::splash::print_splash();
|
||||||
|
::common::logging::init(::tracing::Level::DEBUG);
|
||||||
|
|
||||||
|
Application::new()
|
||||||
|
.get("/index.json", get_index)
|
||||||
|
.with_encryption(&CONFIG.encryption)
|
||||||
|
.serve(&CONFIG.network)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_index() -> &'static str {
|
||||||
|
static INDEX: LazyLock<String> =
|
||||||
|
LazyLock::new(|| fs::read_to_string("assets/config/index.json").unwrap());
|
||||||
|
|
||||||
|
&*INDEX
|
||||||
|
}
|
|
@ -1,44 +0,0 @@
|
||||||
{
|
|
||||||
"PackageVersion": "1.4.0",
|
|
||||||
"LauncherVersion": "1.4.0",
|
|
||||||
"ResourceVersion": "1.4.3",
|
|
||||||
"LauncherIndexSha1": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
||||||
"ResourceIndexSha1": {
|
|
||||||
"1.4.1": "0C747E444EC6DBA11D0C57B34E2638DE5182029F",
|
|
||||||
"1.4.2": "41751A94DBD406653DB52262E85BB716E45D9864",
|
|
||||||
"1.4.3": "424EDFD97BA8699CBC2CB9E27D3FCBB6F7C70806"
|
|
||||||
},
|
|
||||||
"ChangeList": "2585778",
|
|
||||||
"CompatibleChangeLists": [],
|
|
||||||
"Versions": [
|
|
||||||
{
|
|
||||||
"Name": "en",
|
|
||||||
"Version": "1.4.0",
|
|
||||||
"IndexSha1": {
|
|
||||||
"1.4.0": "34F9AC326B3E81E972B93094E43751967A6A3396"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"Name": "ja",
|
|
||||||
"Version": "1.4.0",
|
|
||||||
"IndexSha1": {
|
|
||||||
"1.4.0": "3233223428D586FBC0A2EC817F4E94DDD10C646F"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"Name": "ko",
|
|
||||||
"Version": "1.4.0",
|
|
||||||
"IndexSha1": {
|
|
||||||
"1.4.0": "3CC0312BDFFF5FA4D3F5256E2618D0EF6AA84912"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"Name": "zh",
|
|
||||||
"Version": "1.4.0",
|
|
||||||
"IndexSha1": {
|
|
||||||
"1.4.0": "2F3AD1412C6991FE0E146C5E2F6F86D8D9F44A5D"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"UpdateTime": 1729243476
|
|
||||||
}
|
|
|
@ -1,13 +0,0 @@
|
||||||
service_id = 2
|
|
||||||
|
|
||||||
[database]
|
|
||||||
host = "wicked-waifus-postgres:5432"
|
|
||||||
user_name = "wicked_waifus_user"
|
|
||||||
password = "wicked_waifus_pass"
|
|
||||||
db_name = "wicked_waifus_db"
|
|
||||||
|
|
||||||
[service_end_point]
|
|
||||||
addr = "tcp://0.0.0.0:10004"
|
|
||||||
|
|
||||||
[gateway_end_point]
|
|
||||||
addr = "tcp://wicked-waifus-gateway-server:10003"
|
|
|
@ -1,12 +0,0 @@
|
||||||
[network]
|
|
||||||
http_addr = "0.0.0.0:5500"
|
|
||||||
|
|
||||||
[gateway]
|
|
||||||
host = "host.docker.internal"
|
|
||||||
port = 7777
|
|
||||||
|
|
||||||
[database]
|
|
||||||
host = "wicked-waifus-postgres:5432"
|
|
||||||
user_name = "wicked_waifus_user"
|
|
||||||
password = "wicked_waifus_pass"
|
|
||||||
db_name = "wicked_waifus_db"
|
|
|
@ -1,3 +0,0 @@
|
||||||
CREATE DATABASE wicked_waifus_db;
|
|
||||||
CREATE USER wicked_waifus_user WITH encrypted password 'wicked_waifus_pass';
|
|
||||||
GRANT ALL PRIVILEGES ON DATABASE wicked_waifus_db to wicked_waifus_user;
|
|
|
@ -1,2 +0,0 @@
|
||||||
\c wicked_waifus_db;
|
|
||||||
GRANT ALL ON SCHEMA public TO wicked_waifus_user;
|
|
|
@ -1,13 +0,0 @@
|
||||||
name: wicked-waifus-ps
|
|
||||||
|
|
||||||
services:
|
|
||||||
wicked-waifus-hotpatch-server:
|
|
||||||
image: wicked-waifus-hotpatch-server:2.1.0-SNAPSHOT
|
|
||||||
depends_on:
|
|
||||||
wicked-waifus-postgres:
|
|
||||||
condition: service_healthy
|
|
||||||
ports:
|
|
||||||
- '10002:10002'
|
|
||||||
volumes:
|
|
||||||
- "./data/docker/hotpatch.toml:/app/hotpatch.toml"
|
|
||||||
- "./data/assets/hotpatch-server:/app/assets/hotpatch"
|
|
|
@ -1,53 +1,63 @@
|
||||||
name: wicked-waifus-ps
|
name: shorekeeper-ps
|
||||||
|
|
||||||
services:
|
services:
|
||||||
wicked-waifus-config-server:
|
shorekeeper-config-server:
|
||||||
image: wicked-waifus-config-server:2.1.0-SNAPSHOT
|
image: shorekeeper-config-server:1.3.0-SNAPSHOT
|
||||||
depends_on:
|
depends_on:
|
||||||
wicked-waifus-postgres:
|
shorekeeper-postgres:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
ports:
|
ports:
|
||||||
- '10001:10001'
|
- '10001:10001'
|
||||||
volumes:
|
volumes:
|
||||||
- "./data/docker/configserver.toml:/app/configserver.toml"
|
- "./docker/configserver.toml:/app/configserver.toml"
|
||||||
- "./data/assets/config:/app/assets/config"
|
- "./assets/config:/app/assets/config"
|
||||||
wicked-waifus-login-server:
|
shorekeeper-hotpatch-server:
|
||||||
image: wicked-waifus-login-server:2.1.0-SNAPSHOT
|
image: shorekeeper-hotpatch-server:1.3.0-SNAPSHOT
|
||||||
depends_on:
|
depends_on:
|
||||||
wicked-waifus-postgres:
|
shorekeeper-postgres:
|
||||||
|
condition: service_healthy
|
||||||
|
ports:
|
||||||
|
- '10002:10002'
|
||||||
|
volumes:
|
||||||
|
- "./docker/hotpatch.toml:/app/hotpatch.toml"
|
||||||
|
- "./assets/hotpatch:/app/assets/hotpatch"
|
||||||
|
shorekeeper-login-server:
|
||||||
|
image: shorekeeper-login-server:1.3.0-SNAPSHOT
|
||||||
|
depends_on:
|
||||||
|
shorekeeper-postgres:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
ports:
|
ports:
|
||||||
- '5500:5500'
|
- '5500:5500'
|
||||||
volumes:
|
volumes:
|
||||||
- "./data/docker/loginserver.toml:/app/loginserver.toml"
|
- "./docker/loginserver.toml:/app/loginserver.toml"
|
||||||
wicked-waifus-gateway-server:
|
shorekeeper-gateway-server:
|
||||||
image: wicked-waifus-gateway-server:2.1.0-SNAPSHOT
|
image: shorekeeper-gateway-server:1.3.0-SNAPSHOT
|
||||||
depends_on:
|
depends_on:
|
||||||
wicked-waifus-postgres:
|
shorekeeper-postgres:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
ports:
|
ports:
|
||||||
# Uncomment this if you want to have manual access
|
# Uncomment this if you want to have manual access
|
||||||
# - '10003:10003'
|
# - '10003:10003'
|
||||||
- '7777:7777/udp'
|
- '7777:7777/udp'
|
||||||
volumes:
|
volumes:
|
||||||
- "./data/docker/gateway.toml:/app/gateway.toml"
|
- "./docker/gateway.toml:/app/gateway.toml"
|
||||||
wicked-waifus-game-server:
|
shorekeeper-game-server:
|
||||||
image: wicked-waifus-game-server:2.1.0-SNAPSHOT
|
image: shorekeeper-game-server:1.3.0-SNAPSHOT
|
||||||
depends_on:
|
depends_on:
|
||||||
wicked-waifus-postgres:
|
shorekeeper-postgres:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
# Uncomment this if you want to have manual access
|
# Uncomment this if you want to have manual access
|
||||||
# ports:
|
# ports:
|
||||||
# - '10004:10004'
|
# - '10004:10004'
|
||||||
volumes:
|
volumes:
|
||||||
- "./data/docker/gameserver.toml:/app/gameserver.toml"
|
- "./docker/gameserver.toml:/app/gameserver.toml"
|
||||||
- "./data/assets/game-data:/app/data/assets/game-data"
|
- "./assets/logic:/app/assets/logic"
|
||||||
wicked-waifus-postgres:
|
shorekeeper-postgres:
|
||||||
image: postgres:16.4-alpine3.20
|
image: postgres:16.4-alpine3.20
|
||||||
user: postgres
|
user: postgres
|
||||||
# Uncomment this if you want to have manual access
|
# Uncomment this if you want to have manual access
|
||||||
ports:
|
# ports:
|
||||||
- '5432:5432'
|
# - '5432:5432'
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test: ["CMD-SHELL", "pg_isready"]
|
test: ["CMD-SHELL", "pg_isready"]
|
||||||
interval: 10s
|
interval: 10s
|
||||||
|
@ -56,8 +66,8 @@ services:
|
||||||
environment:
|
environment:
|
||||||
- "POSTGRES_PASSWORD=toor"
|
- "POSTGRES_PASSWORD=toor"
|
||||||
volumes:
|
volumes:
|
||||||
- "./data/docker/postgres/scripts:/docker-entrypoint-initdb.d"
|
- "./docker/postgres/scripts:/docker-entrypoint-initdb.d"
|
||||||
- wicked-waifus-postgres-vol:/var/lib/postgresql/data
|
- shorekeeper-postgres-vol:/var/lib/postgresql/data
|
||||||
volumes:
|
volumes:
|
||||||
wicked-waifus-postgres-vol:
|
shorekeeper-postgres-vol:
|
||||||
external: true
|
external: true
|
|
@ -4,7 +4,3 @@ http_addr = "0.0.0.0:10001"
|
||||||
[encryption]
|
[encryption]
|
||||||
key = "t+AEu5SGdpz06tomonajLMau9AJgmyTvVhz9VtGf1+0="
|
key = "t+AEu5SGdpz06tomonajLMau9AJgmyTvVhz9VtGf1+0="
|
||||||
iv = "fprc5lBWADQB7tim0R2JxQ=="
|
iv = "fprc5lBWADQB7tim0R2JxQ=="
|
||||||
|
|
||||||
[serve]
|
|
||||||
serve_web_path = "/"
|
|
||||||
serve_dir_path = "assets/config-server"
|
|
13
docker/gameserver.toml
Normal file
13
docker/gameserver.toml
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
service_id = 2
|
||||||
|
|
||||||
|
[database]
|
||||||
|
host = "shorekeeper-postgres:5432"
|
||||||
|
user_name = "shorekeeper_user"
|
||||||
|
password = "shorekeeper_pass"
|
||||||
|
db_name = "shorekeeper_db"
|
||||||
|
|
||||||
|
[service_end_point]
|
||||||
|
addr = "tcp://0.0.0.0:10004"
|
||||||
|
|
||||||
|
[gateway_end_point]
|
||||||
|
addr = "tcp://shorekeeper-gateway-server:10003"
|
|
@ -11,10 +11,10 @@ use_client_key = true
|
||||||
addr = "tcp://0.0.0.0:10003"
|
addr = "tcp://0.0.0.0:10003"
|
||||||
|
|
||||||
[game_server_end_point]
|
[game_server_end_point]
|
||||||
addr = "tcp://wicked-waifus-game-server:10004"
|
addr = "tcp://shorekeeper-game-server:10004"
|
||||||
|
|
||||||
[database]
|
[database]
|
||||||
host = "wicked-waifus-postgres:5432"
|
host = "shorekeeper-postgres:5432"
|
||||||
user_name = "wicked_waifus_user"
|
user_name = "shorekeeper_user"
|
||||||
password = "wicked_waifus_pass"
|
password = "shorekeeper_pass"
|
||||||
db_name = "wicked_waifus_db"
|
db_name = "shorekeeper_db"
|
12
docker/loginserver.toml
Normal file
12
docker/loginserver.toml
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
[network]
|
||||||
|
http_addr = "0.0.0.0:5500"
|
||||||
|
|
||||||
|
[gateway]
|
||||||
|
host = "host.docker.internal"
|
||||||
|
port = 7777
|
||||||
|
|
||||||
|
[database]
|
||||||
|
host = "shorekeeper-postgres:5432"
|
||||||
|
user_name = "shorekeeper_user"
|
||||||
|
password = "shorekeeper_pass"
|
||||||
|
db_name = "shorekeeper_db"
|
3
docker/postgres/scripts/0_INIT.sql
Normal file
3
docker/postgres/scripts/0_INIT.sql
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
CREATE DATABASE shorekeeper_db;
|
||||||
|
CREATE USER shorekeeper_user WITH encrypted password 'shorekeeper_pass';
|
||||||
|
GRANT ALL PRIVILEGES ON DATABASE shorekeeper_db to shorekeeper_user;
|
2
docker/postgres/scripts/1_PERMS.sql
Normal file
2
docker/postgres/scripts/1_PERMS.sql
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
\c shorekeeper_db;
|
||||||
|
GRANT ALL ON SCHEMA public TO shorekeeper_user;
|
28
game-server/Cargo.toml
Normal file
28
game-server/Cargo.toml
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
[package]
|
||||||
|
name = "game-server"
|
||||||
|
edition = "2021"
|
||||||
|
version.workspace = true
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
# Framework
|
||||||
|
tokio.workspace = true
|
||||||
|
|
||||||
|
# Serialization
|
||||||
|
serde.workspace = true
|
||||||
|
|
||||||
|
# Util
|
||||||
|
anyhow.workspace = true
|
||||||
|
thiserror.workspace = true
|
||||||
|
paste.workspace = true
|
||||||
|
dashmap.workspace = true
|
||||||
|
hex.workspace = true
|
||||||
|
|
||||||
|
# Tracing
|
||||||
|
tracing.workspace = true
|
||||||
|
|
||||||
|
# Internal
|
||||||
|
common.workspace = true
|
||||||
|
shorekeeper-data.workspace = true
|
||||||
|
shorekeeper-database.workspace = true
|
||||||
|
shorekeeper-network.workspace = true
|
||||||
|
shorekeeper-protocol.workspace = true
|
13
game-server/gameserver.default.toml
Normal file
13
game-server/gameserver.default.toml
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
service_id = 2
|
||||||
|
|
||||||
|
[database]
|
||||||
|
host = "localhost:5432"
|
||||||
|
user_name = "postgres"
|
||||||
|
password = ""
|
||||||
|
db_name = "shorekeeper"
|
||||||
|
|
||||||
|
[service_end_point]
|
||||||
|
addr = "tcp://127.0.0.1:10004"
|
||||||
|
|
||||||
|
[gateway_end_point]
|
||||||
|
addr = "tcp://127.0.0.1:10003"
|
16
game-server/src/config.rs
Normal file
16
game-server/src/config.rs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
use common::config_util::TomlConfig;
|
||||||
|
use serde::Deserialize;
|
||||||
|
use shorekeeper_database::DatabaseSettings;
|
||||||
|
use shorekeeper_network::config::ServiceEndPoint;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
pub struct ServiceConfig {
|
||||||
|
pub service_id: u32,
|
||||||
|
pub database: DatabaseSettings,
|
||||||
|
pub service_end_point: ServiceEndPoint,
|
||||||
|
pub gateway_end_point: ServiceEndPoint,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TomlConfig for ServiceConfig {
|
||||||
|
const DEFAULT_TOML: &str = include_str!("../gameserver.default.toml");
|
||||||
|
}
|
|
@ -1,6 +1,6 @@
|
||||||
use std::sync::OnceLock;
|
use std::sync::OnceLock;
|
||||||
|
|
||||||
use wicked_waifus_network::{config::ServiceEndPoint, ServiceClient, ServiceMessage};
|
use shorekeeper_network::{config::ServiceEndPoint, ServiceClient, ServiceMessage};
|
||||||
|
|
||||||
static CLIENT: OnceLock<ServiceClient> = OnceLock::new();
|
static CLIENT: OnceLock<ServiceClient> = OnceLock::new();
|
||||||
|
|
198
game-server/src/logic/components/attribute.rs
Normal file
198
game-server/src/logic/components/attribute.rs
Normal file
|
@ -0,0 +1,198 @@
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
|
use shorekeeper_data::BasePropertyData;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
entity_component_pb::ComponentPb, AttrData, AttributeComponentPb, EAttributeType,
|
||||||
|
EntityComponentPb, LivingStatus,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::logic::ecs::component::Component;
|
||||||
|
|
||||||
|
pub struct Attribute {
|
||||||
|
pub attr_map: HashMap<EAttributeType, (i32, i32)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! impl_from_data {
|
||||||
|
($($name:ident),*) => {
|
||||||
|
pub fn from_data(base_property: &BasePropertyData) -> Self {
|
||||||
|
Self {
|
||||||
|
attr_map: HashMap::from([$(
|
||||||
|
::paste::paste!((EAttributeType::[<$name:camel>], (base_property.$name, 0))),
|
||||||
|
)*]),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Component for Attribute {
|
||||||
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
|
pb.living_status = (if self.is_alive() { LivingStatus::Alive } else { LivingStatus::Dead })
|
||||||
|
.into();
|
||||||
|
|
||||||
|
pb.component_pbs.push(EntityComponentPb {
|
||||||
|
component_pb: Some(ComponentPb::AttributeComponent(AttributeComponentPb {
|
||||||
|
attr_data: self
|
||||||
|
.attr_map
|
||||||
|
.iter()
|
||||||
|
.map(|(ty, (base, incr))| AttrData {
|
||||||
|
attribute_type: (*ty).into(),
|
||||||
|
base_value: *base,
|
||||||
|
increment: *incr,
|
||||||
|
})
|
||||||
|
.collect(),
|
||||||
|
hardness_mode_id: 0,
|
||||||
|
rage_mode_id: 0,
|
||||||
|
})),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Attribute {
|
||||||
|
pub fn is_alive(&self) -> bool {
|
||||||
|
self.attr_map
|
||||||
|
.get(&EAttributeType::Life)
|
||||||
|
.copied()
|
||||||
|
.unwrap_or_default()
|
||||||
|
.0
|
||||||
|
> 0
|
||||||
|
}
|
||||||
|
|
||||||
|
impl_from_data!(
|
||||||
|
lv,
|
||||||
|
life_max,
|
||||||
|
life,
|
||||||
|
sheild,
|
||||||
|
sheild_damage_change,
|
||||||
|
sheild_damage_reduce,
|
||||||
|
atk,
|
||||||
|
crit,
|
||||||
|
crit_damage,
|
||||||
|
def,
|
||||||
|
energy_efficiency,
|
||||||
|
cd_reduse,
|
||||||
|
reaction_efficiency,
|
||||||
|
damage_change_normal_skill,
|
||||||
|
damage_change,
|
||||||
|
damage_reduce,
|
||||||
|
damage_change_auto,
|
||||||
|
damage_change_cast,
|
||||||
|
damage_change_ultra,
|
||||||
|
damage_change_qte,
|
||||||
|
damage_change_phys,
|
||||||
|
damage_change_element1,
|
||||||
|
damage_change_element2,
|
||||||
|
damage_change_element3,
|
||||||
|
damage_change_element4,
|
||||||
|
damage_change_element5,
|
||||||
|
damage_change_element6,
|
||||||
|
damage_resistance_phys,
|
||||||
|
damage_resistance_element1,
|
||||||
|
damage_resistance_element2,
|
||||||
|
damage_resistance_element3,
|
||||||
|
damage_resistance_element4,
|
||||||
|
damage_resistance_element5,
|
||||||
|
damage_resistance_element6,
|
||||||
|
heal_change,
|
||||||
|
healed_change,
|
||||||
|
damage_reduce_phys,
|
||||||
|
damage_reduce_element1,
|
||||||
|
damage_reduce_element2,
|
||||||
|
damage_reduce_element3,
|
||||||
|
damage_reduce_element4,
|
||||||
|
damage_reduce_element5,
|
||||||
|
damage_reduce_element6,
|
||||||
|
reaction_change1,
|
||||||
|
reaction_change2,
|
||||||
|
reaction_change3,
|
||||||
|
reaction_change4,
|
||||||
|
reaction_change5,
|
||||||
|
reaction_change6,
|
||||||
|
reaction_change7,
|
||||||
|
reaction_change8,
|
||||||
|
reaction_change9,
|
||||||
|
reaction_change10,
|
||||||
|
reaction_change11,
|
||||||
|
reaction_change12,
|
||||||
|
reaction_change13,
|
||||||
|
reaction_change14,
|
||||||
|
reaction_change15,
|
||||||
|
energy_max,
|
||||||
|
energy,
|
||||||
|
special_energy_1_max,
|
||||||
|
special_energy_1,
|
||||||
|
special_energy_2_max,
|
||||||
|
special_energy_2,
|
||||||
|
special_energy_3_max,
|
||||||
|
special_energy_3,
|
||||||
|
special_energy_4_max,
|
||||||
|
special_energy_4,
|
||||||
|
strength_max,
|
||||||
|
strength,
|
||||||
|
strength_recover,
|
||||||
|
strength_punish_time,
|
||||||
|
strength_run,
|
||||||
|
strength_swim,
|
||||||
|
strength_fast_swim,
|
||||||
|
hardness_max,
|
||||||
|
hardness,
|
||||||
|
hardness_recover,
|
||||||
|
hardness_punish_time,
|
||||||
|
hardness_change,
|
||||||
|
hardness_reduce,
|
||||||
|
rage_max,
|
||||||
|
rage,
|
||||||
|
rage_recover,
|
||||||
|
rage_punish_time,
|
||||||
|
rage_change,
|
||||||
|
rage_reduce,
|
||||||
|
tough_max,
|
||||||
|
tough,
|
||||||
|
tough_recover,
|
||||||
|
tough_change,
|
||||||
|
tough_reduce,
|
||||||
|
tough_recover_delay_time,
|
||||||
|
element_power1,
|
||||||
|
element_power2,
|
||||||
|
element_power3,
|
||||||
|
element_power4,
|
||||||
|
element_power5,
|
||||||
|
element_power6,
|
||||||
|
special_damage_change,
|
||||||
|
strength_fast_climb_cost,
|
||||||
|
element_property_type,
|
||||||
|
weak_time,
|
||||||
|
ignore_def_rate,
|
||||||
|
ignore_damage_resistance_phys,
|
||||||
|
ignore_damage_resistance_element1,
|
||||||
|
ignore_damage_resistance_element2,
|
||||||
|
ignore_damage_resistance_element3,
|
||||||
|
ignore_damage_resistance_element4,
|
||||||
|
ignore_damage_resistance_element5,
|
||||||
|
ignore_damage_resistance_element6,
|
||||||
|
skill_tough_ratio,
|
||||||
|
strength_climb_jump,
|
||||||
|
strength_gliding,
|
||||||
|
mass,
|
||||||
|
braking_friction_factor,
|
||||||
|
gravity_scale,
|
||||||
|
speed_ratio,
|
||||||
|
damage_change_phantom,
|
||||||
|
auto_attack_speed,
|
||||||
|
cast_attack_speed,
|
||||||
|
status_build_up_1_max,
|
||||||
|
status_build_up_1,
|
||||||
|
status_build_up_2_max,
|
||||||
|
status_build_up_2,
|
||||||
|
status_build_up_3_max,
|
||||||
|
status_build_up_3,
|
||||||
|
status_build_up_4_max,
|
||||||
|
status_build_up_4,
|
||||||
|
status_build_up_5_max,
|
||||||
|
status_build_up_5,
|
||||||
|
paralysis_time_max,
|
||||||
|
paralysis_time,
|
||||||
|
paralysis_time_recover,
|
||||||
|
element_energy_max,
|
||||||
|
element_energy
|
||||||
|
);
|
||||||
|
}
|
15
game-server/src/logic/components/entity_config.rs
Normal file
15
game-server/src/logic/components/entity_config.rs
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
use shorekeeper_protocol::EntityConfigType;
|
||||||
|
|
||||||
|
use crate::logic::ecs::component::Component;
|
||||||
|
|
||||||
|
pub struct EntityConfig {
|
||||||
|
pub config_id: i32,
|
||||||
|
pub config_type: EntityConfigType,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Component for EntityConfig {
|
||||||
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
|
pb.config_id = self.config_id;
|
||||||
|
pb.config_type = self.config_type.into();
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::logic::ecs::component::Component;
|
use crate::logic::ecs::component::Component;
|
||||||
use wicked_waifus_protocol::entity_component_pb::ComponentPb;
|
use shorekeeper_protocol::entity_component_pb::ComponentPb;
|
||||||
use wicked_waifus_protocol::{EntityComponentPb, EquipComponentPb};
|
use shorekeeper_protocol::{EntityComponentPb, EquipComponentPb};
|
||||||
|
|
||||||
pub struct Equip {
|
pub struct Equip {
|
||||||
pub weapon_id: i32,
|
pub weapon_id: i32,
|
||||||
|
@ -8,7 +8,7 @@ pub struct Equip {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Component for Equip {
|
impl Component for Equip {
|
||||||
fn set_pb_data(&self, pb: &mut wicked_waifus_protocol::EntityPb) {
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
pb.component_pbs.push(EntityComponentPb {
|
pb.component_pbs.push(EntityComponentPb {
|
||||||
component_pb: Some(ComponentPb::EquipComponent(EquipComponentPb {
|
component_pb: Some(ComponentPb::EquipComponent(EquipComponentPb {
|
||||||
weapon_id: self.weapon_id,
|
weapon_id: self.weapon_id,
|
19
game-server/src/logic/components/mod.rs
Normal file
19
game-server/src/logic/components/mod.rs
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
mod attribute;
|
||||||
|
mod entity_config;
|
||||||
|
mod equip;
|
||||||
|
mod movement;
|
||||||
|
mod owner_player;
|
||||||
|
mod player_entity_marker;
|
||||||
|
mod position;
|
||||||
|
mod visibility;
|
||||||
|
mod vision_skill;
|
||||||
|
|
||||||
|
pub use attribute::Attribute;
|
||||||
|
pub use entity_config::EntityConfig;
|
||||||
|
pub use equip::Equip;
|
||||||
|
pub use movement::Movement;
|
||||||
|
pub use owner_player::OwnerPlayer;
|
||||||
|
pub use player_entity_marker::PlayerEntityMarker;
|
||||||
|
pub use position::Position;
|
||||||
|
pub use visibility::Visibility;
|
||||||
|
pub use vision_skill::VisionSkill;
|
|
@ -1,6 +1,6 @@
|
||||||
use std::collections::VecDeque;
|
use std::collections::VecDeque;
|
||||||
|
|
||||||
use wicked_waifus_protocol::MoveReplaySample;
|
use shorekeeper_protocol::MoveReplaySample;
|
||||||
|
|
||||||
use crate::logic::ecs::component::Component;
|
use crate::logic::ecs::component::Component;
|
||||||
|
|
||||||
|
@ -10,5 +10,5 @@ pub struct Movement {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Component for Movement {
|
impl Component for Movement {
|
||||||
fn set_pb_data(&self, _: &mut wicked_waifus_protocol::EntityPb) {}
|
fn set_pb_data(&self, _: &mut shorekeeper_protocol::EntityPb) {}
|
||||||
}
|
}
|
|
@ -3,7 +3,7 @@ use crate::logic::ecs::component::Component;
|
||||||
pub struct OwnerPlayer(pub i32);
|
pub struct OwnerPlayer(pub i32);
|
||||||
|
|
||||||
impl Component for OwnerPlayer {
|
impl Component for OwnerPlayer {
|
||||||
fn set_pb_data(&self, pb: &mut wicked_waifus_protocol::EntityPb) {
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
pb.player_id = self.0;
|
pb.player_id = self.0;
|
||||||
}
|
}
|
||||||
}
|
}
|
11
game-server/src/logic/components/player_entity_marker.rs
Normal file
11
game-server/src/logic/components/player_entity_marker.rs
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
use shorekeeper_protocol::EEntityType;
|
||||||
|
|
||||||
|
use crate::logic::ecs::component::Component;
|
||||||
|
|
||||||
|
pub struct PlayerEntityMarker;
|
||||||
|
|
||||||
|
impl Component for PlayerEntityMarker {
|
||||||
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
|
pb.entity_type = EEntityType::Player.into();
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,7 +3,7 @@ use crate::logic::{ecs::component::Component, math::Transform};
|
||||||
pub struct Position(pub Transform);
|
pub struct Position(pub Transform);
|
||||||
|
|
||||||
impl Component for Position {
|
impl Component for Position {
|
||||||
fn set_pb_data(&self, pb: &mut wicked_waifus_protocol::EntityPb) {
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
pb.pos = Some(self.0.get_position_protobuf());
|
pb.pos = Some(self.0.get_position_protobuf());
|
||||||
pb.rot = Some(self.0.get_rotation_protobuf());
|
pb.rot = Some(self.0.get_rotation_protobuf());
|
||||||
pb.init_pos = Some(self.0.get_position_protobuf());
|
pb.init_pos = Some(self.0.get_position_protobuf());
|
9
game-server/src/logic/components/visibility.rs
Normal file
9
game-server/src/logic/components/visibility.rs
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
use crate::logic::ecs::component::Component;
|
||||||
|
|
||||||
|
pub struct Visibility(pub bool);
|
||||||
|
|
||||||
|
impl Component for Visibility {
|
||||||
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
|
pb.is_visible = self.0;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
use wicked_waifus_protocol::{
|
use shorekeeper_protocol::{
|
||||||
entity_component_pb::ComponentPb, EntityComponentPb, VisionSkillComponentPb,
|
entity_component_pb::ComponentPb, EntityComponentPb, VisionSkillComponentPb,
|
||||||
VisionSkillInformation,
|
VisionSkillInformation,
|
||||||
};
|
};
|
||||||
|
@ -10,7 +10,7 @@ pub struct VisionSkill {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Component for VisionSkill {
|
impl Component for VisionSkill {
|
||||||
fn set_pb_data(&self, pb: &mut wicked_waifus_protocol::EntityPb) {
|
fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
pb.component_pbs.push(EntityComponentPb {
|
pb.component_pbs.push(EntityComponentPb {
|
||||||
component_pb: Some(ComponentPb::VisionSkillComponent(VisionSkillComponentPb {
|
component_pb: Some(ComponentPb::VisionSkillComponent(VisionSkillComponentPb {
|
||||||
vision_skill_infos: vec![VisionSkillInformation {
|
vision_skill_infos: vec![VisionSkillInformation {
|
|
@ -1,4 +1,4 @@
|
||||||
use wicked_waifus_protocol::EntityPb;
|
use shorekeeper_protocol::EntityPb;
|
||||||
|
|
||||||
use crate::logic::components::*;
|
use crate::logic::components::*;
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@ macro_rules! impl_component_container {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ComponentContainer {
|
impl ComponentContainer {
|
||||||
pub fn set_pb_data(&self, pb: &mut wicked_waifus_protocol::EntityPb) {
|
pub fn set_pb_data(&self, pb: &mut shorekeeper_protocol::EntityPb) {
|
||||||
match self {
|
match self {
|
||||||
$(
|
$(
|
||||||
Self::$comp(comp) => comp.set_pb_data(pb),
|
Self::$comp(comp) => comp.set_pb_data(pb),
|
||||||
|
@ -28,22 +28,10 @@ impl_component_container! {
|
||||||
OwnerPlayer;
|
OwnerPlayer;
|
||||||
Visibility;
|
Visibility;
|
||||||
Attribute;
|
Attribute;
|
||||||
PlayerOwnedEntityMarker;
|
PlayerEntityMarker;
|
||||||
Movement;
|
Movement;
|
||||||
Equip;
|
Equip;
|
||||||
VisionSkill;
|
VisionSkill;
|
||||||
MonsterAi;
|
|
||||||
Fsm;
|
|
||||||
RoleSkin;
|
|
||||||
FightBuff;
|
|
||||||
StateTag;
|
|
||||||
Tag;
|
|
||||||
Autonomous;
|
|
||||||
Interact;
|
|
||||||
Concomitant;
|
|
||||||
Summoner;
|
|
||||||
SoarWingSkin;
|
|
||||||
ParaglidingSkin;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait Component {
|
pub trait Component {
|
57
game-server/src/logic/ecs/entity.rs
Normal file
57
game-server/src/logic/ecs/entity.rs
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
use std::{cell::RefCell, collections::HashSet};
|
||||||
|
|
||||||
|
use super::component::ComponentContainer;
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
|
||||||
|
pub struct Entity(i64);
|
||||||
|
|
||||||
|
pub struct EntityBuilder<'comp>(Entity, &'comp mut Vec<RefCell<ComponentContainer>>);
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct EntityManager {
|
||||||
|
entity_id_counter: i64,
|
||||||
|
active_entity_set: HashSet<Entity>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EntityManager {
|
||||||
|
pub fn create(&mut self) -> Entity {
|
||||||
|
self.entity_id_counter += 1;
|
||||||
|
let entity = Entity(self.entity_id_counter);
|
||||||
|
|
||||||
|
self.active_entity_set.insert(entity);
|
||||||
|
entity
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get(&self, id: i64) -> Option<Entity> {
|
||||||
|
self.active_entity_set.get(&Entity(id)).copied()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[expect(dead_code)]
|
||||||
|
pub fn remove(&mut self, entity: Entity) -> bool {
|
||||||
|
self.active_entity_set.remove(&entity)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'comp> EntityBuilder<'comp> {
|
||||||
|
pub fn builder(
|
||||||
|
entity: Entity,
|
||||||
|
components: &'comp mut Vec<RefCell<ComponentContainer>>,
|
||||||
|
) -> Self {
|
||||||
|
Self(entity, components)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn with(self, component: ComponentContainer) -> Self {
|
||||||
|
self.1.push(RefCell::new(component));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build(self) -> Entity {
|
||||||
|
self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Entity> for i64 {
|
||||||
|
fn from(value: Entity) -> Self {
|
||||||
|
value.0
|
||||||
|
}
|
||||||
|
}
|
64
game-server/src/logic/ecs/mod.rs
Normal file
64
game-server/src/logic/ecs/mod.rs
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
pub mod component;
|
||||||
|
pub mod entity;
|
||||||
|
pub mod world;
|
||||||
|
|
||||||
|
// Query specified components from all entities
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! query_with {
|
||||||
|
($world:expr, $($comp:ident),*) => {
|
||||||
|
$world.components().iter().filter(|(_, comps)| {
|
||||||
|
$(comps.iter().any(|comp| matches!(&*comp.borrow(), ComponentContainer::$comp(_))) && )* true
|
||||||
|
})
|
||||||
|
.map(|(e, comps)| {
|
||||||
|
(*e,
|
||||||
|
$(
|
||||||
|
comps.iter().find_map(|comp| {
|
||||||
|
let r = comp.try_borrow_mut().ok()?;
|
||||||
|
if matches!(&*r, ComponentContainer::$comp(_)) {
|
||||||
|
Some(::std::cell::RefMut::map(r, |r| {
|
||||||
|
let ComponentContainer::$comp(comp_inner) = r else { unreachable!() };
|
||||||
|
comp_inner
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}).unwrap(),
|
||||||
|
)*
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! ident_as_none {
|
||||||
|
($t:ident) => {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query components of specified entity
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! query_components {
|
||||||
|
($world:expr, $entity_id:expr, $($comp:ident),*) => {
|
||||||
|
$world.components().iter().find(|(id, _)| $entity_id == i64::from(**id))
|
||||||
|
.map(|(_, comps)| {
|
||||||
|
($(
|
||||||
|
comps.iter().find_map(|comp| {
|
||||||
|
let r = comp.try_borrow_mut().ok()?;
|
||||||
|
if matches!(&*r, ComponentContainer::$comp(_)) {
|
||||||
|
Some(::std::cell::RefMut::map(r, |r| {
|
||||||
|
let ComponentContainer::$comp(comp_inner) = r else { unreachable!() };
|
||||||
|
comp_inner
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
)*)
|
||||||
|
})
|
||||||
|
.unwrap_or_else(|| ($( $crate::ident_as_none!($comp), )*))
|
||||||
|
};
|
||||||
|
}
|
58
game-server/src/logic/ecs/world.rs
Normal file
58
game-server/src/logic/ecs/world.rs
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
use std::cell::{RefCell, RefMut};
|
||||||
|
use std::collections::hash_map::{Keys, Values};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
|
use crate::logic::player::InWorldPlayer;
|
||||||
|
|
||||||
|
use super::component::ComponentContainer;
|
||||||
|
use super::entity::{Entity, EntityBuilder, EntityManager};
|
||||||
|
|
||||||
|
pub struct World {
|
||||||
|
components: HashMap<Entity, Vec<RefCell<ComponentContainer>>>,
|
||||||
|
entity_manager: EntityManager,
|
||||||
|
in_world_players: HashMap<i32, InWorldPlayer>, // joined players metadata
|
||||||
|
}
|
||||||
|
|
||||||
|
impl World {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
components: HashMap::new(),
|
||||||
|
entity_manager: EntityManager::default(),
|
||||||
|
in_world_players: HashMap::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_entity(&mut self) -> EntityBuilder {
|
||||||
|
let entity = self.entity_manager.create();
|
||||||
|
EntityBuilder::builder(entity, self.components.entry(entity).or_default())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_in_world(&self, entity_id: i64) -> bool {
|
||||||
|
self.entity_manager.get(entity_id).is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn components(&self) -> &HashMap<Entity, Vec<RefCell<ComponentContainer>>> {
|
||||||
|
&self.components
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_entity_components(&self, entity: Entity) -> Vec<RefMut<ComponentContainer>> {
|
||||||
|
let Some(components) = self.components.get(&entity) else {
|
||||||
|
return Vec::with_capacity(0);
|
||||||
|
};
|
||||||
|
|
||||||
|
components.iter().map(|rc| rc.borrow_mut()).collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn player_ids(&self) -> Keys<'_, i32, InWorldPlayer> {
|
||||||
|
self.in_world_players.keys()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn players(&self) -> Values<'_, i32, InWorldPlayer> {
|
||||||
|
self.in_world_players.values()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_in_world_player_data(&mut self, in_world_player: InWorldPlayer) {
|
||||||
|
self.in_world_players
|
||||||
|
.insert(in_world_player.player_id, in_world_player);
|
||||||
|
}
|
||||||
|
}
|
11
game-server/src/logic/handler/misc.rs
Normal file
11
game-server/src/logic/handler/misc.rs
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
use shorekeeper_protocol::{IYa, InputSettingRequest, InputSettingResponse};
|
||||||
|
|
||||||
|
use crate::logic::player::Player;
|
||||||
|
|
||||||
|
pub fn on_input_setting_request(
|
||||||
|
_: &Player,
|
||||||
|
_: InputSettingRequest,
|
||||||
|
response: &mut InputSettingResponse,
|
||||||
|
) {
|
||||||
|
response.i_ya = Some(IYa::default());
|
||||||
|
}
|
94
game-server/src/logic/handler/mod.rs
Normal file
94
game-server/src/logic/handler/mod.rs
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
mod misc;
|
||||||
|
mod scene;
|
||||||
|
mod skill;
|
||||||
|
pub use misc::*;
|
||||||
|
pub use scene::*;
|
||||||
|
pub use skill::*;
|
||||||
|
|
||||||
|
use shorekeeper_protocol::message::Message;
|
||||||
|
|
||||||
|
macro_rules! handle_request {
|
||||||
|
($($name:ident $(, $inner_package:ident)?;)*) => {
|
||||||
|
fn handle_request(player: &mut super::player::Player, mut msg: Message) {
|
||||||
|
use ::shorekeeper_protocol::{MessageID, Protobuf};
|
||||||
|
|
||||||
|
::paste::paste! {
|
||||||
|
match msg.get_message_id() {
|
||||||
|
$(
|
||||||
|
::shorekeeper_protocol::$($inner_package::)?[<$name Request>]::MESSAGE_ID => {
|
||||||
|
let Ok(request) = ::shorekeeper_protocol::$($inner_package::)?[<$name Request>]::decode(&*msg.remove_payload()) else {
|
||||||
|
tracing::debug!("failed to decode {}, player_id: {}", stringify!($($inner_package::)?[<$name Request>]), player.basic_info.id);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
tracing::debug!("logic: processing request {}", stringify!($($inner_package::)?[<$name Request>]));
|
||||||
|
|
||||||
|
let mut response = ::shorekeeper_protocol::$($inner_package::)?[<$name Response>]::default();
|
||||||
|
[<on_ $($inner_package:snake _)? $name:snake _request>](player, request, &mut response);
|
||||||
|
|
||||||
|
player.respond(response, msg.get_rpc_id());
|
||||||
|
},
|
||||||
|
)*
|
||||||
|
unhandled => ::tracing::warn!("can't find handler for request with message_id={unhandled}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! handle_push {
|
||||||
|
($($name:ident $(, $inner_package:ident)?;)*) => {
|
||||||
|
fn handle_push(player: &mut super::player::Player, mut msg: Message) {
|
||||||
|
use ::shorekeeper_protocol::{MessageID, Protobuf};
|
||||||
|
|
||||||
|
::paste::paste! {
|
||||||
|
match msg.get_message_id() {
|
||||||
|
$(
|
||||||
|
::shorekeeper_protocol::$($inner_package::)?[<$name Push>]::MESSAGE_ID => {
|
||||||
|
let Ok(push) = ::shorekeeper_protocol::$($inner_package::)?[<$name Push>]::decode(&*msg.remove_payload()) else {
|
||||||
|
tracing::debug!("failed to decode {}, player_id: {}", stringify!($($inner_package::)?[<$name Push>]), player.basic_info.id);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
tracing::debug!("logic: processing push {}", stringify!($($inner_package::)?[<$name Push>]));
|
||||||
|
|
||||||
|
[<on_ $($inner_package:snake _)? $name:snake _push>](player, push);
|
||||||
|
},
|
||||||
|
)*
|
||||||
|
unhandled => ::tracing::warn!("can't find handler for push with message_id={unhandled}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
handle_request! {
|
||||||
|
// Scene
|
||||||
|
UpdateSceneDate;
|
||||||
|
EntityActive;
|
||||||
|
EntityOnLanded;
|
||||||
|
CombatSendPack, combat_message;
|
||||||
|
|
||||||
|
// Skill
|
||||||
|
VisionExploreSkillSet;
|
||||||
|
|
||||||
|
// Misc
|
||||||
|
InputSetting;
|
||||||
|
}
|
||||||
|
|
||||||
|
handle_push! {
|
||||||
|
MovePackage;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_logic_message(player: &mut super::player::Player, msg: Message) {
|
||||||
|
match msg {
|
||||||
|
Message::Request { .. } => handle_request(player, msg),
|
||||||
|
Message::Push { .. } => handle_push(player, msg),
|
||||||
|
_ => tracing::warn!(
|
||||||
|
"handle_logic_message: wrong message type: {}, message_id: {}, player_id: {}",
|
||||||
|
msg.get_message_type(),
|
||||||
|
msg.get_message_id(),
|
||||||
|
player.basic_info.id,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
}
|
87
game-server/src/logic/handler/scene.rs
Normal file
87
game-server/src/logic/handler/scene.rs
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
use shorekeeper_protocol::combat_message::{CombatSendPackRequest, CombatSendPackResponse};
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
EntityActiveRequest, EntityActiveResponse, EntityOnLandedRequest, EntityOnLandedResponse,
|
||||||
|
ErrorCode, MovePackagePush, UpdateSceneDateRequest, UpdateSceneDateResponse,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{logic::ecs::component::ComponentContainer, logic::player::Player, query_components};
|
||||||
|
|
||||||
|
pub fn on_update_scene_date_request(
|
||||||
|
_player: &Player,
|
||||||
|
_request: UpdateSceneDateRequest,
|
||||||
|
response: &mut UpdateSceneDateResponse,
|
||||||
|
) {
|
||||||
|
response.error_code = ErrorCode::Success.into();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_combat_message_combat_send_pack_request(
|
||||||
|
_player: &Player,
|
||||||
|
_request: CombatSendPackRequest,
|
||||||
|
response: &mut CombatSendPackResponse,
|
||||||
|
) {
|
||||||
|
response.error_code = ErrorCode::Success.into();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_entity_active_request(
|
||||||
|
player: &Player,
|
||||||
|
request: EntityActiveRequest,
|
||||||
|
response: &mut EntityActiveResponse,
|
||||||
|
) {
|
||||||
|
let world = player.world.borrow();
|
||||||
|
|
||||||
|
if !world.is_in_world(request.entity_id) {
|
||||||
|
tracing::debug!(
|
||||||
|
"EntityActiveRequest: entity with id {} doesn't exist, player_id: {}",
|
||||||
|
request.entity_id,
|
||||||
|
player.basic_info.id
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(position) = query_components!(world, request.entity_id, Position).0 {
|
||||||
|
// TODO: proper entity "activation" logic
|
||||||
|
response.pos = Some(position.0.get_position_protobuf());
|
||||||
|
response.rot = Some(position.0.get_rotation_protobuf());
|
||||||
|
}
|
||||||
|
|
||||||
|
response.component_pbs = Vec::new(); // not implemented
|
||||||
|
response.error_code = ErrorCode::Success.into();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_entity_on_landed_request(
|
||||||
|
_: &Player,
|
||||||
|
request: EntityOnLandedRequest,
|
||||||
|
_: &mut EntityOnLandedResponse,
|
||||||
|
) {
|
||||||
|
tracing::debug!(
|
||||||
|
"EntityOnLandedRequest: entity with id {} landed",
|
||||||
|
request.entity_id
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_move_package_push(player: &mut Player, push: MovePackagePush) {
|
||||||
|
let world = player.world.borrow();
|
||||||
|
|
||||||
|
for moving_entity in push.moving_entities {
|
||||||
|
if !world.is_in_world(moving_entity.entity_id) {
|
||||||
|
tracing::debug!(
|
||||||
|
"MovePackage: entity with id {} doesn't exist",
|
||||||
|
moving_entity.entity_id
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let Some(mut movement) = query_components!(world, moving_entity.entity_id, Movement).0
|
||||||
|
else {
|
||||||
|
tracing::warn!(
|
||||||
|
"MovePackage: entity {} doesn't have movement component",
|
||||||
|
moving_entity.entity_id
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
movement
|
||||||
|
.pending_movement_vec
|
||||||
|
.extend(moving_entity.move_infos);
|
||||||
|
}
|
||||||
|
}
|
32
game-server/src/logic/handler/skill.rs
Normal file
32
game-server/src/logic/handler/skill.rs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
use crate::logic::ecs::component::ComponentContainer;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
VisionExploreSkillSetRequest, VisionExploreSkillSetResponse, VisionSkillChangeNotify,
|
||||||
|
VisionSkillInformation,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{logic::player::Player, query_with};
|
||||||
|
|
||||||
|
pub fn on_vision_explore_skill_set_request(
|
||||||
|
player: &mut Player,
|
||||||
|
request: VisionExploreSkillSetRequest,
|
||||||
|
response: &mut VisionExploreSkillSetResponse,
|
||||||
|
) {
|
||||||
|
player.explore_tools.active_explore_skill = request.skill_id;
|
||||||
|
|
||||||
|
let world = player.world.borrow();
|
||||||
|
for (entity, owner, mut vision_skill) in query_with!(world, OwnerPlayer, VisionSkill) {
|
||||||
|
if owner.0 == player.basic_info.id {
|
||||||
|
vision_skill.skill_id = request.skill_id;
|
||||||
|
player.notify(VisionSkillChangeNotify {
|
||||||
|
entity_id: entity.into(),
|
||||||
|
vision_skill_infos: vec![VisionSkillInformation {
|
||||||
|
skill_id: request.skill_id,
|
||||||
|
..Default::default()
|
||||||
|
}],
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
response.skill_id = request.skill_id;
|
||||||
|
}
|
|
@ -1,6 +1,5 @@
|
||||||
mod transform;
|
mod transform;
|
||||||
mod vector;
|
mod vector;
|
||||||
mod area_range;
|
|
||||||
|
|
||||||
pub use transform::Transform;
|
pub use transform::Transform;
|
||||||
pub use vector::Vector3f;
|
pub use vector::Vector3f;
|
|
@ -1,6 +1,4 @@
|
||||||
use wicked_waifus_data::RawVectorData;
|
use shorekeeper_protocol::{Rotator, TransformData};
|
||||||
use wicked_waifus_protocol::Rotator;
|
|
||||||
use wicked_waifus_protocol_internal::TransformData;
|
|
||||||
|
|
||||||
use super::Vector3f;
|
use super::Vector3f;
|
||||||
|
|
||||||
|
@ -11,19 +9,19 @@ pub struct Transform {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Transform {
|
impl Transform {
|
||||||
pub fn get_position_protobuf(&self) -> wicked_waifus_protocol::Vector {
|
pub fn get_position_protobuf(&self) -> shorekeeper_protocol::Vector {
|
||||||
self.position.to_protobuf()
|
self.position.to_protobuf()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_rotation_protobuf(&self) -> Rotator {
|
pub fn get_rotation_protobuf(&self) -> Rotator {
|
||||||
Rotator {
|
Rotator {
|
||||||
roll: self.rotation.x,
|
pitch: self.rotation.x,
|
||||||
pitch: self.rotation.y,
|
yaw: self.rotation.y,
|
||||||
yaw: self.rotation.z,
|
roll: self.rotation.z,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_position_from_protobuf(&mut self, pos: &wicked_waifus_protocol::Vector) {
|
pub fn set_position_from_protobuf(&mut self, pos: &shorekeeper_protocol::Vector) {
|
||||||
self.position.x = pos.x;
|
self.position.x = pos.x;
|
||||||
self.position.y = pos.y;
|
self.position.y = pos.y;
|
||||||
self.position.z = pos.z;
|
self.position.z = pos.z;
|
||||||
|
@ -49,15 +47,3 @@ impl Transform {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<&[RawVectorData]> for Transform {
|
|
||||||
// - transform[0] is position component
|
|
||||||
// - transform[1] is rotation component
|
|
||||||
// - transform[2] is scale component
|
|
||||||
fn from(transform: &[RawVectorData]) -> Self {
|
|
||||||
Self {
|
|
||||||
position: Vector3f::from_raw_scaled(&transform[0], &transform[2]),
|
|
||||||
rotation: Vector3f::from_raw_scaled(&transform[1], &transform[2]),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
42
game-server/src/logic/math/vector.rs
Normal file
42
game-server/src/logic/math/vector.rs
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
use shorekeeper_protocol::{Vector, VectorData};
|
||||||
|
|
||||||
|
#[derive(Default, Clone, PartialEq, Debug)]
|
||||||
|
pub struct Vector3f {
|
||||||
|
pub x: f32,
|
||||||
|
pub y: f32,
|
||||||
|
pub z: f32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Vector3f {
|
||||||
|
pub fn to_protobuf(&self) -> Vector {
|
||||||
|
Vector {
|
||||||
|
x: self.x,
|
||||||
|
y: self.y,
|
||||||
|
z: self.z,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_save(data: VectorData) -> Self {
|
||||||
|
Self {
|
||||||
|
x: data.x,
|
||||||
|
y: data.y,
|
||||||
|
z: data.z,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn save_data(&self) -> VectorData {
|
||||||
|
VectorData {
|
||||||
|
x: self.x,
|
||||||
|
y: self.y,
|
||||||
|
z: self.z,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_data(data: &shorekeeper_data::VectorData) -> Self {
|
||||||
|
Self {
|
||||||
|
x: data.get_x(),
|
||||||
|
y: data.get_y(),
|
||||||
|
z: data.get_z(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,4 +7,3 @@ pub mod role;
|
||||||
pub mod systems;
|
pub mod systems;
|
||||||
pub mod thread_mgr;
|
pub mod thread_mgr;
|
||||||
pub mod utils;
|
pub mod utils;
|
||||||
pub mod gacha;
|
|
|
@ -1,8 +1,6 @@
|
||||||
use wicked_waifus_protocol::{
|
use shorekeeper_protocol::{
|
||||||
player_attr, BasicInfoNotify, PlayerAttr, PlayerAttrKey, PlayerAttrType,
|
player_attr, BasicInfoNotify, PlayerAttr, PlayerAttrKey, PlayerAttrType, PlayerBasicData,
|
||||||
};
|
};
|
||||||
use wicked_waifus_protocol_internal::PlayerBasicData;
|
|
||||||
use crate::logic::player::player_inventory::PlayerInventory;
|
|
||||||
|
|
||||||
pub struct PlayerBasicInfo {
|
pub struct PlayerBasicInfo {
|
||||||
pub id: i32,
|
pub id: i32,
|
||||||
|
@ -12,33 +10,20 @@ pub struct PlayerBasicInfo {
|
||||||
pub exp: i32,
|
pub exp: i32,
|
||||||
pub head_photo: i32,
|
pub head_photo: i32,
|
||||||
pub head_frame: i32,
|
pub head_frame: i32,
|
||||||
pub cur_map_id: i32,
|
|
||||||
pub role_show_list: Vec<i32>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PlayerBasicInfo {
|
impl PlayerBasicInfo {
|
||||||
pub fn build_notify(&self, inventory: &PlayerInventory) -> BasicInfoNotify {
|
pub fn build_notify(&self) -> BasicInfoNotify {
|
||||||
BasicInfoNotify {
|
BasicInfoNotify {
|
||||||
id: self.id,
|
id: self.id,
|
||||||
attributes: vec![
|
attributes: vec![
|
||||||
|
build_str_attr(PlayerAttrKey::Name, self.name.as_str()),
|
||||||
build_int_attr(PlayerAttrKey::Level, self.level),
|
build_int_attr(PlayerAttrKey::Level, self.level),
|
||||||
build_int_attr(PlayerAttrKey::Exp, self.exp),
|
build_int_attr(PlayerAttrKey::Exp, self.exp),
|
||||||
build_int_attr(PlayerAttrKey::Coin, inventory.get_shell_credits()),
|
build_int_attr(PlayerAttrKey::Sex, self.sex),
|
||||||
build_int_attr(PlayerAttrKey::RareCoin, inventory.get_astrite()),
|
|
||||||
build_int_attr(PlayerAttrKey::HeadPhoto, self.head_photo),
|
build_int_attr(PlayerAttrKey::HeadPhoto, self.head_photo),
|
||||||
build_int_attr(PlayerAttrKey::HeadFrame, self.head_frame),
|
build_int_attr(PlayerAttrKey::HeadFrame, self.head_frame),
|
||||||
build_int_attr(PlayerAttrKey::AreaId, 1), // TODO:
|
|
||||||
build_str_attr(PlayerAttrKey::Name, self.name.as_str()),
|
|
||||||
build_str_attr(PlayerAttrKey::Sign, ""), // TODO:
|
|
||||||
build_int_attr(PlayerAttrKey::Sex, self.sex),
|
|
||||||
build_int_attr(PlayerAttrKey::OriginWorldLevel, 1), // TODO:
|
|
||||||
build_int_attr(PlayerAttrKey::CurWorldLevel, 1), // TODO:
|
|
||||||
build_int_attr(PlayerAttrKey::WorldLevelTimeStamp, 0), // TODO:
|
|
||||||
build_int_attr(PlayerAttrKey::CashCoin, inventory.get_lunite()),
|
|
||||||
build_int_attr(PlayerAttrKey::WorldPermission, 0), // TODO:
|
|
||||||
],
|
],
|
||||||
card_unlock_list: vec![], // TODO: 80060000
|
|
||||||
cur_card_id: 0, // TODO: 80060000
|
|
||||||
..Default::default()
|
..Default::default()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -52,8 +37,6 @@ impl PlayerBasicInfo {
|
||||||
exp: data.exp,
|
exp: data.exp,
|
||||||
head_photo: data.head_photo,
|
head_photo: data.head_photo,
|
||||||
head_frame: data.head_frame,
|
head_frame: data.head_frame,
|
||||||
cur_map_id: data.cur_map_id,
|
|
||||||
role_show_list: data.role_show_list,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -66,24 +49,6 @@ impl PlayerBasicInfo {
|
||||||
exp: self.exp,
|
exp: self.exp,
|
||||||
head_photo: self.head_photo,
|
head_photo: self.head_photo,
|
||||||
head_frame: self.head_frame,
|
head_frame: self.head_frame,
|
||||||
cur_map_id: self.cur_map_id,
|
|
||||||
role_show_list: self.role_show_list.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for PlayerBasicInfo {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self {
|
|
||||||
id: 0,
|
|
||||||
name: "".to_string(),
|
|
||||||
sex: 0,
|
|
||||||
level: 0,
|
|
||||||
exp: 0,
|
|
||||||
head_photo: 0,
|
|
||||||
head_frame: 0,
|
|
||||||
cur_map_id: 0,
|
|
||||||
role_show_list: vec![],
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,8 +1,10 @@
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
|
|
||||||
use wicked_waifus_data::explore_tools_data;
|
use shorekeeper_data::explore_tools_data;
|
||||||
use wicked_waifus_protocol::{ExploreSkillRoulette, ExploreSkillRouletteUpdateNotify, ExploreToolAllNotify, VisionExploreSkillNotify};
|
use shorekeeper_protocol::{
|
||||||
use wicked_waifus_protocol_internal::PlayerExploreToolsData;
|
ExploreSkillRoulette, ExploreSkillRouletteUpdateNotify, ExploreToolAllNotify,
|
||||||
|
PlayerExploreToolsData,
|
||||||
|
};
|
||||||
|
|
||||||
type Roulette = [i32; 8];
|
type Roulette = [i32; 8];
|
||||||
|
|
||||||
|
@ -33,7 +35,6 @@ impl ExploreTools {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn build_explore_tool_all_notify(&self) -> ExploreToolAllNotify {
|
pub fn build_explore_tool_all_notify(&self) -> ExploreToolAllNotify {
|
||||||
// TODO: NewUnlock handling
|
|
||||||
ExploreToolAllNotify {
|
ExploreToolAllNotify {
|
||||||
skill_list: self.unlocked_explore_skills.iter().cloned().collect(),
|
skill_list: self.unlocked_explore_skills.iter().cloned().collect(),
|
||||||
explore_skill: self.active_explore_skill,
|
explore_skill: self.active_explore_skill,
|
||||||
|
@ -41,12 +42,6 @@ impl ExploreTools {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn build_vision_explore_skill_notify(&self) -> VisionExploreSkillNotify {
|
|
||||||
VisionExploreSkillNotify {
|
|
||||||
explore_skill: self.active_explore_skill
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn build_roulette_update_notify(&self) -> ExploreSkillRouletteUpdateNotify {
|
pub fn build_roulette_update_notify(&self) -> ExploreSkillRouletteUpdateNotify {
|
||||||
ExploreSkillRouletteUpdateNotify {
|
ExploreSkillRouletteUpdateNotify {
|
||||||
roulette_info: vec![ExploreSkillRoulette {
|
roulette_info: vec![ExploreSkillRoulette {
|
|
@ -1,5 +1,5 @@
|
||||||
use wicked_waifus_data::instance_dungeon_data;
|
use shorekeeper_data::instance_dungeon_data;
|
||||||
use wicked_waifus_protocol_internal::PlayerLocationData;
|
use shorekeeper_protocol::PlayerLocationData;
|
||||||
|
|
||||||
use crate::logic::math::{Transform, Vector3f};
|
use crate::logic::math::{Transform, Vector3f};
|
||||||
|
|
268
game-server/src/logic/player/mod.rs
Normal file
268
game-server/src/logic/player/mod.rs
Normal file
|
@ -0,0 +1,268 @@
|
||||||
|
use std::{cell::RefCell, collections::HashSet, rc::Rc, sync::Arc};
|
||||||
|
|
||||||
|
use basic_info::PlayerBasicInfo;
|
||||||
|
use common::time_util;
|
||||||
|
use explore_tools::ExploreTools;
|
||||||
|
use location::PlayerLocation;
|
||||||
|
use player_func::PlayerFunc;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
message::Message, ItemPkgOpenNotify, PbGetRoleListNotify, PlayerBasicData, PlayerRoleData,
|
||||||
|
PlayerSaveData, ProtocolUnit,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::session::Session;
|
||||||
|
|
||||||
|
use super::{
|
||||||
|
ecs::world::World,
|
||||||
|
role::{Role, RoleFormation},
|
||||||
|
};
|
||||||
|
|
||||||
|
mod basic_info;
|
||||||
|
mod explore_tools;
|
||||||
|
mod in_world_player;
|
||||||
|
mod location;
|
||||||
|
mod player_func;
|
||||||
|
|
||||||
|
pub use in_world_player::InWorldPlayer;
|
||||||
|
|
||||||
|
pub struct Player {
|
||||||
|
session: Option<Arc<Session>>,
|
||||||
|
// Persistent
|
||||||
|
pub basic_info: PlayerBasicInfo,
|
||||||
|
pub role_list: Vec<Role>,
|
||||||
|
pub formation_list: Vec<RoleFormation>,
|
||||||
|
pub location: PlayerLocation,
|
||||||
|
pub func: PlayerFunc,
|
||||||
|
pub explore_tools: ExploreTools,
|
||||||
|
// Runtime
|
||||||
|
pub world: Rc<RefCell<World>>,
|
||||||
|
pub last_save_time: u64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Player {
|
||||||
|
pub fn init(&mut self) {
|
||||||
|
if self.role_list.is_empty() {
|
||||||
|
self.on_first_enter();
|
||||||
|
}
|
||||||
|
|
||||||
|
// we need shorekeeper
|
||||||
|
// TODO: remove this part after implementing team switch
|
||||||
|
if !self.role_list.iter().any(|r| r.role_id == 1505) {
|
||||||
|
let mut shorekeeper = Role::new(1505);
|
||||||
|
shorekeeper.equip_weapon = 21050036;
|
||||||
|
self.role_list.push(shorekeeper);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.formation_list.clear();
|
||||||
|
self.formation_list.push(RoleFormation {
|
||||||
|
id: 1,
|
||||||
|
cur_role: 1505,
|
||||||
|
role_id_set: HashSet::from([1505]),
|
||||||
|
is_current: true,
|
||||||
|
});
|
||||||
|
// End shorekeeper hardcode part
|
||||||
|
|
||||||
|
self.ensure_current_formation();
|
||||||
|
self.ensure_basic_unlock_func();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn notify_general_data(&self) {
|
||||||
|
self.notify(self.basic_info.build_notify());
|
||||||
|
self.notify(self.func.build_func_open_notify());
|
||||||
|
self.notify(self.build_role_list_notify());
|
||||||
|
self.notify(self.explore_tools.build_explore_tool_all_notify());
|
||||||
|
self.notify(self.explore_tools.build_roulette_update_notify());
|
||||||
|
|
||||||
|
self.notify(ItemPkgOpenNotify {
|
||||||
|
open_pkg: (0..8).collect(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_first_enter(&mut self) {
|
||||||
|
self.role_list.push(Self::create_main_character_role(
|
||||||
|
self.basic_info.name.clone(),
|
||||||
|
self.basic_info.sex,
|
||||||
|
));
|
||||||
|
|
||||||
|
let role = &self.role_list[0];
|
||||||
|
|
||||||
|
self.formation_list.push(RoleFormation {
|
||||||
|
id: 1,
|
||||||
|
cur_role: role.role_id,
|
||||||
|
role_id_set: HashSet::from([role.role_id]),
|
||||||
|
is_current: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
self.location = PlayerLocation::default();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure basic functionality is unlocked
|
||||||
|
// Should be handled by quest progression,
|
||||||
|
// but as of right now, just unlock what we need
|
||||||
|
fn ensure_basic_unlock_func(&mut self) {
|
||||||
|
self.func.unlock(10026); // explore tools
|
||||||
|
}
|
||||||
|
|
||||||
|
fn ensure_current_formation(&mut self) {
|
||||||
|
if self.formation_list.is_empty() {
|
||||||
|
let role = &self.role_list[0];
|
||||||
|
|
||||||
|
self.formation_list.push(RoleFormation {
|
||||||
|
id: 1,
|
||||||
|
cur_role: role.role_id,
|
||||||
|
role_id_set: HashSet::from([role.role_id]),
|
||||||
|
is_current: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if !self.formation_list.iter().any(|rf| rf.is_current) {
|
||||||
|
self.formation_list[0].is_current = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(rf) = self.formation_list.iter_mut().find(|rf| rf.is_current) {
|
||||||
|
if rf.role_id_set.is_empty() {
|
||||||
|
rf.role_id_set.insert(self.role_list[0].role_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if !rf.role_id_set.contains(&rf.cur_role) {
|
||||||
|
rf.cur_role = *rf.role_id_set.iter().nth(0).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_in_world_player(&self) -> InWorldPlayer {
|
||||||
|
InWorldPlayer {
|
||||||
|
player_id: self.basic_info.id,
|
||||||
|
player_name: self.basic_info.name.clone(),
|
||||||
|
player_icon: 0,
|
||||||
|
level: self.basic_info.level,
|
||||||
|
group_type: 1,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_formation_role_list(&self) -> Vec<&Role> {
|
||||||
|
self.formation_list
|
||||||
|
.iter()
|
||||||
|
.find(|rf| rf.is_current)
|
||||||
|
.unwrap()
|
||||||
|
.role_id_set
|
||||||
|
.iter()
|
||||||
|
.flat_map(|id| self.role_list.iter().find(|r| r.role_id == *id))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_cur_role_id(&self) -> i32 {
|
||||||
|
self.formation_list
|
||||||
|
.iter()
|
||||||
|
.find(|rf| rf.is_current)
|
||||||
|
.unwrap()
|
||||||
|
.cur_role
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn load_from_save(save_data: PlayerSaveData) -> Self {
|
||||||
|
let role_data = save_data.role_data.unwrap_or_default();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
session: None,
|
||||||
|
basic_info: PlayerBasicInfo::load_from_save(save_data.basic_data.unwrap_or_default()),
|
||||||
|
role_list: role_data
|
||||||
|
.role_list
|
||||||
|
.into_iter()
|
||||||
|
.map(Role::load_from_save)
|
||||||
|
.collect(),
|
||||||
|
formation_list: role_data
|
||||||
|
.role_formation_list
|
||||||
|
.into_iter()
|
||||||
|
.map(RoleFormation::load_from_save)
|
||||||
|
.collect(),
|
||||||
|
location: save_data
|
||||||
|
.location_data
|
||||||
|
.map(PlayerLocation::load_from_save)
|
||||||
|
.unwrap_or_default(),
|
||||||
|
func: save_data
|
||||||
|
.func_data
|
||||||
|
.map(PlayerFunc::load_from_save)
|
||||||
|
.unwrap_or_default(),
|
||||||
|
explore_tools: save_data
|
||||||
|
.explore_tools_data
|
||||||
|
.map(ExploreTools::load_from_save)
|
||||||
|
.unwrap_or_default(),
|
||||||
|
world: Rc::new(RefCell::new(World::new())),
|
||||||
|
last_save_time: time_util::unix_timestamp(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_save_data(&self) -> PlayerSaveData {
|
||||||
|
PlayerSaveData {
|
||||||
|
basic_data: Some(self.basic_info.build_save_data()),
|
||||||
|
role_data: Some(PlayerRoleData {
|
||||||
|
role_list: self.role_list.iter().map(|r| r.build_save_data()).collect(),
|
||||||
|
role_formation_list: self
|
||||||
|
.formation_list
|
||||||
|
.iter()
|
||||||
|
.map(|rf| rf.build_save_data())
|
||||||
|
.collect(),
|
||||||
|
}),
|
||||||
|
location_data: Some(self.location.build_save_data()),
|
||||||
|
func_data: Some(self.func.build_save_data()),
|
||||||
|
explore_tools_data: Some(self.explore_tools.build_save_data()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_session(&mut self, session: Arc<Session>) {
|
||||||
|
self.session = Some(session);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_role_list_notify(&self) -> PbGetRoleListNotify {
|
||||||
|
PbGetRoleListNotify {
|
||||||
|
role_list: self.role_list.iter().map(|r| r.to_protobuf()).collect(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn notify(&self, content: impl ProtocolUnit) {
|
||||||
|
if let Some(session) = self.session.as_ref() {
|
||||||
|
session.forward_to_gateway(Message::Push {
|
||||||
|
sequence_number: 0,
|
||||||
|
message_id: content.get_message_id(),
|
||||||
|
payload: Some(content.encode_to_vec().into_boxed_slice()),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn respond(&self, content: impl ProtocolUnit, rpc_id: u16) {
|
||||||
|
if let Some(session) = self.session.as_ref() {
|
||||||
|
session.forward_to_gateway(Message::Response {
|
||||||
|
sequence_number: 0,
|
||||||
|
message_id: content.get_message_id(),
|
||||||
|
rpc_id,
|
||||||
|
payload: Some(content.encode_to_vec().into_boxed_slice()),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_main_character_role(name: String, sex: i32) -> Role {
|
||||||
|
let mut role = match sex {
|
||||||
|
0 => Role::new(Role::MAIN_CHARACTER_FEMALE_ID),
|
||||||
|
1 => Role::new(Role::MAIN_CHARACTER_MALE_ID),
|
||||||
|
_ => unreachable!(),
|
||||||
|
};
|
||||||
|
|
||||||
|
role.name = name;
|
||||||
|
role
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_default_save_data(id: i32, name: String, sex: i32) -> PlayerSaveData {
|
||||||
|
PlayerSaveData {
|
||||||
|
basic_data: Some(PlayerBasicData {
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
sex,
|
||||||
|
level: 1,
|
||||||
|
head_photo: 1505,
|
||||||
|
head_frame: 80060009,
|
||||||
|
..Default::default()
|
||||||
|
}),
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,9 +1,7 @@
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use wicked_waifus_data::function_condition_data;
|
use shorekeeper_data::function_condition_data;
|
||||||
use wicked_waifus_protocol::{FuncOpenNotify, Function};
|
use shorekeeper_protocol::{FuncOpenNotify, Function, PlayerFuncData};
|
||||||
use wicked_waifus_protocol_internal::PlayerFuncData;
|
|
||||||
use crate::config;
|
|
||||||
|
|
||||||
pub struct PlayerFunc {
|
pub struct PlayerFunc {
|
||||||
pub func_map: HashMap<i32, i32>,
|
pub func_map: HashMap<i32, i32>,
|
||||||
|
@ -42,19 +40,11 @@ impl PlayerFunc {
|
||||||
|
|
||||||
impl Default for PlayerFunc {
|
impl Default for PlayerFunc {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
if config::get_config().default_unlocks.unlock_all_functions {
|
|
||||||
Self {
|
Self {
|
||||||
func_map: function_condition_data::iter()
|
func_map: function_condition_data::iter()
|
||||||
|
.filter(|fc| fc.open_condition_id == 0 && fc.is_on)
|
||||||
.map(|fc| (fc.function_id, 2))
|
.map(|fc| (fc.function_id, 2))
|
||||||
.collect(),
|
.collect(),
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
Self {
|
|
||||||
func_map: function_condition_data::iter()
|
|
||||||
.filter(|fc| fc.open_condition_id == 0)
|
|
||||||
.map(|fc| (fc.function_id, 2))
|
|
||||||
.collect(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
30
game-server/src/logic/role/formation.rs
Normal file
30
game-server/src/logic/role/formation.rs
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use shorekeeper_protocol::RoleFormationData;
|
||||||
|
|
||||||
|
pub struct RoleFormation {
|
||||||
|
pub id: i32,
|
||||||
|
pub cur_role: i32,
|
||||||
|
pub role_id_set: HashSet<i32>,
|
||||||
|
pub is_current: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RoleFormation {
|
||||||
|
pub fn load_from_save(data: RoleFormationData) -> Self {
|
||||||
|
Self {
|
||||||
|
id: data.formation_id,
|
||||||
|
cur_role: data.cur_role,
|
||||||
|
role_id_set: data.role_id_list.into_iter().collect(),
|
||||||
|
is_current: data.is_current,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_save_data(&self) -> RoleFormationData {
|
||||||
|
RoleFormationData {
|
||||||
|
formation_id: self.id,
|
||||||
|
cur_role: self.cur_role,
|
||||||
|
role_id_list: self.role_id_set.iter().cloned().collect(),
|
||||||
|
is_current: self.is_current,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
92
game-server/src/logic/role/mod.rs
Normal file
92
game-server/src/logic/role/mod.rs
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
|
use common::time_util;
|
||||||
|
pub use formation::RoleFormation;
|
||||||
|
use shorekeeper_data::role_info_data;
|
||||||
|
use shorekeeper_protocol::{ArrayIntInt, RoleData, RoleInfo};
|
||||||
|
|
||||||
|
mod formation;
|
||||||
|
pub struct Role {
|
||||||
|
pub role_id: i32,
|
||||||
|
pub name: String,
|
||||||
|
pub level: i32,
|
||||||
|
pub exp: i32,
|
||||||
|
pub breakthrough: i32,
|
||||||
|
pub skill_map: HashMap<i32, i32>,
|
||||||
|
pub star: i32,
|
||||||
|
pub favor: i32,
|
||||||
|
pub create_time: u32,
|
||||||
|
pub equip_weapon: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Role {
|
||||||
|
pub const MAIN_CHARACTER_MALE_ID: i32 = 1501;
|
||||||
|
pub const MAIN_CHARACTER_FEMALE_ID: i32 = 1502;
|
||||||
|
|
||||||
|
pub fn new(role_id: i32) -> Self {
|
||||||
|
let data = role_info_data::iter().find(|d| d.id == role_id).unwrap();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
role_id,
|
||||||
|
name: String::with_capacity(0),
|
||||||
|
level: data.max_level,
|
||||||
|
exp: 0,
|
||||||
|
breakthrough: 0,
|
||||||
|
skill_map: HashMap::new(), // TODO!
|
||||||
|
star: 0,
|
||||||
|
favor: 0,
|
||||||
|
create_time: time_util::unix_timestamp() as u32,
|
||||||
|
equip_weapon: data.init_weapon_item_id,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn to_protobuf(&self) -> RoleInfo {
|
||||||
|
RoleInfo {
|
||||||
|
role_id: self.role_id,
|
||||||
|
name: self.name.clone(),
|
||||||
|
level: self.level,
|
||||||
|
exp: self.exp,
|
||||||
|
breakthrough: self.breakthrough,
|
||||||
|
create_time: self.create_time,
|
||||||
|
skills: self
|
||||||
|
.skill_map
|
||||||
|
.iter()
|
||||||
|
.map(|(k, v)| ArrayIntInt { key: *k, value: *v })
|
||||||
|
.collect(),
|
||||||
|
star: self.star,
|
||||||
|
favor: self.favor,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn load_from_save(data: RoleData) -> Self {
|
||||||
|
Self {
|
||||||
|
role_id: data.role_id,
|
||||||
|
name: data.name,
|
||||||
|
level: data.level,
|
||||||
|
exp: data.exp,
|
||||||
|
breakthrough: data.breakthrough,
|
||||||
|
skill_map: data.skill_map,
|
||||||
|
star: data.star,
|
||||||
|
favor: data.favor,
|
||||||
|
create_time: data.create_time,
|
||||||
|
equip_weapon: data.equip_weapon,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_save_data(&self) -> RoleData {
|
||||||
|
RoleData {
|
||||||
|
role_id: self.role_id,
|
||||||
|
name: self.name.clone(),
|
||||||
|
level: self.level,
|
||||||
|
exp: self.exp,
|
||||||
|
breakthrough: self.breakthrough,
|
||||||
|
skill_map: self.skill_map.clone(),
|
||||||
|
star: self.star,
|
||||||
|
favor: self.favor,
|
||||||
|
create_time: self.create_time,
|
||||||
|
equip_weapon: self.equip_weapon,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
use wicked_waifus_protocol::{MovePackageNotify, MovingEntityData};
|
use shorekeeper_protocol::{MovePackageNotify, MovingEntityData};
|
||||||
|
|
||||||
use crate::{logic::ecs::component::ComponentContainer, query_components};
|
use crate::{logic::ecs::component::ComponentContainer, query_components};
|
||||||
use std::cell::RefMut;
|
use std::cell::RefMut;
|
||||||
|
@ -15,9 +15,8 @@ pub(super) struct MovementSystem;
|
||||||
impl System for MovementSystem {
|
impl System for MovementSystem {
|
||||||
fn tick(&self, world: &mut World, players: &mut [RefMut<Player>]) {
|
fn tick(&self, world: &mut World, players: &mut [RefMut<Player>]) {
|
||||||
let mut notify = MovePackageNotify::default();
|
let mut notify = MovePackageNotify::default();
|
||||||
let world_entity = world.get_world_entity();
|
|
||||||
|
|
||||||
for (entity, mut movement, mut position) in query_with!(world_entity, Movement, Position) {
|
for (entity, mut movement, mut position) in query_with!(world, Movement, Position) {
|
||||||
if movement.pending_movement_vec.is_empty() {
|
if movement.pending_movement_vec.is_empty() {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -47,12 +46,9 @@ impl System for MovementSystem {
|
||||||
|
|
||||||
notify.moving_entities.push(moving_entity_data);
|
notify.moving_entities.push(moving_entity_data);
|
||||||
|
|
||||||
if let (Some(_), Some(owner)) = query_components!(
|
if let (Some(_), Some(owner)) =
|
||||||
world_entity,
|
query_components!(world, i64::from(entity), PlayerEntityMarker, OwnerPlayer)
|
||||||
i64::from(entity),
|
{
|
||||||
PlayerOwnedEntityMarker,
|
|
||||||
OwnerPlayer
|
|
||||||
) {
|
|
||||||
if let Some(player) = players.iter_mut().find(|pl| pl.basic_info.id == owner.0) {
|
if let Some(player) = players.iter_mut().find(|pl| pl.basic_info.id == owner.0) {
|
||||||
player.location.position = position.0.clone();
|
player.location.position = position.0.clone();
|
||||||
}
|
}
|
|
@ -1,7 +1,3 @@
|
||||||
use wicked_waifus_commons::time_util;
|
|
||||||
use wicked_waifus_protocol_internal::PlayerSaveData;
|
|
||||||
use wicked_waifus_protocol::{message::Message, AfterJoinSceneNotify, EnterGameResponse, JoinSceneNotify, SilenceNpcNotify, TransitionOptionPb};
|
|
||||||
use std::collections::hash_map::Entry::Vacant;
|
|
||||||
use std::{
|
use std::{
|
||||||
cell::RefCell,
|
cell::RefCell,
|
||||||
collections::{HashMap, VecDeque},
|
collections::{HashMap, VecDeque},
|
||||||
|
@ -13,9 +9,24 @@ use std::{
|
||||||
thread,
|
thread,
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use common::time_util;
|
||||||
|
use shorekeeper_protocol::PlayerSaveData;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
message::Message, AfterJoinSceneNotify, EnterGameResponse, JoinSceneNotify, JsPatchNotify,
|
||||||
|
TransitionOptionPb,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
player_save_task::{self, PlayerSaveReason},
|
||||||
|
session::Session,
|
||||||
|
};
|
||||||
|
|
||||||
use super::{ecs::world::World, player::Player, utils::world_util};
|
use super::{ecs::world::World, player::Player, utils::world_util};
|
||||||
use crate::logic::ecs::world::WorldEntity;
|
|
||||||
use crate::{logic, player_save_task::{self, PlayerSaveReason}, session::Session};
|
const WATER_MASK: &str = include_str!("../../watermask.js");
|
||||||
|
const UID_FIX: &str = include_str!("../../uidfix.js");
|
||||||
|
const CENSORSHIP_FIX: &str = include_str!("../../censorshipfix.js");
|
||||||
|
|
||||||
pub enum LogicInput {
|
pub enum LogicInput {
|
||||||
AddPlayer {
|
AddPlayer {
|
||||||
|
@ -130,29 +141,13 @@ fn handle_logic_input(state: &mut LogicState, input: LogicInput) {
|
||||||
session,
|
session,
|
||||||
player_save_data,
|
player_save_data,
|
||||||
} => {
|
} => {
|
||||||
let (player, is_player) = if let Vacant(e) = state.players.entry(player_id) {
|
let player = state
|
||||||
(
|
.players
|
||||||
e.insert(RefCell::new(Player::load_from_save(player_save_data))),
|
.entry(player_id)
|
||||||
true,
|
.or_insert(RefCell::new(Player::load_from_save(player_save_data)));
|
||||||
)
|
|
||||||
} else {
|
|
||||||
if let Some(player) = state.players.get_mut(&player_id) {
|
|
||||||
(player, false)
|
|
||||||
} else {
|
|
||||||
tracing::warn!("logic_thread: get player requested, but player {player_id} with data doesn't exist");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut player = player.borrow_mut();
|
let mut player = player.borrow_mut();
|
||||||
if is_player {
|
|
||||||
player
|
|
||||||
.world
|
|
||||||
.borrow_mut()
|
|
||||||
.world_entitys
|
|
||||||
.insert(player.basic_info.cur_map_id, WorldEntity::default());
|
|
||||||
state.worlds.insert(player_id, player.world.clone());
|
state.worlds.insert(player_id, player.world.clone());
|
||||||
}
|
|
||||||
|
|
||||||
player.init();
|
player.init();
|
||||||
player.set_session(session);
|
player.set_session(session);
|
||||||
|
@ -164,30 +159,33 @@ fn handle_logic_input(state: &mut LogicState, input: LogicInput) {
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
.set_in_world_player_data(player.build_in_world_player());
|
.set_in_world_player_data(player.build_in_world_player());
|
||||||
|
|
||||||
world_util::add_player_entities(&player);
|
world_util::add_player_entities(&mut player.world.borrow_mut(), &player);
|
||||||
let scene_info = world_util::build_scene_information(&player);
|
let scene_info = world_util::build_scene_information(
|
||||||
|
&player.world.borrow(),
|
||||||
player.notify(SilenceNpcNotify::default());
|
player.location.instance_id,
|
||||||
|
player.basic_info.id,
|
||||||
|
);
|
||||||
|
|
||||||
player.notify(JoinSceneNotify {
|
player.notify(JoinSceneNotify {
|
||||||
scene_info: Some(scene_info),
|
|
||||||
max_entity_id: i64::MAX,
|
max_entity_id: i64::MAX,
|
||||||
|
scene_info: Some(scene_info),
|
||||||
transition_option: Some(TransitionOptionPb::default()),
|
transition_option: Some(TransitionOptionPb::default()),
|
||||||
});
|
});
|
||||||
|
|
||||||
player.notify(AfterJoinSceneNotify::default());
|
player.notify(AfterJoinSceneNotify::default());
|
||||||
player.notify(player.build_update_formation_notify());
|
|
||||||
|
|
||||||
let map = logic::utils::quadrant_util::get_map(player.basic_info.cur_map_id);
|
// TODO: maybe move somewhere else?
|
||||||
let quadrant_id = map.get_quadrant_id(
|
player.notify(JsPatchNotify {
|
||||||
player.location.position.position.x * 100.0,
|
content: WATER_MASK.to_string(),
|
||||||
player.location.position.position.y * 100.0,
|
});
|
||||||
);
|
player.notify(JsPatchNotify {
|
||||||
player.quadrant_id = quadrant_id;
|
content: UID_FIX
|
||||||
player.notify_month_card();
|
.replace("{PLAYER_USERNAME}", &player.basic_info.name)
|
||||||
|
.replace("{SELECTED_COLOR}", "50FC71"),
|
||||||
let entities = map.get_initial_entities(quadrant_id);
|
});
|
||||||
world_util::add_entities(&player, &entities, false);
|
player.notify(JsPatchNotify {
|
||||||
|
content: CENSORSHIP_FIX.to_string(),
|
||||||
|
});
|
||||||
|
|
||||||
drop(player);
|
drop(player);
|
||||||
|
|
||||||
|
@ -213,7 +211,6 @@ fn handle_logic_input(state: &mut LogicState, input: LogicInput) {
|
||||||
|
|
||||||
let _ = state.worlds.remove(&player_id);
|
let _ = state.worlds.remove(&player_id);
|
||||||
// TODO: kick co-op players from removed world
|
// TODO: kick co-op players from removed world
|
||||||
// TODO: Remove all entities
|
|
||||||
|
|
||||||
player_save_task::push(
|
player_save_task::push(
|
||||||
player_id,
|
player_id,
|
25
game-server/src/logic/utils/entity_serializer.rs
Normal file
25
game-server/src/logic/utils/entity_serializer.rs
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
use crate::logic::ecs::component::ComponentContainer;
|
||||||
|
use shorekeeper_protocol::{EntityPb, PlayerSceneAoiData};
|
||||||
|
|
||||||
|
use crate::{logic::ecs::world::World, query_with};
|
||||||
|
|
||||||
|
pub fn build_scene_add_on_init_data(world: &World) -> PlayerSceneAoiData {
|
||||||
|
let entities = query_with!(world, PlayerEntityMarker)
|
||||||
|
.into_iter()
|
||||||
|
.map(|(e, _)| e)
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
let mut aoi_data = PlayerSceneAoiData::default();
|
||||||
|
for entity in entities {
|
||||||
|
let mut pb = EntityPb { id: entity.into(), ..Default::default() };
|
||||||
|
|
||||||
|
world
|
||||||
|
.get_entity_components(entity)
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|comp| comp.set_pb_data(&mut pb));
|
||||||
|
|
||||||
|
aoi_data.entities.push(pb);
|
||||||
|
}
|
||||||
|
|
||||||
|
aoi_data
|
||||||
|
}
|
2
game-server/src/logic/utils/mod.rs
Normal file
2
game-server/src/logic/utils/mod.rs
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
pub mod entity_serializer;
|
||||||
|
pub mod world_util;
|
133
game-server/src/logic/utils/world_util.rs
Normal file
133
game-server/src/logic/utils/world_util.rs
Normal file
|
@ -0,0 +1,133 @@
|
||||||
|
use shorekeeper_data::base_property_data;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
EntityConfigType, FightRoleInfo, FightRoleInfos, LivingStatus, SceneInformation, SceneMode,
|
||||||
|
ScenePlayerInformation, SceneTimeInfo,
|
||||||
|
};
|
||||||
|
|
||||||
|
use super::entity_serializer;
|
||||||
|
use crate::{
|
||||||
|
logic::{
|
||||||
|
components::{
|
||||||
|
Attribute, EntityConfig, Equip, Movement, OwnerPlayer, PlayerEntityMarker, Position,
|
||||||
|
Visibility, VisionSkill,
|
||||||
|
},
|
||||||
|
ecs::{component::ComponentContainer, world::World},
|
||||||
|
player::Player,
|
||||||
|
},
|
||||||
|
query_with,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn add_player_entities(world: &mut World, player: &Player) {
|
||||||
|
let cur_role_id = player.get_cur_role_id();
|
||||||
|
|
||||||
|
for role in player.get_current_formation_role_list() {
|
||||||
|
let id = world
|
||||||
|
.create_entity()
|
||||||
|
.with(ComponentContainer::PlayerEntityMarker(PlayerEntityMarker))
|
||||||
|
.with(ComponentContainer::EntityConfig(EntityConfig {
|
||||||
|
config_id: role.role_id,
|
||||||
|
config_type: EntityConfigType::Character,
|
||||||
|
}))
|
||||||
|
.with(ComponentContainer::OwnerPlayer(OwnerPlayer(
|
||||||
|
player.basic_info.id,
|
||||||
|
)))
|
||||||
|
.with(ComponentContainer::Position(Position(
|
||||||
|
player.location.position.clone(),
|
||||||
|
)))
|
||||||
|
.with(ComponentContainer::Visibility(Visibility(
|
||||||
|
role.role_id == cur_role_id,
|
||||||
|
)))
|
||||||
|
.with(ComponentContainer::Attribute(Attribute::from_data(
|
||||||
|
base_property_data::iter()
|
||||||
|
.find(|d| d.id == role.role_id)
|
||||||
|
.unwrap(),
|
||||||
|
)))
|
||||||
|
.with(ComponentContainer::Movement(Movement::default()))
|
||||||
|
.with(ComponentContainer::Equip(Equip {
|
||||||
|
weapon_id: role.equip_weapon,
|
||||||
|
weapon_breach_level: 0, // TODO: store this too
|
||||||
|
}))
|
||||||
|
.with(ComponentContainer::VisionSkill(VisionSkill {
|
||||||
|
skill_id: player.explore_tools.active_explore_skill,
|
||||||
|
}))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
tracing::debug!(
|
||||||
|
"created player entity, id: {}, role_id: {}",
|
||||||
|
i64::from(id),
|
||||||
|
role.role_id
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_scene_information(world: &World, instance_id: i32, owner_id: i32) -> SceneInformation {
|
||||||
|
SceneInformation {
|
||||||
|
scene_id: String::new(),
|
||||||
|
instance_id,
|
||||||
|
owner_id,
|
||||||
|
dynamic_entity_list: Vec::new(),
|
||||||
|
blackboard_params: Vec::new(),
|
||||||
|
end_time: 0,
|
||||||
|
aoi_data: Some(entity_serializer::build_scene_add_on_init_data(world)),
|
||||||
|
player_infos: build_player_info_list(world),
|
||||||
|
mode: SceneMode::Single.into(),
|
||||||
|
time_info: Some(SceneTimeInfo {
|
||||||
|
owner_time_clock_time_span: 0,
|
||||||
|
hour: 8,
|
||||||
|
minute: 0,
|
||||||
|
}),
|
||||||
|
cur_context_id: owner_id as i64,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn build_player_info_list(world: &World) -> Vec<ScenePlayerInformation> {
|
||||||
|
world
|
||||||
|
.players()
|
||||||
|
.map(|sp| {
|
||||||
|
let (cur_role_id, transform) = query_with!(
|
||||||
|
world,
|
||||||
|
PlayerEntityMarker,
|
||||||
|
OwnerPlayer,
|
||||||
|
Visibility,
|
||||||
|
EntityConfig,
|
||||||
|
Position
|
||||||
|
)
|
||||||
|
.into_iter()
|
||||||
|
.find_map(|(_, _, owner, visibility, conf, pos)| {
|
||||||
|
(sp.player_id == owner.0 && visibility.0).then_some((conf.config_id, pos.0.clone()))
|
||||||
|
})
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let active_characters =
|
||||||
|
query_with!(world, PlayerEntityMarker, OwnerPlayer, EntityConfig)
|
||||||
|
.into_iter()
|
||||||
|
.filter(|(_, _, owner, _)| owner.0 == sp.player_id);
|
||||||
|
|
||||||
|
ScenePlayerInformation {
|
||||||
|
cur_role: cur_role_id,
|
||||||
|
group_type: sp.group_type,
|
||||||
|
player_id: sp.player_id,
|
||||||
|
player_icon: sp.player_icon,
|
||||||
|
player_name: sp.player_name.clone(),
|
||||||
|
level: sp.level,
|
||||||
|
location: Some(transform.get_position_protobuf()),
|
||||||
|
rotation: Some(transform.get_rotation_protobuf()),
|
||||||
|
fight_role_infos: Vec::from([FightRoleInfos {
|
||||||
|
group_type: sp.group_type,
|
||||||
|
living_status: LivingStatus::Alive.into(),
|
||||||
|
cur_role: cur_role_id,
|
||||||
|
is_retain: true,
|
||||||
|
fight_role_infos: active_characters
|
||||||
|
.map(|(id, _, _, conf)| FightRoleInfo {
|
||||||
|
entity_id: id.into(),
|
||||||
|
role_id: conf.config_id,
|
||||||
|
})
|
||||||
|
.collect(),
|
||||||
|
..Default::default()
|
||||||
|
}]),
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
}
|
35
game-server/src/main.rs
Normal file
35
game-server/src/main.rs
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
use std::sync::{Arc, LazyLock};
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
use common::config_util;
|
||||||
|
use config::ServiceConfig;
|
||||||
|
use session::SessionManager;
|
||||||
|
|
||||||
|
mod config;
|
||||||
|
mod gateway_connection;
|
||||||
|
mod logic;
|
||||||
|
mod player_save_task;
|
||||||
|
mod service_message_handler;
|
||||||
|
mod session;
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() -> Result<()> {
|
||||||
|
static CONFIG: LazyLock<ServiceConfig> =
|
||||||
|
LazyLock::new(|| config_util::load_or_create("gameserver.toml"));
|
||||||
|
static SESSION_MGR: LazyLock<SessionManager> = LazyLock::new(SessionManager::default);
|
||||||
|
|
||||||
|
::common::splash::print_splash();
|
||||||
|
::common::logging::init(::tracing::Level::DEBUG);
|
||||||
|
shorekeeper_data::load_json_data("assets/logic/json")?;
|
||||||
|
|
||||||
|
let database = Arc::new(shorekeeper_database::connect_to(&CONFIG.database).await?);
|
||||||
|
shorekeeper_database::run_migrations(database.as_ref()).await?;
|
||||||
|
|
||||||
|
logic::thread_mgr::start_logic_threads(1);
|
||||||
|
|
||||||
|
player_save_task::start(database.clone());
|
||||||
|
gateway_connection::init(CONFIG.service_id, &CONFIG.gateway_end_point);
|
||||||
|
service_message_handler::run(&CONFIG.service_end_point, &SESSION_MGR, database).await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
|
@ -1,9 +1,8 @@
|
||||||
use std::sync::{Arc, OnceLock};
|
use std::sync::{Arc, OnceLock};
|
||||||
use tokio::sync::mpsc;
|
use tokio::sync::mpsc;
|
||||||
|
|
||||||
use wicked_waifus_database::{query, PgPool};
|
use shorekeeper_database::{query, PgPool};
|
||||||
use wicked_waifus_protocol::Protobuf;
|
use shorekeeper_protocol::{PlayerSaveData, Protobuf};
|
||||||
use wicked_waifus_protocol_internal::PlayerSaveData;
|
|
||||||
|
|
||||||
static SENDER: OnceLock<mpsc::Sender<PlayerSaveQuery>> = OnceLock::new();
|
static SENDER: OnceLock<mpsc::Sender<PlayerSaveQuery>> = OnceLock::new();
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
use std::sync::{Arc, OnceLock};
|
use std::sync::{Arc, OnceLock};
|
||||||
|
|
||||||
use wicked_waifus_protocol_internal::{
|
use common::time_util;
|
||||||
CreatePlayerDataRequest, CreatePlayerDataResponse, ForwardClientMessagePush, MessageID,
|
use shorekeeper_database::{models, query, query_as, PgPool};
|
||||||
PlayerSaveData, StartPlayerSessionRequest, StartPlayerSessionResponse, StopPlayerSessionPush,
|
use shorekeeper_network::{config::ServiceEndPoint, ServiceListener, ServiceMessage};
|
||||||
|
use shorekeeper_protocol::{
|
||||||
|
message::Message, CreatePlayerDataRequest, CreatePlayerDataResponse, ErrorCode,
|
||||||
|
ForwardClientMessagePush, MessageID, PlayerSaveData, Protobuf, ProtocolUnit,
|
||||||
|
StartPlayerSessionRequest, StartPlayerSessionResponse, StopPlayerSessionPush,
|
||||||
};
|
};
|
||||||
use wicked_waifus_commons::time_util;
|
|
||||||
use wicked_waifus_database::{models, PgPool, query, query_as};
|
|
||||||
use wicked_waifus_network::{config::ServiceEndPoint, ServiceListener, ServiceMessage};
|
|
||||||
use wicked_waifus_protocol::{ErrorCode, message::Message, Protobuf, ProtocolUnit};
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
gateway_connection,
|
gateway_connection,
|
|
@ -1,8 +1,7 @@
|
||||||
mod manager;
|
mod manager;
|
||||||
pub use manager::SessionManager;
|
pub use manager::SessionManager;
|
||||||
use wicked_waifus_network::ServiceMessage;
|
use shorekeeper_network::ServiceMessage;
|
||||||
use wicked_waifus_protocol_internal::{ForwardClientMessagePush, MessageID};
|
use shorekeeper_protocol::{message::Message, ForwardClientMessagePush, MessageID, Protobuf};
|
||||||
use wicked_waifus_protocol::{message::Message, Protobuf};
|
|
||||||
|
|
||||||
use crate::{gateway_connection, logic::thread_mgr::LogicThreadHandle};
|
use crate::{gateway_connection, logic::thread_mgr::LogicThreadHandle};
|
||||||
|
|
38
game-server/watermask.js
Normal file
38
game-server/watermask.js
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
const UE = require("ue"),
|
||||||
|
Info_1 = require("../../../Core/Common/Info"),
|
||||||
|
MathUtils_1 = require("../../../Core/Utils/MathUtils"),
|
||||||
|
EventDefine_1 = require("../../Common/Event/EventDefine"),
|
||||||
|
EventSystem_1 = require("../../Common/Event/EventSystem"),
|
||||||
|
UiControllerBase_1 = require("../../Ui/Base/UiControllerBase"),
|
||||||
|
UiLayerType_1 = require("../../Ui/Define/UiLayerType"),
|
||||||
|
UiLayer_1 = require("../../Ui/UiLayer");
|
||||||
|
|
||||||
|
var _a = require('../Module/WaterMask/WaterMaskController').WaterMaskView;
|
||||||
|
_a.vOo = function () {
|
||||||
|
void 0 !== _a.SOo && _a.EOo();
|
||||||
|
var e = UiLayer_1.UiLayer.GetLayerRootUiItem(UiLayerType_1.ELayerType.WaterMask),
|
||||||
|
t = (_a.SOo = UE.KuroActorManager.SpawnActor(Info_1.Info.World, UE.UIContainerActor.StaticClass(),
|
||||||
|
MathUtils_1.MathUtils.DefaultTransform, void 0), _a.SOo.RootComponent),
|
||||||
|
e = (t.SetDisplayName("WaterMaskContainer"), UE.KuroStaticLibrary.SetActorPermanent(_a.SOo, !0, !0), _a.SOo
|
||||||
|
.K2_AttachRootComponentTo(e), t.GetRootCanvas().GetOwner().RootComponent),
|
||||||
|
i = e.widget.width % _a.yOo / 2,
|
||||||
|
r = e.widget.height % _a.IOo / 2,
|
||||||
|
n = e.widget.width / 2,
|
||||||
|
_ = e.widget.height / 2,
|
||||||
|
s = Math.ceil(e.widget.width / _a.yOo),
|
||||||
|
o = Math.ceil(e.widget.height / _a.IOo),
|
||||||
|
v = "discord.gg/reversedrooms";
|
||||||
|
for (let a = 0; a < s; a++)
|
||||||
|
for (let e = 0; e < o; e++) {
|
||||||
|
var E = UE.KuroActorManager.SpawnActor(Info_1.Info.World, UE.UITextActor.StaticClass(), MathUtils_1
|
||||||
|
.MathUtils.DefaultTransform, void 0),
|
||||||
|
U = E.RootComponent,
|
||||||
|
U = (E.K2_AttachRootComponentTo(t), U.SetDisplayName("WaterMaskText"), E.GetComponentByClass(UE
|
||||||
|
.UIText.StaticClass()));
|
||||||
|
U.SetFontSize(_a.vFt), U.SetOverflowType(0), U.SetAlpha(_a.LOo), U.SetFont(UE.LGUIFontData
|
||||||
|
.GetDefaultFont()), U.SetText(v), U.SetUIRelativeLocation(new UE.Vector(a * _a.yOo - n + i, e *
|
||||||
|
_a.IOo - _ + r, 0)), U.SetUIRelativeRotation(new UE.Rotator(0, _a.TOo, 0)), UE.KuroStaticLibrary
|
||||||
|
.SetActorPermanent(E, !0, !0)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
_a.vOo();
|
|
@ -1,11 +1,8 @@
|
||||||
[package]
|
[package]
|
||||||
name = "wicked-waifus-gateway-server"
|
name = "gateway-server"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
version.workspace = true
|
version.workspace = true
|
||||||
|
|
||||||
[features]
|
|
||||||
debug-msg = []
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
# Framework
|
# Framework
|
||||||
tokio.workspace = true
|
tokio.workspace = true
|
||||||
|
@ -27,9 +24,8 @@ dashmap.workspace = true
|
||||||
hex.workspace = true
|
hex.workspace = true
|
||||||
|
|
||||||
# Internal
|
# Internal
|
||||||
wicked-waifus-commons.workspace = true
|
common.workspace = true
|
||||||
wicked-waifus-database.workspace = true
|
shorekeeper-database.workspace = true
|
||||||
wicked-waifus-network.workspace = true
|
shorekeeper-network.workspace = true
|
||||||
wicked-waifus-protocol-internal.workspace = true
|
shorekeeper-protocol.workspace = true
|
||||||
wicked-waifus-protocol.workspace = true
|
shorekeeper-protokey.workspace = true
|
||||||
wicked-waifus-protokey.workspace = true
|
|
|
@ -1,8 +1,8 @@
|
||||||
use wicked_waifus_commons::config_util::TomlConfig;
|
use common::config_util::TomlConfig;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use wicked_waifus_database::DatabaseSettings;
|
use shorekeeper_database::DatabaseSettings;
|
||||||
use wicked_waifus_network::config::ServiceEndPoint;
|
use shorekeeper_network::config::ServiceEndPoint;
|
||||||
use wicked_waifus_protokey::ProtoKeySettings;
|
use shorekeeper_protokey::ProtoKeySettings;
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
#[derive(Deserialize)]
|
||||||
pub struct ServerConfig {
|
pub struct ServerConfig {
|
|
@ -1,12 +1,10 @@
|
||||||
use std::sync::OnceLock;
|
use std::sync::OnceLock;
|
||||||
|
|
||||||
use tokio::sync::mpsc;
|
use shorekeeper_network::ServiceMessage;
|
||||||
|
use shorekeeper_protocol::{
|
||||||
use wicked_waifus_protocol_internal::{ForwardClientMessagePush, MessageID};
|
message::Message, proto_config, ForwardClientMessagePush, MessageID, MessageRoute, Protobuf,
|
||||||
use wicked_waifus_network::ServiceMessage;
|
|
||||||
use wicked_waifus_protocol::{
|
|
||||||
message::Message, MessageRoute, proto_config, Protobuf,
|
|
||||||
};
|
};
|
||||||
|
use tokio::sync::mpsc;
|
||||||
|
|
||||||
use crate::session::{Session, SessionManager};
|
use crate::session::{Session, SessionManager};
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue