Compare commits
8 Commits
master
...
5ba61cf461
Author | SHA1 | Date | |
---|---|---|---|
5ba61cf461 | |||
1d9787402c | |||
bca78c0611 | |||
647bba57da | |||
1e5bbb51ff | |||
ba0a6cf327 | |||
226577f91a | |||
23a3d1212e |
14
.drone.yml
14
.drone.yml
@ -6,19 +6,7 @@ name: test elseware
|
|||||||
concurrency:
|
concurrency:
|
||||||
limit: 1
|
limit: 1
|
||||||
|
|
||||||
environment:
|
|
||||||
CARGO_INCREMENTAL: false
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: clean cache
|
|
||||||
image: rustlang/rust:nightly
|
|
||||||
volumes:
|
|
||||||
- name: cache
|
|
||||||
path: /usr/local/cargo
|
|
||||||
- name: target-cache
|
|
||||||
path: /drone/src/target
|
|
||||||
commands:
|
|
||||||
- cargo prune
|
|
||||||
- name: build
|
- name: build
|
||||||
image: rustlang/rust:nightly
|
image: rustlang/rust:nightly
|
||||||
volumes:
|
volumes:
|
||||||
@ -45,7 +33,7 @@ steps:
|
|||||||
- name: target-cache
|
- name: target-cache
|
||||||
path: /drone/src/target
|
path: /drone/src/target
|
||||||
commands:
|
commands:
|
||||||
- cargo test --jobs 1
|
- cargo test
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
- name: cache
|
- name: cache
|
||||||
|
68
Cargo.toml
68
Cargo.toml
@ -4,46 +4,8 @@ version = "0.1.0"
|
|||||||
authors = ["Jake Probst <jake.probst@gmail.com>"]
|
authors = ["Jake Probst <jake.probst@gmail.com>"]
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[workspace]
|
[dependencies]
|
||||||
members = [
|
libpso = { git = "http://git.sharnoth.com/jake/libpso" }
|
||||||
"src/client",
|
|
||||||
"src/drops",
|
|
||||||
"src/entity",
|
|
||||||
"src/items",
|
|
||||||
"src/location",
|
|
||||||
"src/maps",
|
|
||||||
"src/networking",
|
|
||||||
"src/pktbuilder",
|
|
||||||
"src/quests",
|
|
||||||
"src/room",
|
|
||||||
"src/shops",
|
|
||||||
"src/stats",
|
|
||||||
"src/trade",
|
|
||||||
"src/patch_server",
|
|
||||||
"src/login_server",
|
|
||||||
"src/ship_server",
|
|
||||||
]
|
|
||||||
|
|
||||||
[workspace.dependencies]
|
|
||||||
entity = { path = "./src/entity" }
|
|
||||||
maps = { path = "./src/maps" }
|
|
||||||
networking = { path = "./src/networking" }
|
|
||||||
shops = { path = "./src/shops" }
|
|
||||||
stats = { path = "./src/stats" }
|
|
||||||
items = { path = "./src/items" }
|
|
||||||
pktbuilder = { path = "./src/pktbuilder" }
|
|
||||||
quests = { path = "./src/quests" }
|
|
||||||
location = { path = "./src/location" }
|
|
||||||
client = { path = "./src/client" }
|
|
||||||
drops = { path = "./src/drops" }
|
|
||||||
trade = { path = "./src/trade" }
|
|
||||||
room = { path = "./src/room" }
|
|
||||||
patch_server = { path = "./src/patch_server" }
|
|
||||||
login_server = { path = "./src/login_server" }
|
|
||||||
ship_server = { path = "./src/ship_server" }
|
|
||||||
|
|
||||||
libpso = { git = "http://git.sharnoth.com/jake/libpso", rev="90246b6" }
|
|
||||||
|
|
||||||
async-std = { version = "1.9.0", features = ["unstable", "attributes"] }
|
async-std = { version = "1.9.0", features = ["unstable", "attributes"] }
|
||||||
futures = "0.3.5"
|
futures = "0.3.5"
|
||||||
rand = "0.7.3"
|
rand = "0.7.3"
|
||||||
@ -65,33 +27,9 @@ ages-prs = "0.1"
|
|||||||
async-trait = "0.1.51"
|
async-trait = "0.1.51"
|
||||||
async-recursion= "1.0.0"
|
async-recursion= "1.0.0"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
|
barrel = { version = "0.6.5", features = ["pg"] }
|
||||||
refinery = { version = "0.5.0", features = ["postgres"] }
|
refinery = { version = "0.5.0", features = ["postgres"] }
|
||||||
sqlx = { version = "0.6.2", features = ["runtime-async-std-native-tls", "postgres", "json", "chrono"] }
|
sqlx = { version = "0.6.2", features = ["runtime-async-std-native-tls", "postgres", "json", "chrono"] }
|
||||||
strum = "0.19.5"
|
strum = "0.19.5"
|
||||||
strum_macros = "0.19"
|
strum_macros = "0.19"
|
||||||
anyhow = { version = "1.0.68", features = ["backtrace"] }
|
anyhow = { version = "1.0.68", features = ["backtrace"] }
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
networking = { workspace = true }
|
|
||||||
patch_server = { workspace = true }
|
|
||||||
login_server = { workspace = true }
|
|
||||||
ship_server = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
bcrypt = { workspace = true }
|
|
||||||
chrono = { workspace = true }
|
|
||||||
fern = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
log = { workspace = true }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
drops = { workspace = true }
|
|
||||||
shops = { workspace = true }
|
|
||||||
items = { workspace = true }
|
|
||||||
quests = { workspace = true }
|
|
||||||
stats = { workspace = true }
|
|
||||||
async-trait = { workspace = true }
|
|
@ -1,8 +1,8 @@
|
|||||||
use log::{info};
|
use log::{info};
|
||||||
use entity::gateway::postgres::PostgresGateway;
|
use elseware::entity::gateway::postgres::PostgresGateway;
|
||||||
use login_server::login::LoginServerState;
|
use elseware::login::login::LoginServerState;
|
||||||
use login_server::character::CharacterServerState;
|
use elseware::login::character::CharacterServerState;
|
||||||
use networking::interserver::AuthToken;
|
use elseware::common::interserver::AuthToken;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let colors = fern::colors::ColoredLevelConfig::new()
|
let colors = fern::colors::ColoredLevelConfig::new()
|
||||||
@ -38,17 +38,17 @@ fn main() {
|
|||||||
|
|
||||||
let login_state = LoginServerState::new(entity_gateway.clone(), charserv_ip);
|
let login_state = LoginServerState::new(entity_gateway.clone(), charserv_ip);
|
||||||
let login_loop = async_std::task::spawn(async move {
|
let login_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(login_state, login_server::login::LOGIN_PORT).await;
|
elseware::common::mainloop::run_server(login_state, elseware::login::login::LOGIN_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
let char_state = CharacterServerState::new(entity_gateway, AuthToken(shipgate_token));
|
let char_state = CharacterServerState::new(entity_gateway, AuthToken(shipgate_token));
|
||||||
let sub_char_state = char_state.clone();
|
let sub_char_state = char_state.clone();
|
||||||
let character_loop = async_std::task::spawn(async move {
|
let character_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_char_state, login_server::character::CHARACTER_PORT).await;
|
elseware::common::mainloop::run_server(sub_char_state, elseware::login::character::CHARACTER_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
let inter_character_loop = async_std::task::spawn(async move {
|
let inter_character_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_listen(char_state, login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_listen(char_state, elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
info!("[auth/character] starting server");
|
info!("[auth/character] starting server");
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
use std::net::Ipv4Addr;
|
use std::net::Ipv4Addr;
|
||||||
use log::{info};
|
use log::{info};
|
||||||
|
|
||||||
use networking::interserver::AuthToken;
|
use elseware::common::interserver::AuthToken;
|
||||||
use login_server::login::LoginServerState;
|
use elseware::login::login::LoginServerState;
|
||||||
use login_server::character::CharacterServerState;
|
use elseware::login::character::CharacterServerState;
|
||||||
use patch_server::{PatchServerState, generate_patch_tree, load_config, load_motd};
|
use elseware::patch::patch::{PatchServerState, generate_patch_tree, load_config, load_motd};
|
||||||
use ship_server::ShipServerStateBuilder;
|
use elseware::ship::ship::{ShipServerStateBuilder, ShipEvent};
|
||||||
|
|
||||||
use maps::Holiday;
|
#[allow(unused_imports)]
|
||||||
use entity::gateway::{EntityGateway, InMemoryGateway};
|
use elseware::entity::gateway::{EntityGateway, InMemoryGateway, PostgresGateway};
|
||||||
use entity::account::{NewUserAccountEntity, NewUserSettingsEntity};
|
use elseware::entity::account::{NewUserAccountEntity, NewUserSettingsEntity};
|
||||||
use entity::character::NewCharacterEntity;
|
use elseware::entity::character::NewCharacterEntity;
|
||||||
use entity::item::{NewItemEntity, ItemDetail, InventoryItemEntity};
|
use elseware::entity::item::{NewItemEntity, ItemDetail, InventoryItemEntity};
|
||||||
use entity::item;
|
use elseware::entity::item;
|
||||||
|
|
||||||
fn setup_logger() {
|
fn setup_logger() {
|
||||||
let colors = fern::colors::ColoredLevelConfig::new()
|
let colors = fern::colors::ColoredLevelConfig::new()
|
||||||
@ -338,73 +338,73 @@ fn main() {
|
|||||||
let (patch_file_tree, patch_file_lookup) = generate_patch_tree(patch_config.path.as_str());
|
let (patch_file_tree, patch_file_lookup) = generate_patch_tree(patch_config.path.as_str());
|
||||||
let patch_state = PatchServerState::new(patch_file_tree, patch_file_lookup, patch_motd);
|
let patch_state = PatchServerState::new(patch_file_tree, patch_file_lookup, patch_motd);
|
||||||
let patch_loop = async_std::task::spawn(async move {
|
let patch_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(patch_state, patch_config.port).await;
|
elseware::common::mainloop::run_server(patch_state, patch_config.port).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
info!("[auth] starting server");
|
info!("[auth] starting server");
|
||||||
let login_state = LoginServerState::new(entity_gateway.clone(), "127.0.0.1".parse().unwrap());
|
let login_state = LoginServerState::new(entity_gateway.clone(), "127.0.0.1".parse().unwrap());
|
||||||
let login_loop = async_std::task::spawn(async move {
|
let login_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(login_state, login_server::login::LOGIN_PORT).await;
|
elseware::common::mainloop::run_server(login_state, elseware::login::login::LOGIN_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
info!("[character] starting server");
|
info!("[character] starting server");
|
||||||
let char_state = CharacterServerState::new(entity_gateway.clone(), AuthToken("".into()));
|
let char_state = CharacterServerState::new(entity_gateway.clone(), AuthToken("".into()));
|
||||||
let sub_char_state = char_state.clone();
|
let sub_char_state = char_state.clone();
|
||||||
let character_loop = async_std::task::spawn(async move {
|
let character_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_char_state, login_server::character::CHARACTER_PORT).await;
|
elseware::common::mainloop::run_server(sub_char_state, elseware::login::character::CHARACTER_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
let sub_char_state = char_state.clone();
|
let sub_char_state = char_state.clone();
|
||||||
let inter_character_loop = async_std::task::spawn(async move {
|
let inter_character_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_listen(sub_char_state, login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_listen(sub_char_state, elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
info!("[ship] starting servers");
|
info!("[ship] starting servers");
|
||||||
let ship_state = ShipServerStateBuilder::default()
|
let ship_state = ShipServerStateBuilder::default()
|
||||||
.name("US/Sona-Nyl".into())
|
.name("US/Sona-Nyl".into())
|
||||||
.ip(Ipv4Addr::new(127,0,0,1))
|
.ip(Ipv4Addr::new(127,0,0,1))
|
||||||
.port(ship_server::SHIP_PORT)
|
.port(elseware::ship::ship::SHIP_PORT)
|
||||||
.event(Holiday::Halloween)
|
.event(ShipEvent::Halloween)
|
||||||
.gateway(entity_gateway.clone())
|
.gateway(entity_gateway.clone())
|
||||||
.build();
|
.build();
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let ship_loop1 = async_std::task::spawn(async move {
|
let ship_loop1 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_ship_state, ship_server::SHIP_PORT).await;
|
elseware::common::mainloop::run_server(sub_ship_state, elseware::ship::ship::SHIP_PORT).await;
|
||||||
});
|
});
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let inter_ship_loop1 = async_std::task::spawn(async move {
|
let inter_ship_loop1 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
let ship_state = ShipServerStateBuilder::default()
|
let ship_state = ShipServerStateBuilder::default()
|
||||||
.name("EU/Dylath-Leen".into())
|
.name("EU/Dylath-Leen".into())
|
||||||
.ip(Ipv4Addr::new(127,0,0,1))
|
.ip(Ipv4Addr::new(127,0,0,1))
|
||||||
.port(ship_server::SHIP_PORT+2000)
|
.port(elseware::ship::ship::SHIP_PORT+2000)
|
||||||
.event(Holiday::Christmas)
|
.event(ShipEvent::Christmas)
|
||||||
.gateway(entity_gateway.clone())
|
.gateway(entity_gateway.clone())
|
||||||
.build();
|
.build();
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let ship_loop2 = async_std::task::spawn(async move {
|
let ship_loop2 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_ship_state, ship_server::SHIP_PORT+2000).await;
|
elseware::common::mainloop::run_server(sub_ship_state, elseware::ship::ship::SHIP_PORT+2000).await;
|
||||||
});
|
});
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let inter_ship_loop2 = async_std::task::spawn(async move {
|
let inter_ship_loop2 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
let ship_state = ShipServerStateBuilder::default()
|
let ship_state = ShipServerStateBuilder::default()
|
||||||
.name("JP/Thalarion".into())
|
.name("JP/Thalarion".into())
|
||||||
.ip(Ipv4Addr::new(127,0,0,1))
|
.ip(Ipv4Addr::new(127,0,0,1))
|
||||||
.port(ship_server::SHIP_PORT+3000)
|
.port(elseware::ship::ship::SHIP_PORT+3000)
|
||||||
.gateway(entity_gateway.clone())
|
.gateway(entity_gateway.clone())
|
||||||
.build();
|
.build();
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let ship_loop3 = async_std::task::spawn(async move {
|
let ship_loop3 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_ship_state, ship_server::SHIP_PORT+3000).await;
|
elseware::common::mainloop::run_server(sub_ship_state, elseware::ship::ship::SHIP_PORT+3000).await;
|
||||||
});
|
});
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let inter_ship_loop3 = async_std::task::spawn(async move {
|
let inter_ship_loop3 = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_connect(sub_ship_state, std::net::Ipv4Addr::new(127, 0, 0, 1), elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
futures::future::join_all(vec![patch_loop, login_loop, character_loop, inter_character_loop,
|
futures::future::join_all(vec![patch_loop, login_loop, character_loop, inter_character_loop,
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
use patch_server::{PatchServerState, generate_patch_tree, load_config_env, load_motd};
|
use elseware::patch::patch::{PatchServerState, generate_patch_tree, load_config_env, load_motd};
|
||||||
use log::info;
|
use log::{info};
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
info!("[patch] starting server");
|
info!("[patch] starting server");
|
||||||
@ -9,7 +9,7 @@ fn main() {
|
|||||||
let patch_state = PatchServerState::new(patch_file_tree, patch_file_lookup, patch_motd);
|
let patch_state = PatchServerState::new(patch_file_tree, patch_file_lookup, patch_motd);
|
||||||
|
|
||||||
let patch_loop = async_std::task::spawn(async move {
|
let patch_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(patch_state, patch_config.port).await;
|
elseware::common::mainloop::run_server(patch_state, patch_config.port).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
async_std::task::block_on(patch_loop);
|
async_std::task::block_on(patch_loop);
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
use log::info;
|
use log::{info};
|
||||||
use entity::gateway::postgres::PostgresGateway;
|
use elseware::entity::gateway::postgres::PostgresGateway;
|
||||||
use ship_server::ShipServerStateBuilder;
|
use elseware::ship::ship::ShipServerStateBuilder;
|
||||||
use networking::interserver::AuthToken;
|
use elseware::common::interserver::AuthToken;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let colors = fern::colors::ColoredLevelConfig::new()
|
let colors = fern::colors::ColoredLevelConfig::new()
|
||||||
@ -40,7 +40,7 @@ fn main() {
|
|||||||
let ship_state = ShipServerStateBuilder::default()
|
let ship_state = ShipServerStateBuilder::default()
|
||||||
.name(ship_name)
|
.name(ship_name)
|
||||||
.ip(ip)
|
.ip(ip)
|
||||||
.port(ship_server::SHIP_PORT)
|
.port(elseware::ship::ship::SHIP_PORT)
|
||||||
.gateway(entity_gateway)
|
.gateway(entity_gateway)
|
||||||
.auth_token(AuthToken(shipgate_token))
|
.auth_token(AuthToken(shipgate_token))
|
||||||
.build();
|
.build();
|
||||||
@ -49,10 +49,10 @@ fn main() {
|
|||||||
|
|
||||||
let sub_ship_state = ship_state.clone();
|
let sub_ship_state = ship_state.clone();
|
||||||
let ship_loop = async_std::task::spawn(async move {
|
let ship_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_server(sub_ship_state, ship_server::SHIP_PORT).await;
|
elseware::common::mainloop::run_server(sub_ship_state, elseware::ship::ship::SHIP_PORT).await;
|
||||||
});
|
});
|
||||||
let inter_ship_loop = async_std::task::spawn(async move {
|
let inter_ship_loop = async_std::task::spawn(async move {
|
||||||
networking::mainloop::run_interserver_connect(ship_state, shipgate_ip, login_server::login::COMMUNICATION_PORT).await;
|
elseware::common::mainloop::run_interserver_connect(ship_state, shipgate_ip, elseware::login::login::COMMUNICATION_PORT).await;
|
||||||
});
|
});
|
||||||
|
|
||||||
info!("[auth/character] starting server");
|
info!("[auth/character] starting server");
|
||||||
|
@ -1,20 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "client"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
networking = { workspace = true }
|
|
||||||
shops = { workspace = true }
|
|
||||||
items = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
chrono = { workspace = true }
|
|
@ -2,8 +2,8 @@ use std::net::Ipv4Addr;
|
|||||||
use async_std::channel;
|
use async_std::channel;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use serde::de::DeserializeOwned;
|
use serde::de::DeserializeOwned;
|
||||||
use entity::account::UserAccountId;
|
use crate::entity::account::UserAccountId;
|
||||||
use entity::character::CharacterEntityId;
|
use crate::entity::character::CharacterEntityId;
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, Serialize, Deserialize, Hash, PartialEq, Eq, PartialOrd, Ord)]
|
#[derive(Debug, Copy, Clone, Serialize, Deserialize, Hash, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
pub struct ServerId(pub usize);
|
pub struct ServerId(pub usize);
|
@ -1,6 +1,6 @@
|
|||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
use entity::character::CharacterClass;
|
use crate::entity::character::CharacterClass;
|
||||||
use std::sync::LazyLock;
|
use std::sync::LazyLock;
|
||||||
|
|
||||||
pub static LEVEL_TABLE: LazyLock<CharacterLevelTable> = LazyLock::new(CharacterLevelTable::default);
|
pub static LEVEL_TABLE: LazyLock<CharacterLevelTable> = LazyLock::new(CharacterLevelTable::default);
|
@ -9,8 +9,8 @@ use log::{trace, info, warn, error};
|
|||||||
|
|
||||||
use libpso::crypto::{PSOCipher, NullCipher, CipherError};
|
use libpso::crypto::{PSOCipher, NullCipher, CipherError};
|
||||||
use libpso::PacketParseError;
|
use libpso::PacketParseError;
|
||||||
use crate::serverstate::ClientId;
|
use crate::common::serverstate::ClientId;
|
||||||
use crate::serverstate::{RecvServerPacket, SendServerPacket, ServerState, OnConnect};
|
use crate::common::serverstate::{RecvServerPacket, SendServerPacket, ServerState, OnConnect};
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
@ -72,7 +72,7 @@ impl<C: PSOCipher> PacketReceiver<C> {
|
|||||||
let mut dec_buf = {
|
let mut dec_buf = {
|
||||||
//let mut cipher = self.cipher.lock().await;
|
//let mut cipher = self.cipher.lock().await;
|
||||||
let block_chunk_len = self.recv_buffer.len() / self.cipher.block_size() * self.cipher.block_size();
|
let block_chunk_len = self.recv_buffer.len() / self.cipher.block_size() * self.cipher.block_size();
|
||||||
let buf = self.recv_buffer.drain(..block_chunk_len).collect::<Vec<_>>();
|
let buf = self.recv_buffer.drain(..block_chunk_len).collect();
|
||||||
self.cipher.decrypt(&buf)?
|
self.cipher.decrypt(&buf)?
|
||||||
};
|
};
|
||||||
self.incoming_data.append(&mut dec_buf);
|
self.incoming_data.append(&mut dec_buf);
|
||||||
@ -255,7 +255,7 @@ where
|
|||||||
let (mut socket, addr) = listener.accept().await.unwrap();
|
let (mut socket, addr) = listener.accept().await.unwrap();
|
||||||
id += 1;
|
id += 1;
|
||||||
|
|
||||||
let client_id = crate::serverstate::ClientId(id);
|
let client_id = crate::common::serverstate::ClientId(id);
|
||||||
info!("new client {:?} {:?} {:?}", client_id, socket, addr);
|
info!("new client {:?} {:?} {:?}", client_id, socket, addr);
|
||||||
|
|
||||||
let (client_tx, client_rx) = async_std::channel::unbounded();
|
let (client_tx, client_rx) = async_std::channel::unbounded();
|
@ -8,11 +8,13 @@ use std::collections::HashMap;
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use serde::de::DeserializeOwned;
|
use serde::de::DeserializeOwned;
|
||||||
|
|
||||||
use crate::interserver::{ServerId, InterserverActor};
|
use crate::common::interserver::{ServerId, InterserverActor};
|
||||||
|
|
||||||
use libpso::crypto::{PSOCipher, NullCipher, CipherError};
|
use libpso::crypto::{PSOCipher, NullCipher, CipherError};
|
||||||
use crate::serverstate::{ServerState, SendServerPacket, RecvServerPacket};
|
use crate::common::serverstate::{ServerState, SendServerPacket, RecvServerPacket};
|
||||||
use entity::gateway::entitygateway::EntityGateway;
|
use crate::login::character::CharacterServerState;
|
||||||
|
//use crate::ship::ship::ShipServerState;
|
||||||
|
use crate::entity::gateway::entitygateway::EntityGateway;
|
||||||
|
|
||||||
use async_std::channel;
|
use async_std::channel;
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
@ -147,7 +149,7 @@ where
|
|||||||
info!("[interserver listen] new server: {:?} {:?}", socket, addr);
|
info!("[interserver listen] new server: {:?} {:?}", socket, addr);
|
||||||
|
|
||||||
id += 1;
|
id += 1;
|
||||||
let server_id = crate::interserver::ServerId(id);
|
let server_id = crate::common::interserver::ServerId(id);
|
||||||
let (client_tx, client_rx) = async_std::channel::unbounded();
|
let (client_tx, client_rx) = async_std::channel::unbounded();
|
||||||
state.set_sender(server_id, client_tx.clone()).await;
|
state.set_sender(server_id, client_tx.clone()).await;
|
||||||
|
|
||||||
@ -194,7 +196,7 @@ where
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
id += 1;
|
id += 1;
|
||||||
let server_id = crate::interserver::ServerId(id);
|
let server_id = crate::common::interserver::ServerId(id);
|
||||||
info!("[interserver connect] found loginserv: {:?} {:?}", server_id, socket);
|
info!("[interserver connect] found loginserv: {:?} {:?}", server_id, socket);
|
||||||
|
|
||||||
let (client_tx, client_rx) = async_std::channel::unbounded();
|
let (client_tx, client_rx) = async_std::channel::unbounded();
|
19
src/common/mod.rs
Normal file
19
src/common/mod.rs
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
pub mod cipherkeys;
|
||||||
|
pub mod serverstate;
|
||||||
|
pub mod mainloop;
|
||||||
|
pub mod leveltable;
|
||||||
|
pub mod interserver;
|
||||||
|
|
||||||
|
// https://www.reddit.com/r/rust/comments/33xhhu/how_to_create_an_array_of_structs_that_havent/
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! init_array(
|
||||||
|
($ty:ty, $len:expr, $val:expr) => (
|
||||||
|
{
|
||||||
|
let mut array: [$ty; $len] = unsafe { std::mem::uninitialized() };
|
||||||
|
for i in array.iter_mut() {
|
||||||
|
unsafe { ::std::ptr::write(i, $val); }
|
||||||
|
}
|
||||||
|
array
|
||||||
|
}
|
||||||
|
)
|
||||||
|
);
|
@ -1,17 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "drops"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
stats = { workspace = true }
|
|
||||||
|
|
||||||
rand = { workspace = true }
|
|
||||||
rand_chacha = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
enum-utils = { workspace = true }
|
|
||||||
toml = { workspace = true }
|
|
||||||
chrono = { workspace = true }
|
|
@ -1,23 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "entity"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
libpso = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
chrono = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
async-std = { workspace = true }
|
|
||||||
sqlx = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
async-trait = { workspace = true }
|
|
||||||
enum-utils = { workspace = true }
|
|
||||||
derive_more = { workspace = true }
|
|
||||||
refinery = { workspace = true }
|
|
||||||
lazy_static = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
strum = { workspace = true }
|
|
||||||
strum_macros = { workspace = true }
|
|
||||||
toml = { workspace = true }
|
|
@ -4,8 +4,8 @@ use serde::{Serialize, Deserialize};
|
|||||||
|
|
||||||
use libpso::packet::ship::{UpdateConfig, WriteInfoboard};
|
use libpso::packet::ship::{UpdateConfig, WriteInfoboard};
|
||||||
use libpso::character::settings::{DEFAULT_PALETTE_CONFIG, DEFAULT_TECH_MENU};
|
use libpso::character::settings::{DEFAULT_PALETTE_CONFIG, DEFAULT_TECH_MENU};
|
||||||
use crate::item::tech::Technique;
|
use crate::entity::item::tech::Technique;
|
||||||
use crate::account::UserAccountId;
|
use crate::entity::account::UserAccountId;
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq, enum_utils::FromStr, derive_more::Display, Serialize, Deserialize, Default)]
|
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq, enum_utils::FromStr, derive_more::Display, Serialize, Deserialize, Default)]
|
||||||
pub enum CharacterClass {
|
pub enum CharacterClass {
|
||||||
@ -223,7 +223,7 @@ impl CharacterInfoboard {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn update_infoboard(&mut self, new_board: &WriteInfoboard) {
|
pub fn update_infoboard(&mut self, new_board: &WriteInfoboard) {
|
||||||
self.board = libpso::util::utf8_to_utf16_array(&new_board.message);
|
self.board = libpso::utf8_to_utf16_array!(new_board.message, 172);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -1,10 +1,10 @@
|
|||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
use futures::future::{Future, BoxFuture};
|
use futures::future::{Future, BoxFuture};
|
||||||
|
|
||||||
use crate::account::*;
|
use crate::entity::account::*;
|
||||||
use crate::character::*;
|
use crate::entity::character::*;
|
||||||
use crate::item::*;
|
use crate::entity::item::*;
|
||||||
use crate::room::*;
|
use crate::entity::room::*;
|
||||||
|
|
||||||
|
|
||||||
// TODO: better granularity?
|
// TODO: better granularity?
|
@ -2,11 +2,11 @@ use std::collections::BTreeMap;
|
|||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
use futures::future::{Future, BoxFuture};
|
use futures::future::{Future, BoxFuture};
|
||||||
|
|
||||||
use crate::account::*;
|
use crate::entity::account::*;
|
||||||
use crate::character::*;
|
use crate::entity::character::*;
|
||||||
use crate::gateway::{EntityGateway, EntityGatewayTransaction, GatewayError};
|
use crate::entity::gateway::{EntityGateway, EntityGatewayTransaction, GatewayError};
|
||||||
use crate::item::*;
|
use crate::entity::item::*;
|
||||||
use crate::room::*;
|
use crate::entity::room::*;
|
||||||
|
|
||||||
use async_std::sync::{Arc, Mutex};
|
use async_std::sync::{Arc, Mutex};
|
||||||
|
|
3
src/entity/gateway/postgres/migrations/mod.rs
Normal file
3
src/entity/gateway/postgres/migrations/mod.rs
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
use refinery::include_migration_mods;
|
||||||
|
|
||||||
|
include_migration_mods!("src/entity/gateway/postgres/migrations");
|
@ -4,13 +4,13 @@ use std::convert::Into;
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use libpso::character::settings;
|
use libpso::character::settings;
|
||||||
use libpso::util::vec_to_array;
|
use libpso::util::vec_to_array;
|
||||||
use crate::account::*;
|
use crate::entity::account::*;
|
||||||
use crate::character::*;
|
use crate::entity::character::*;
|
||||||
use crate::item::*;
|
use crate::entity::item::*;
|
||||||
use crate::room::*;
|
use crate::entity::room::*;
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use maps::room::{Episode, Difficulty};
|
use crate::ship::room::{Episode, Difficulty};
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::monster::MonsterType;
|
||||||
|
|
||||||
#[derive(Debug, sqlx::FromRow)]
|
#[derive(Debug, sqlx::FromRow)]
|
||||||
pub struct PgUserAccount {
|
pub struct PgUserAccount {
|
||||||
@ -253,7 +253,7 @@ impl From<PgCharacter> for CharacterEntity {
|
|||||||
raw_data: vec_to_array(other.config)
|
raw_data: vec_to_array(other.config)
|
||||||
},
|
},
|
||||||
info_board: CharacterInfoboard {
|
info_board: CharacterInfoboard {
|
||||||
board: libpso::util::utf8_to_utf16_array(other.infoboard),
|
board: libpso::utf8_to_utf16_array!(other.infoboard, 172),
|
||||||
},
|
},
|
||||||
guildcard: CharacterGuildCard {
|
guildcard: CharacterGuildCard {
|
||||||
description: other.guildcard,
|
description: other.guildcard,
|
@ -1,16 +1,16 @@
|
|||||||
// this lint is currently bugged and suggests incorrect code https://github.com/rust-lang/rust-clippy/issues/9123
|
// this lint is currently bugged and suggests incorrect code https://github.com/rust-lang/rust-clippy/issues/9123
|
||||||
#![allow(clippy::explicit_auto_deref)]
|
#![allow(clippy::explicit_auto_deref)]
|
||||||
|
|
||||||
use std::convert::{From, Into};
|
use std::convert::{From, TryFrom, Into};
|
||||||
use futures::future::{Future, BoxFuture};
|
use futures::future::{Future, BoxFuture};
|
||||||
use futures::stream::{StreamExt, FuturesOrdered};
|
use futures::stream::{StreamExt, FuturesOrdered};
|
||||||
use async_std::sync::{Arc, Mutex};
|
use async_std::sync::{Arc, Mutex};
|
||||||
use libpso::character::guildcard;
|
use libpso::character::guildcard;
|
||||||
use crate::account::*;
|
use crate::entity::account::*;
|
||||||
use crate::character::*;
|
use crate::entity::character::*;
|
||||||
use crate::gateway::{EntityGateway, EntityGatewayTransaction, GatewayError};
|
use crate::entity::gateway::{EntityGateway, EntityGatewayTransaction, GatewayError};
|
||||||
use crate::item::*;
|
use crate::entity::item::*;
|
||||||
use crate::room::*;
|
use crate::entity::room::*;
|
||||||
use super::models::*;
|
use super::models::*;
|
||||||
|
|
||||||
use sqlx::postgres::PgPoolOptions;
|
use sqlx::postgres::PgPoolOptions;
|
||||||
@ -19,7 +19,7 @@ use sqlx::Connection;
|
|||||||
|
|
||||||
mod embedded {
|
mod embedded {
|
||||||
use refinery::embed_migrations;
|
use refinery::embed_migrations;
|
||||||
embed_migrations!("src/gateway/postgres/migrations");
|
embed_migrations!("src/entity/gateway/postgres/migrations");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use crate::item::ItemEntityId;
|
use crate::entity::item::ItemEntityId;
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
#[derive(Debug, Copy, Clone)]
|
||||||
pub enum ItemParseError {
|
pub enum ItemParseError {
|
@ -1,9 +1,9 @@
|
|||||||
use std::collections::HashMap;
|
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
use std::collections::HashMap;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use crate::item::tool::ToolType;
|
use crate::entity::item::tool::ToolType;
|
||||||
use crate::character::{CharacterClass, SectionID};
|
use crate::entity::character::{CharacterClass, SectionID};
|
||||||
use crate::item::ItemEntityId;
|
use crate::entity::item::ItemEntityId;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
|
||||||
use std::cmp::Ordering::{Less, Greater, Equal};
|
use std::cmp::Ordering::{Less, Greater, Equal};
|
||||||
@ -1029,10 +1029,30 @@ impl Mag {
|
|||||||
MAG_STATS.get(&self.mag).map(|stats| {
|
MAG_STATS.get(&self.mag).map(|stats| {
|
||||||
MAG_FEEDING_TABLES.get(stats.feed_table).map(|feeding_table| {
|
MAG_FEEDING_TABLES.get(stats.feed_table).map(|feeding_table| {
|
||||||
feeding_table.get(&tool).map(|feed_stats| {
|
feeding_table.get(&tool).map(|feed_stats| {
|
||||||
self.def = std::cmp::max(std::cmp::max((self.def as i16) + feed_stats.def, 0) as u16, self.def()*100);
|
self.def = {
|
||||||
self.pow = std::cmp::max(std::cmp::max((self.pow as i16) + feed_stats.pow, 0) as u16, self.pow()*100);
|
if (self.def as i16 + feed_stats.def) < ((self.def()*100) as i16) {
|
||||||
self.dex = std::cmp::max(std::cmp::max((self.dex as i16) + feed_stats.dex, 0) as u16, self.dex()*100);
|
self.def
|
||||||
self.mnd = std::cmp::max(std::cmp::max((self.mnd as i16) + feed_stats.mnd, 0) as u16, self.mind()*100);
|
} else {
|
||||||
|
std::cmp::max(std::cmp::max((self.def as i16) + feed_stats.def, 0) as u16, self.def()*100)
|
||||||
|
}};
|
||||||
|
self.pow = {
|
||||||
|
if (self.pow as i16 + feed_stats.pow) < ((self.pow()*100) as i16) {
|
||||||
|
self.pow
|
||||||
|
} else {
|
||||||
|
std::cmp::max(std::cmp::max((self.pow as i16) + feed_stats.pow, 0) as u16, self.pow()*100)
|
||||||
|
}};
|
||||||
|
self.dex = {
|
||||||
|
if (self.dex as i16 + feed_stats.dex) < ((self.dex()*100) as i16) {
|
||||||
|
self.dex
|
||||||
|
} else {
|
||||||
|
std::cmp::max(std::cmp::max((self.dex as i16) + feed_stats.dex, 0) as u16, self.dex()*100)
|
||||||
|
}};
|
||||||
|
self.mnd = {
|
||||||
|
if (self.mnd as i16 + feed_stats.mnd) < ((self.mind()*100) as i16) {
|
||||||
|
self.mnd
|
||||||
|
} else {
|
||||||
|
std::cmp::max(std::cmp::max((self.mnd as i16) + feed_stats.mnd, 0) as u16, self.mind()*100)
|
||||||
|
}};
|
||||||
self.iq = std::cmp::min(((self.iq as i16) + feed_stats.iq as i16) as u8, 200);
|
self.iq = std::cmp::min(((self.iq as i16) + feed_stats.iq as i16) as u8, 200);
|
||||||
self.synchro = std::cmp::min(((self.synchro as i8) + feed_stats.syn) as u8, 120);
|
self.synchro = std::cmp::min(((self.synchro as i8) + feed_stats.syn) as u8, 120);
|
||||||
})
|
})
|
||||||
@ -1188,7 +1208,7 @@ mod test {
|
|||||||
f.read_to_string(&mut s).unwrap();
|
f.read_to_string(&mut s).unwrap();
|
||||||
|
|
||||||
let mut feed: HashMap<String, Vec<HashMap<String, MagFeedTable>>> = toml::from_str(&s).unwrap();
|
let mut feed: HashMap<String, Vec<HashMap<String, MagFeedTable>>> = toml::from_str(&s).unwrap();
|
||||||
let feed = feed.remove("feedtable".into()).unwrap();
|
let feed = feed.remove("feedtable").unwrap();
|
||||||
let _feed = feed.into_iter()
|
let _feed = feed.into_iter()
|
||||||
.map(|table| {
|
.map(|table| {
|
||||||
table.into_iter()
|
table.into_iter()
|
||||||
@ -1219,7 +1239,7 @@ mod test {
|
|||||||
}
|
}
|
||||||
assert!(mag == Mag {
|
assert!(mag == Mag {
|
||||||
mag: MagType::Sato,
|
mag: MagType::Sato,
|
||||||
def: 507,
|
def: 509,
|
||||||
pow: 5019,
|
pow: 5019,
|
||||||
dex: 4505,
|
dex: 4505,
|
||||||
mnd: 0,
|
mnd: 0,
|
@ -9,11 +9,11 @@ pub mod mag;
|
|||||||
pub mod esweapon;
|
pub mod esweapon;
|
||||||
|
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use crate::character::CharacterEntityId;
|
use crate::entity::character::CharacterEntityId;
|
||||||
use crate::room::RoomEntityId;
|
use crate::entity::room::RoomEntityId;
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::monster::MonsterType;
|
||||||
//use crate::ship::drops::ItemDropType;
|
use crate::ship::drops::ItemDropType;
|
||||||
|
|
||||||
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct ItemEntityId(pub u32);
|
pub struct ItemEntityId(pub u32);
|
||||||
@ -156,6 +156,26 @@ impl ItemDetail {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn parse_item_from_bytes(data: [u8; 16]) -> Option<ItemDropType> {
|
||||||
|
let item_type = weapon::WeaponType::parse_type([data[0],data[1],data[2]]).map(ItemType::Weapon)
|
||||||
|
.or_else(|_| armor::ArmorType::parse_type([data[0],data[1],data[2]]).map(ItemType::Armor))
|
||||||
|
.or_else(|_| shield::ShieldType::parse_type([data[0],data[1],data[2]]).map(ItemType::Shield))
|
||||||
|
.or_else(|_| unit::UnitType::parse_type([data[0],data[1],data[2]]).map(ItemType::Unit))
|
||||||
|
.or_else(|_| mag::MagType::parse_type([data[0],data[1],data[2]]).map(ItemType::Mag))
|
||||||
|
.or_else(|_| tool::ToolType::parse_type([data[0],data[1],data[2]]).map(ItemType::Tool))
|
||||||
|
.or_else(|_| esweapon::ESWeaponType::parse_type([data[0],data[1],data[2]]).map(ItemType::ESWeapon)).ok()?;
|
||||||
|
|
||||||
|
match item_type {
|
||||||
|
ItemType::Weapon(_w) => Some(ItemDropType::Weapon(weapon::Weapon::from_bytes(data).ok()?)),
|
||||||
|
ItemType::Armor(_a) => Some(ItemDropType::Armor(armor::Armor::from_bytes(data).ok()?)),
|
||||||
|
ItemType::Shield(_s) => Some(ItemDropType::Shield(shield::Shield::from_bytes(data).ok()?)),
|
||||||
|
ItemType::Unit(_u) => Some(ItemDropType::Unit(unit::Unit::from_bytes(data).ok()?)),
|
||||||
|
ItemType::Mag(_m) => Some(ItemDropType::Mag(mag::Mag::from_bytes(data).ok()?)),
|
||||||
|
ItemType::Tool(_t) => Some(ItemDropType::Tool(tool::Tool::from_bytes(data).ok()?)),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn as_client_bytes(&self) -> [u8; 16] {
|
pub fn as_client_bytes(&self) -> [u8; 16] {
|
||||||
match self {
|
match self {
|
||||||
ItemDetail::Weapon(w) => w.as_bytes(),
|
ItemDetail::Weapon(w) => w.as_bytes(),
|
@ -1,4 +1,4 @@
|
|||||||
use crate::item::ItemEntityId;
|
use crate::entity::item::ItemEntityId;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
#[derive(Debug, Copy, Clone)]
|
@ -1,8 +1,8 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
|
|
||||||
|
|
||||||
use crate::character::{CharacterEntityId, SectionID};
|
use crate::entity::character::{CharacterEntityId, SectionID};
|
||||||
use maps::room::{Episode, Difficulty};
|
use crate::ship::room::{Episode, Difficulty};
|
||||||
|
|
||||||
|
|
||||||
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash, PartialOrd, Ord, Serialize, Deserialize)]
|
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash, PartialOrd, Ord, Serialize, Deserialize)]
|
@ -1,3 +0,0 @@
|
|||||||
use refinery::include_migration_mods;
|
|
||||||
|
|
||||||
include_migration_mods!("src/gateway/postgres/migrations");
|
|
@ -1,31 +0,0 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
|
||||||
|
|
||||||
use super::account::UserAccountId;
|
|
||||||
|
|
||||||
// [2022-10-23 00:11:18][elseware::common::mainloop::client][WARN] error RecvServerPacket::from_bytes: WrongPacketForServerType(490, [40, 0, 234, 1, 0, 0, 0, 0, 9, 0, 74, 0, 97, 0, 115, 0, 100, 0, 102, 0, 0, 0, 0, 0, 192, 52, 67, 3, 60, 159, 129, 0, 32, 64, 233, 10, 196, 156, 152, 0])
|
|
||||||
// [2022-10-23 00:20:14][elseware::common::mainloop::client][WARN] error RecvServerPacket::from_bytes: WrongPacketForServerType(490, [40, 0, 234, 1, 0, 0, 0, 0, 9, 0, 74, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 97, 0, 0, 0, 152, 0])
|
|
||||||
|
|
||||||
|
|
||||||
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash, PartialOrd, Ord, Serialize, Deserialize)]
|
|
||||||
pub struct TeamEntityId(pub u32);
|
|
||||||
|
|
||||||
pub struct NewTeamEntity {
|
|
||||||
pub created_by: UserAccountId,
|
|
||||||
pub name: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct TeamEntity {
|
|
||||||
pub id: TeamEntityId,
|
|
||||||
pub owner: UserAccountId,
|
|
||||||
pub name: String,
|
|
||||||
|
|
||||||
pub team_flag: [u8; 2048],
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -1,25 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "items"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
shops = { workspace = true }
|
|
||||||
location = { workspace = true }
|
|
||||||
drops = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
enum-utils = { workspace = true }
|
|
||||||
derive_more = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
rand = { workspace = true }
|
|
||||||
rand_chacha = { workspace = true }
|
|
||||||
async-recursion = { workspace = true }
|
|
||||||
async-std = { workspace = true }
|
|
||||||
async-trait = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
@ -1,38 +0,0 @@
|
|||||||
use crate::ClientItemId;
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub enum TradeItem {
|
|
||||||
Individual(ClientItemId),
|
|
||||||
Stacked(ClientItemId, usize),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TradeItem {
|
|
||||||
pub fn stacked(&self) -> Option<(ClientItemId, usize)> {
|
|
||||||
match self {
|
|
||||||
TradeItem::Stacked(item_id, amount) => Some((*item_id, *amount)),
|
|
||||||
_ => None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn stacked_mut(&mut self) -> Option<(ClientItemId, &mut usize)> {
|
|
||||||
match self {
|
|
||||||
TradeItem::Stacked(item_id, ref mut amount) => Some((*item_id, amount)),
|
|
||||||
_ => None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn item_id(&self) -> ClientItemId {
|
|
||||||
match self {
|
|
||||||
TradeItem::Individual(item_id) => *item_id,
|
|
||||||
TradeItem::Stacked(item_id, _) => *item_id,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn amount(&self) -> usize {
|
|
||||||
match self {
|
|
||||||
TradeItem::Individual(_) => 1,
|
|
||||||
TradeItem::Stacked(_, amount) => *amount,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
16
src/lib.rs
Normal file
16
src/lib.rs
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#![allow(clippy::type_complexity)]
|
||||||
|
#![allow(incomplete_features)]
|
||||||
|
#![feature(inline_const)]
|
||||||
|
#![feature(extract_if)]
|
||||||
|
#![feature(try_blocks)]
|
||||||
|
#![feature(test)]
|
||||||
|
#![feature(error_generic_member_access)]
|
||||||
|
#![feature(lazy_cell)]
|
||||||
|
|
||||||
|
extern crate test;
|
||||||
|
|
||||||
|
pub mod common;
|
||||||
|
pub mod entity;
|
||||||
|
pub mod patch;
|
||||||
|
pub mod login;
|
||||||
|
pub mod ship;
|
@ -1,12 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "location"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
networking = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
derive_more = { workspace = true }
|
|
||||||
futures= { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
@ -12,31 +12,30 @@ use libpso::packet::login::*;
|
|||||||
use libpso::packet::ship::{MenuDetail, SmallLeftDialog};
|
use libpso::packet::ship::{MenuDetail, SmallLeftDialog};
|
||||||
use libpso::{PacketParseError, PSOPacket};
|
use libpso::{PacketParseError, PSOPacket};
|
||||||
use libpso::crypto::bb::PSOBBCipher;
|
use libpso::crypto::bb::PSOBBCipher;
|
||||||
|
use crate::entity::item;
|
||||||
use libpso::character::character;
|
use libpso::character::character;
|
||||||
use entity::item;
|
|
||||||
|
|
||||||
use networking::cipherkeys::{ELSEWHERE_PRIVATE_KEY, ELSEWHERE_PARRAY};
|
use crate::common::cipherkeys::{ELSEWHERE_PRIVATE_KEY, ELSEWHERE_PARRAY};
|
||||||
use networking::serverstate::{SendServerPacket, RecvServerPacket, ServerState, OnConnect, ClientId};
|
use crate::common::serverstate::{SendServerPacket, RecvServerPacket, ServerState, OnConnect, ClientId};
|
||||||
use networking::interserver::{ServerId, InterserverActor, LoginMessage, ShipMessage, Ship};
|
use crate::common::interserver::{ServerId, InterserverActor, LoginMessage, ShipMessage, Ship};
|
||||||
use stats::leveltable::LEVEL_TABLE;
|
use crate::common::leveltable::LEVEL_TABLE;
|
||||||
use libpso::util::{utf8_to_array, utf8_to_utf16_array};
|
use libpso::{utf8_to_array, utf8_to_utf16_array};
|
||||||
|
|
||||||
use entity::gateway::{EntityGateway, GatewayError};
|
use crate::entity::gateway::{EntityGateway, GatewayError};
|
||||||
use entity::account::{UserAccountId, UserAccountEntity, NewUserSettingsEntity, USERFLAG_NEWCHAR, USERFLAG_DRESSINGROOM};
|
use crate::entity::account::{UserAccountId, UserAccountEntity, NewUserSettingsEntity, USERFLAG_NEWCHAR, USERFLAG_DRESSINGROOM};
|
||||||
use entity::item::{NewItemEntity, ItemDetail, ItemNote, InventoryItemEntity, InventoryEntity, BankEntity, BankIdentifier, EquippedEntity, Meseta};
|
use crate::entity::item::{NewItemEntity, ItemDetail, ItemNote, InventoryItemEntity, InventoryEntity, BankEntity, BankIdentifier, EquippedEntity, Meseta};
|
||||||
use entity::item::weapon::Weapon;
|
use crate::entity::item::weapon::Weapon;
|
||||||
use entity::item::armor::Armor;
|
use crate::entity::item::armor::Armor;
|
||||||
use entity::item::tech::Technique;
|
use crate::entity::item::tech::Technique;
|
||||||
use entity::item::tool::Tool;
|
use crate::entity::item::tool::Tool;
|
||||||
use entity::item::mag::Mag;
|
use crate::entity::item::mag::Mag;
|
||||||
use entity::character::{CharacterEntity, NewCharacterEntity, CharacterClass, TechLevel};
|
use crate::entity::character::{CharacterEntity, NewCharacterEntity, CharacterClass, TechLevel};
|
||||||
|
|
||||||
use crate::login::get_login_status;
|
use crate::login::login::{get_login_status};
|
||||||
use networking::interserver::AuthToken;
|
use crate::common::interserver::AuthToken;
|
||||||
|
|
||||||
use pktbuilder::ship::SHIP_MENU_ID;
|
|
||||||
|
|
||||||
pub const CHARACTER_PORT: u16 = 12001;
|
pub const CHARACTER_PORT: u16 = 12001;
|
||||||
|
pub const SHIP_MENU_ID: u32 = 1;
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum CharacterError {
|
pub enum CharacterError {
|
||||||
@ -150,7 +149,7 @@ fn generate_param_data(path: &str) -> (ParamDataHeader, Vec<u8>) {
|
|||||||
size: len as u32,
|
size: len as u32,
|
||||||
checksum: crc.sum32(),
|
checksum: crc.sum32(),
|
||||||
offset: buffer.len() as u32,
|
offset: buffer.len() as u32,
|
||||||
filename: utf8_to_array(param.file_name().unwrap().to_str().unwrap()),
|
filename: utf8_to_array!(param.file_name().unwrap().to_str().unwrap(), 0x40),
|
||||||
});
|
});
|
||||||
|
|
||||||
buffer.append(&mut filebuf);
|
buffer.append(&mut filebuf);
|
||||||
@ -177,7 +176,7 @@ impl ClientState {
|
|||||||
user: None,
|
user: None,
|
||||||
characters: None,
|
characters: None,
|
||||||
guildcard_data_buffer: None,
|
guildcard_data_buffer: None,
|
||||||
session: Session::default(),
|
session: Session::new(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -342,15 +341,15 @@ impl<EG: EntityGateway + Clone> CharacterServerState<EG> {
|
|||||||
if let Some(connected_client) = self.connected_clients.read().await.get(&user.id) {
|
if let Some(connected_client) = self.connected_clients.read().await.get(&user.id) {
|
||||||
if let Some(expires) = connected_client.expires {
|
if let Some(expires) = connected_client.expires {
|
||||||
if expires > chrono::Utc::now() {
|
if expires > chrono::Utc::now() {
|
||||||
return Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(AccountStatus::AlreadyOnline, Session::default()))]);
|
return Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(AccountStatus::AlreadyOnline, Session::new()))]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(AccountStatus::AlreadyOnline, Session::default()))]);
|
return Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(AccountStatus::AlreadyOnline, Session::new()))]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut response = LoginResponse::by_status(AccountStatus::Ok, Session::default());
|
let mut response = LoginResponse::by_status(AccountStatus::Ok, Session::new());
|
||||||
response.guildcard = user.guildcard;
|
response.guildcard = user.guildcard;
|
||||||
response.team_id = user.team_id.map_or(0, |ti| ti);
|
response.team_id = user.team_id.map_or(0, |ti| ti);
|
||||||
|
|
||||||
@ -367,7 +366,7 @@ impl<EG: EntityGateway + Clone> CharacterServerState<EG> {
|
|||||||
Ok(vec![SendCharacterPacket::LoginResponse(response)])
|
Ok(vec![SendCharacterPacket::LoginResponse(response)])
|
||||||
},
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(err, Session::default()))])
|
Ok(vec![SendCharacterPacket::LoginResponse(LoginResponse::by_status(err, Session::new()))])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -379,7 +378,7 @@ impl<EG: EntityGateway + Clone> CharacterServerState<EG> {
|
|||||||
menu: SHIP_MENU_ID,
|
menu: SHIP_MENU_ID,
|
||||||
item: i.0 as u32,
|
item: i.0 as u32,
|
||||||
flags: 0,
|
flags: 0,
|
||||||
name: utf8_to_utf16_array(&s.name)
|
name: utf8_to_utf16_array!(s.name, 0x11)
|
||||||
}
|
}
|
||||||
}).collect()))
|
}).collect()))
|
||||||
])
|
])
|
||||||
@ -824,7 +823,7 @@ impl<'a> SelectScreenCharacterBuilder<'a> {
|
|||||||
hair_b: character.appearance.hair_b,
|
hair_b: character.appearance.hair_b,
|
||||||
prop_x: character.appearance.prop_x,
|
prop_x: character.appearance.prop_x,
|
||||||
prop_y: character.appearance.prop_y,
|
prop_y: character.appearance.prop_y,
|
||||||
name: utf8_to_utf16_array(&character.name),
|
name: utf8_to_utf16_array!(character.name, 16),
|
||||||
play_time: character.playtime,
|
play_time: character.playtime,
|
||||||
..character::SelectScreenCharacter::default()
|
..character::SelectScreenCharacter::default()
|
||||||
}
|
}
|
||||||
@ -835,21 +834,9 @@ impl<'a> SelectScreenCharacterBuilder<'a> {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
use entity::account::*;
|
use crate::entity::account::*;
|
||||||
use libpso::character::{settings, character};
|
use libpso::character::{settings, character};
|
||||||
use entity::gateway::{InMemoryGateway, EntityGatewayTransaction, GatewayError};
|
use crate::entity::gateway::{InMemoryGateway, GatewayError};
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
struct CharTestDb;
|
|
||||||
|
|
||||||
impl EntityGateway for CharTestDb {
|
|
||||||
type Transaction<'t> = CharTestDb where Self: 't;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl EntityGatewayTransaction for CharTestDb {
|
|
||||||
type ParentGateway = CharTestDb;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#[async_std::test]
|
#[async_std::test]
|
||||||
async fn test_option_send() {
|
async fn test_option_send() {
|
||||||
@ -859,7 +846,7 @@ mod test {
|
|||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'a> = CharTestDb where Self: 'a;
|
type Transaction<'a> = () where Self: 'a;
|
||||||
async fn get_user_settings_by_user(&mut self, user: &UserAccountEntity) -> Result<UserSettingsEntity, GatewayError> {
|
async fn get_user_settings_by_user(&mut self, user: &UserAccountEntity) -> Result<UserSettingsEntity, GatewayError> {
|
||||||
Ok(UserSettingsEntity {
|
Ok(UserSettingsEntity {
|
||||||
id: UserSettingsId(0),
|
id: UserSettingsId(0),
|
||||||
@ -902,7 +889,7 @@ mod test {
|
|||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct TestData;
|
struct TestData;
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'a> = CharTestDb where Self: 'a;
|
type Transaction<'a> = () where Self: 'a;
|
||||||
}
|
}
|
||||||
let mut server = CharacterServerState::new(TestData {}, AuthToken("".into()));
|
let mut server = CharacterServerState::new(TestData {}, AuthToken("".into()));
|
||||||
let send = server.handle(ClientId(1), RecvCharacterPacket::Checksum(Checksum {checksum: 1234,
|
let send = server.handle(ClientId(1), RecvCharacterPacket::Checksum(Checksum {checksum: 1234,
|
@ -11,11 +11,11 @@ use libpso::{PacketParseError, PSOPacket};
|
|||||||
use libpso::crypto::bb::PSOBBCipher;
|
use libpso::crypto::bb::PSOBBCipher;
|
||||||
use libpso::util::array_to_utf8;
|
use libpso::util::array_to_utf8;
|
||||||
|
|
||||||
use networking::cipherkeys::{ELSEWHERE_PRIVATE_KEY, ELSEWHERE_PARRAY};
|
use crate::common::cipherkeys::{ELSEWHERE_PRIVATE_KEY, ELSEWHERE_PARRAY};
|
||||||
use networking::serverstate::{SendServerPacket, RecvServerPacket, ServerState, OnConnect, ClientId};
|
use crate::common::serverstate::{SendServerPacket, RecvServerPacket, ServerState, OnConnect, ClientId};
|
||||||
|
|
||||||
use entity::gateway::EntityGateway;
|
use crate::entity::gateway::EntityGateway;
|
||||||
use entity::account::{UserAccountEntity};
|
use crate::entity::account::{UserAccountEntity};
|
||||||
|
|
||||||
pub const LOGIN_PORT: u16 = 12000;
|
pub const LOGIN_PORT: u16 = 12000;
|
||||||
pub const COMMUNICATION_PORT: u16 = 12123;
|
pub const COMMUNICATION_PORT: u16 = 12123;
|
||||||
@ -83,13 +83,21 @@ pub async fn get_login_status(entity_gateway: &mut impl EntityGateway, pkt: &Log
|
|||||||
|
|
||||||
pub fn check_if_already_online(user: UserAccountEntity) -> Result<UserAccountEntity, AccountStatus> {
|
pub fn check_if_already_online(user: UserAccountEntity) -> Result<UserAccountEntity, AccountStatus> {
|
||||||
Ok(user)
|
Ok(user)
|
||||||
|
/*
|
||||||
|
if user.is_currently_online() {
|
||||||
|
Err(AccountStatus::PayUp)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
Ok(user)
|
||||||
|
}
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct LoginServerState<EG: EntityGateway + Clone> {
|
pub struct LoginServerState<EG: EntityGateway + Clone> {
|
||||||
character_server_ip: net::Ipv4Addr,
|
character_server_ip: net::Ipv4Addr,
|
||||||
entity_gateway: EG,
|
entity_gateway: EG,
|
||||||
clients: HashMap<ClientId, String>, // TODO: this should be arc/mutex'd?
|
clients: HashMap<ClientId, String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<EG: EntityGateway + Clone> LoginServerState<EG> {
|
impl<EG: EntityGateway + Clone> LoginServerState<EG> {
|
||||||
@ -111,7 +119,7 @@ impl<EG: EntityGateway + Clone> LoginServerState<EG> {
|
|||||||
let response = SendLoginPacket::LoginResponse(LoginResponse::by_status(AccountStatus::Ok, pkt.session));
|
let response = SendLoginPacket::LoginResponse(LoginResponse::by_status(AccountStatus::Ok, pkt.session));
|
||||||
let ip = u32::from_ne_bytes(self.character_server_ip.octets());
|
let ip = u32::from_ne_bytes(self.character_server_ip.octets());
|
||||||
Ok(vec![response,
|
Ok(vec![response,
|
||||||
SendLoginPacket::RedirectClient(RedirectClient::new(ip, crate::character::CHARACTER_PORT))])
|
SendLoginPacket::RedirectClient(RedirectClient::new(ip, crate::login::character::CHARACTER_PORT))])
|
||||||
},
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
Ok(vec![SendLoginPacket::LoginResponse(LoginResponse::by_status(err, pkt.session))])
|
Ok(vec![SendLoginPacket::LoginResponse(LoginResponse::by_status(err, pkt.session))])
|
||||||
@ -170,8 +178,8 @@ impl<EG: EntityGateway + Clone> ServerState for LoginServerState<EG> {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
use entity::account::{UserAccountId};
|
use crate::entity::account::{UserAccountId};
|
||||||
use entity::gateway::{EntityGatewayTransaction, GatewayError};
|
use crate::entity::gateway::{EntityGatewayTransaction, GatewayError};
|
||||||
|
|
||||||
const LOGIN_PACKET: RecvLoginPacket = RecvLoginPacket::Login(Login {
|
const LOGIN_PACKET: RecvLoginPacket = RecvLoginPacket::Login(Login {
|
||||||
tag: 65536,
|
tag: 65536,
|
||||||
@ -197,15 +205,12 @@ mod test {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
#[derive(Clone)]
|
impl EntityGateway for () {
|
||||||
struct LoginTestDb;
|
type Transaction<'t> = () where Self: 't;
|
||||||
|
|
||||||
impl EntityGateway for LoginTestDb {
|
|
||||||
type Transaction<'t> = LoginTestDb where Self: 't;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl EntityGatewayTransaction for LoginTestDb {
|
impl EntityGatewayTransaction for () {
|
||||||
type ParentGateway = LoginTestDb;
|
type ParentGateway = ();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_std::test]
|
#[async_std::test]
|
||||||
@ -216,7 +221,7 @@ mod test {
|
|||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'t> = LoginTestDb where Self: 't;
|
type Transaction<'t> = () where Self: 't;
|
||||||
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
||||||
assert!(name == "testuser");
|
assert!(name == "testuser");
|
||||||
Ok(UserAccountEntity {
|
Ok(UserAccountEntity {
|
||||||
@ -275,7 +280,7 @@ mod test {
|
|||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'t> = LoginTestDb where Self: 't;
|
type Transaction<'t> = () where Self: 't;
|
||||||
async fn get_user_by_name(&mut self, _name: String) -> Result<UserAccountEntity, GatewayError> {
|
async fn get_user_by_name(&mut self, _name: String) -> Result<UserAccountEntity, GatewayError> {
|
||||||
Err(GatewayError::Error)
|
Err(GatewayError::Error)
|
||||||
}
|
}
|
||||||
@ -310,7 +315,7 @@ mod test {
|
|||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'t> = LoginTestDb where Self: 't;
|
type Transaction<'t> = () where Self: 't;
|
||||||
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
||||||
assert!(name == "testuser");
|
assert!(name == "testuser");
|
||||||
Ok(UserAccountEntity {
|
Ok(UserAccountEntity {
|
||||||
@ -360,7 +365,7 @@ mod test {
|
|||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl EntityGateway for TestData {
|
impl EntityGateway for TestData {
|
||||||
type Transaction<'t> = LoginTestDb where Self: 't;
|
type Transaction<'t> = () where Self: 't;
|
||||||
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
async fn get_user_by_name(&mut self, name: String) -> Result<UserAccountEntity, GatewayError> {
|
||||||
assert!(name == "testuser");
|
assert!(name == "testuser");
|
||||||
Ok(UserAccountEntity {
|
Ok(UserAccountEntity {
|
3
src/login/mod.rs
Normal file
3
src/login/mod.rs
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
#[allow(clippy::module_inception)]
|
||||||
|
pub mod login;
|
||||||
|
pub mod character;
|
92
src/login/models.rs
Normal file
92
src/login/models.rs
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
use std::time::SystemTime;
|
||||||
|
use std::io::Write;
|
||||||
|
//use diesel::sql_types::Timestamp;
|
||||||
|
use diesel::{Insertable, Queryable, Identifiable, Associations, AsExpression, FromSqlRow};
|
||||||
|
//use bcrypt::{DEFAULT_COST, hash};
|
||||||
|
use diesel::pg::Pg;
|
||||||
|
use diesel::sql_types;
|
||||||
|
use diesel::deserialize::{self, FromSql};
|
||||||
|
use diesel::serialize::{self, ToSql, Output, IsNull};
|
||||||
|
use diesel::backend::Backend;
|
||||||
|
|
||||||
|
use libpso::character::settings;
|
||||||
|
|
||||||
|
use elseware::schema::*;
|
||||||
|
|
||||||
|
//const ELSEWHERE_COST: u32 = bcrypt::DEFAULT_COST;
|
||||||
|
const ELSEWHERE_COST: u32 = 5;
|
||||||
|
|
||||||
|
#[derive(Debug, AsExpression, FromSqlRow)]
|
||||||
|
#[sql_type="sql_types::Binary"]
|
||||||
|
pub struct EUserSettings(pub settings::UserSettings);
|
||||||
|
|
||||||
|
impl std::ops::Deref for EUserSettings {
|
||||||
|
type Target = settings::UserSettings;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Queryable, Identifiable, Debug)]
|
||||||
|
pub struct UserAccount {
|
||||||
|
pub id: i32,
|
||||||
|
pub username: String,
|
||||||
|
pub password: String,
|
||||||
|
pub guildcard: Option<i32>,
|
||||||
|
pub team_id: Option<i32>,
|
||||||
|
pub banned: bool,
|
||||||
|
pub muted_until: SystemTime,
|
||||||
|
pub created_at: SystemTime,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable)]
|
||||||
|
#[table_name="user_accounts"]
|
||||||
|
pub struct NewUser {
|
||||||
|
username: String,
|
||||||
|
password: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl NewUser {
|
||||||
|
pub fn new(username: String, password: String) -> NewUser {
|
||||||
|
let crypt_password = bcrypt::hash(password, ELSEWHERE_COST).expect("could not hash password?");
|
||||||
|
NewUser {
|
||||||
|
username: username,
|
||||||
|
password: crypt_password,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Queryable, Identifiable, Associations)]
|
||||||
|
#[belongs_to(UserAccount, foreign_key="user_id")]
|
||||||
|
#[table_name="user_settings"]
|
||||||
|
pub struct UserSettings {
|
||||||
|
pub id: i32,
|
||||||
|
pub user_id: i32,
|
||||||
|
//settings: Vec<u8>,
|
||||||
|
pub settings: EUserSettings,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable, Debug)]
|
||||||
|
#[table_name="user_settings"]
|
||||||
|
pub struct NewUserSettings {
|
||||||
|
pub user_id: i32,
|
||||||
|
pub settings: EUserSettings,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToSql<sql_types::Binary, Pg> for EUserSettings {
|
||||||
|
fn to_sql<W: Write>(&self, out: &mut Output<W, Pg>) -> serialize::Result {
|
||||||
|
out.write_all(&self.0.as_bytes()[..])
|
||||||
|
.map(|_| IsNull::No)
|
||||||
|
.map_err(|e| Box::new(e) as Box<dyn std::error::Error + Send + Sync>)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromSql<sql_types::Binary, Pg> for EUserSettings {
|
||||||
|
fn from_sql(bytes: Option<&[u8]>) -> deserialize::Result<Self> {
|
||||||
|
let bytes_vec: Vec<u8> = <Vec<u8> as FromSql<sql_types::Binary, Pg>>::from_sql(bytes)?;
|
||||||
|
let mut static_bytes = [0u8; 0x1160];
|
||||||
|
static_bytes[..0x1160].clone_from_slice(&bytes_vec);
|
||||||
|
Ok(EUserSettings(settings::UserSettings::from_bytes(static_bytes)))
|
||||||
|
}
|
||||||
|
}
|
@ -1,21 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "login_server"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
networking = { workspace = true }
|
|
||||||
pktbuilder = { workspace = true }
|
|
||||||
stats = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
async-trait = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
bcrypt = { workspace = true }
|
|
||||||
crc = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
chrono = { workspace = true }
|
|
||||||
rand= { workspace = true }
|
|
@ -1,2 +0,0 @@
|
|||||||
pub mod login;
|
|
||||||
pub mod character;
|
|
@ -1,14 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "maps"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
byteorder = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
rand = { workspace = true }
|
|
||||||
rand_chacha = { workspace = true }
|
|
||||||
toml = { workspace = true }
|
|
||||||
enum-utils = { workspace = true }
|
|
||||||
derive_more = { workspace = true }
|
|
@ -1,59 +0,0 @@
|
|||||||
pub mod area;
|
|
||||||
pub mod enemy;
|
|
||||||
pub mod object;
|
|
||||||
pub mod variant;
|
|
||||||
pub mod maps;
|
|
||||||
pub mod monster;
|
|
||||||
pub mod room;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy)]
|
|
||||||
pub enum Holiday {
|
|
||||||
None,
|
|
||||||
Christmas,
|
|
||||||
Valentines,
|
|
||||||
Easter,
|
|
||||||
Halloween,
|
|
||||||
Sonic,
|
|
||||||
NewYear,
|
|
||||||
Summer,
|
|
||||||
White,
|
|
||||||
Wedding,
|
|
||||||
Fall,
|
|
||||||
Spring,
|
|
||||||
Summer2,
|
|
||||||
Spring2,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
impl From<Holiday> for u32 {
|
|
||||||
fn from(other: Holiday) -> u32 {
|
|
||||||
u16::from(other) as u32
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Holiday> for u16 {
|
|
||||||
fn from(other: Holiday) -> u16 {
|
|
||||||
u8::from(other) as u16
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Holiday> for u8 {
|
|
||||||
fn from(other: Holiday) -> u8 {
|
|
||||||
match other {
|
|
||||||
Holiday::None => 0,
|
|
||||||
Holiday::Christmas => 1,
|
|
||||||
Holiday::Valentines => 3,
|
|
||||||
Holiday::Easter => 4,
|
|
||||||
Holiday::Halloween => 5,
|
|
||||||
Holiday::Sonic => 6,
|
|
||||||
Holiday::NewYear => 7,
|
|
||||||
Holiday::Summer => 8,
|
|
||||||
Holiday::White => 9,
|
|
||||||
Holiday::Wedding => 10,
|
|
||||||
Holiday::Fall => 11,
|
|
||||||
Holiday::Spring => 12,
|
|
||||||
Holiday::Summer2 => 13,
|
|
||||||
Holiday::Spring2 => 14,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,150 +0,0 @@
|
|||||||
#[derive(Debug, Copy, Clone, derive_more::Display)]
|
|
||||||
pub enum Episode {
|
|
||||||
#[display(fmt="ep1")]
|
|
||||||
One,
|
|
||||||
#[display(fmt="ep2")]
|
|
||||||
Two,
|
|
||||||
#[display(fmt="ep4")]
|
|
||||||
Four,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TryFrom<u8> for Episode {
|
|
||||||
type Error = ();
|
|
||||||
|
|
||||||
fn try_from(value: u8) -> Result<Episode, ()> {
|
|
||||||
match value {
|
|
||||||
1 => Ok(Episode::One),
|
|
||||||
2 => Ok(Episode::Two),
|
|
||||||
3 => Ok(Episode::Four),
|
|
||||||
_ => Err(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Episode> for u8 {
|
|
||||||
fn from(other: Episode) -> u8 {
|
|
||||||
match other {
|
|
||||||
Episode::One => 1,
|
|
||||||
Episode::Two => 2,
|
|
||||||
Episode::Four => 3,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Episode {
|
|
||||||
pub fn from_quest(value: u8) -> Option<Episode> {
|
|
||||||
match value {
|
|
||||||
0 => Some(Episode::One),
|
|
||||||
1 => Some(Episode::Two),
|
|
||||||
2 => Some(Episode::Four),
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq, derive_more::Display)]
|
|
||||||
pub enum Difficulty {
|
|
||||||
Normal,
|
|
||||||
Hard,
|
|
||||||
VeryHard,
|
|
||||||
Ultimate,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TryFrom<u8> for Difficulty {
|
|
||||||
type Error = ();
|
|
||||||
|
|
||||||
fn try_from(value: u8) -> Result<Difficulty, ()> {
|
|
||||||
match value {
|
|
||||||
0 => Ok(Difficulty::Normal),
|
|
||||||
1 => Ok(Difficulty::Hard),
|
|
||||||
2 => Ok(Difficulty::VeryHard),
|
|
||||||
3 => Ok(Difficulty::Ultimate),
|
|
||||||
_ => Err(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Difficulty> for u8 {
|
|
||||||
fn from(other: Difficulty) -> u8 {
|
|
||||||
match other {
|
|
||||||
Difficulty::Normal => 0,
|
|
||||||
Difficulty::Hard => 1,
|
|
||||||
Difficulty::VeryHard => 2,
|
|
||||||
Difficulty::Ultimate => 3,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
|
||||||
pub enum PlayerMode {
|
|
||||||
Single,
|
|
||||||
Multi,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PlayerMode {
|
|
||||||
pub fn value(&self) -> u8 {
|
|
||||||
match self {
|
|
||||||
PlayerMode::Single => 1,
|
|
||||||
PlayerMode::Multi => 0,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, derive_more::Display)]
|
|
||||||
pub enum RoomMode {
|
|
||||||
#[display(fmt="single")]
|
|
||||||
Single {
|
|
||||||
episode: Episode,
|
|
||||||
difficulty: Difficulty,
|
|
||||||
},
|
|
||||||
#[display(fmt="multi")]
|
|
||||||
Multi {
|
|
||||||
episode: Episode,
|
|
||||||
difficulty: Difficulty,
|
|
||||||
},
|
|
||||||
#[display(fmt="challenge")]
|
|
||||||
Challenge {
|
|
||||||
episode: Episode,
|
|
||||||
},
|
|
||||||
#[display(fmt="battle")]
|
|
||||||
Battle {
|
|
||||||
episode: Episode,
|
|
||||||
difficulty: Difficulty,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
impl RoomMode {
|
|
||||||
pub fn difficulty(&self) -> Difficulty {
|
|
||||||
match self {
|
|
||||||
RoomMode::Single {difficulty, ..} => *difficulty,
|
|
||||||
RoomMode::Multi {difficulty, ..} => *difficulty,
|
|
||||||
RoomMode::Battle {difficulty, ..} => *difficulty,
|
|
||||||
RoomMode::Challenge {..} => Difficulty::Normal,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn episode(&self) -> Episode {
|
|
||||||
match self {
|
|
||||||
RoomMode::Single {episode, ..} => *episode,
|
|
||||||
RoomMode::Multi {episode, ..} => *episode,
|
|
||||||
RoomMode::Battle {episode, ..} => *episode,
|
|
||||||
RoomMode::Challenge {episode, ..} => *episode,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn battle(&self) -> bool {
|
|
||||||
matches!(self, RoomMode::Battle {..})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn challenge(&self) -> bool {
|
|
||||||
matches!(self, RoomMode::Challenge {..})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn player_mode(&self) -> PlayerMode {
|
|
||||||
match self {
|
|
||||||
RoomMode::Single {..} => PlayerMode::Single,
|
|
||||||
_ => PlayerMode::Multi,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,18 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "networking"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
entity = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
async-trait = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
log = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
serde_json = { workspace = true }
|
|
||||||
derive_more = { workspace = true }
|
|
@ -1,4 +0,0 @@
|
|||||||
pub mod cipherkeys;
|
|
||||||
pub mod serverstate;
|
|
||||||
pub mod mainloop;
|
|
||||||
pub mod interserver;
|
|
2
src/patch/mod.rs
Normal file
2
src/patch/mod.rs
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
#[allow(clippy::module_inception)]
|
||||||
|
pub mod patch;
|
@ -11,8 +11,8 @@ use libpso::crypto::pc::PSOPCCipher;
|
|||||||
use ron::de::from_str;
|
use ron::de::from_str;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
|
|
||||||
use networking::mainloop::{NetworkError};
|
use crate::common::mainloop::{NetworkError};
|
||||||
use networking::serverstate::{RecvServerPacket, SendServerPacket, ServerState, OnConnect, ClientId};
|
use crate::common::serverstate::{RecvServerPacket, SendServerPacket, ServerState, OnConnect, ClientId};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum PatchError {
|
pub enum PatchError {
|
||||||
@ -341,7 +341,7 @@ impl Iterator for SendFileIterator {
|
|||||||
if len == 0 {
|
if len == 0 {
|
||||||
self.current_file = None;
|
self.current_file = None;
|
||||||
self.chunk_num = 0;
|
self.chunk_num = 0;
|
||||||
Some(SendPatchPacket::EndFileSend(EndFileSend::default()))
|
Some(SendPatchPacket::EndFileSend(EndFileSend::new()))
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
let mut crc = crc32::Digest::new(crc32::IEEE);
|
let mut crc = crc32::Digest::new(crc32::IEEE);
|
@ -1,15 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "patch_server"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
networking = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-trait = { workspace = true }
|
|
||||||
rand = { workspace = true }
|
|
||||||
crc = { workspace = true }
|
|
||||||
ron = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
@ -1,22 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "pktbuilder"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
quests = { workspace = true }
|
|
||||||
stats = { workspace = true }
|
|
||||||
location = { workspace = true }
|
|
||||||
client = { workspace = true }
|
|
||||||
items = { workspace = true }
|
|
||||||
networking = { workspace = true }
|
|
||||||
maps = { workspace = true }
|
|
||||||
room = { workspace = true }
|
|
||||||
shops = { workspace = true }
|
|
||||||
entity = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
@ -1,107 +0,0 @@
|
|||||||
use futures::stream::{FuturesOrdered, StreamExt};
|
|
||||||
use libpso::packet::ship::*;
|
|
||||||
use crate::common::serverstate::ClientId;
|
|
||||||
use crate::entity::gateway::EntityGateway;
|
|
||||||
use crate::ship::client::{Clients, ClientState};
|
|
||||||
use crate::ship::teams::Teams;
|
|
||||||
use crate::ship::location::ClientLocation;
|
|
||||||
use crate::ship::ship::ShipError;
|
|
||||||
use crate::entity::team::TeamEntity;
|
|
||||||
|
|
||||||
|
|
||||||
pub fn client_team_state_changed(client_id: ClientId, client: &ClientState, team: &TeamEntity) -> ClientTeamStateChanged {
|
|
||||||
ClientTeamStateChanged {
|
|
||||||
unknown: 0,
|
|
||||||
guildcard: client.user.guildcard(),
|
|
||||||
team_id: team.id.0,
|
|
||||||
unknown2: [0;2],
|
|
||||||
privilege: 0x40, // TODO: improve
|
|
||||||
team_name: libpso::utf8_to_utf16_array!(team.name, 14),
|
|
||||||
unknown3: 0x00986C84, // TODO: what if we omit this?
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn player_team_info(client_id: ClientId, client: &ClientState, team: &TeamEntity) -> PlayerTeamInfo {
|
|
||||||
PlayerTeamInfo {
|
|
||||||
guildcard: client.user.guildcard(),
|
|
||||||
team_id: team.id.0,
|
|
||||||
info: 0,
|
|
||||||
info2: 0,
|
|
||||||
privilege: 0x40, // TODO: improve
|
|
||||||
team_name: libpso::utf8_to_utf16_array!(team.name, 14),
|
|
||||||
unknown: 0x00986C84, // TODO: what if we omit this?
|
|
||||||
guildcard_again: client.user.guildcard(),
|
|
||||||
client_id: client_id.0 as u32,
|
|
||||||
character_name: libpso::utf8_to_utf16_array!(client.character.name, 12),
|
|
||||||
unknown2: 0,
|
|
||||||
unknown3: 0,
|
|
||||||
team_flag: team.team_flag,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn team_info_individual(client_id: ClientId, client: &ClientState, team: &TeamEntity) -> TeamInfo {
|
|
||||||
TeamInfo {
|
|
||||||
clients: vec![player_team_info(client_id, client, team)]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
pub async fn player_team_info_list<EG>(id: ClientId,
|
|
||||||
client_location: &ClientLocation,
|
|
||||||
clients: &Clients,
|
|
||||||
teams: &Teams<EG>,
|
|
||||||
) -> Result<Vec<PlayerTeamInfo>, ShipError>
|
|
||||||
where
|
|
||||||
EG: EntityGateway + Clone + 'static,
|
|
||||||
{
|
|
||||||
Ok(futures::stream::iter(client_location.get_all_clients_by_client(id).await?.into_iter())
|
|
||||||
.filter_map(|area_client| {
|
|
||||||
let clients = clients.clone();
|
|
||||||
async move {
|
|
||||||
clients.with(area_client.client, |client| {
|
|
||||||
let mut teams = teams.clone();
|
|
||||||
Box::pin(async move {
|
|
||||||
let team = teams.get_team(area_client.client).await.ok()??;
|
|
||||||
Some(player_team_info(area_client.client, client, &team))
|
|
||||||
})}).await.ok()?
|
|
||||||
}})
|
|
||||||
.collect::<Vec<_>>()
|
|
||||||
.await)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn team_info<EG>(id: ClientId,
|
|
||||||
client_location: &ClientLocation,
|
|
||||||
clients: &Clients,
|
|
||||||
teams: &Teams<EG>,
|
|
||||||
) -> Result<TeamInfo, ShipError>
|
|
||||||
where
|
|
||||||
EG: EntityGateway + Clone + 'static,
|
|
||||||
{
|
|
||||||
Ok(TeamInfo {
|
|
||||||
clients: player_team_info_list(id, client_location, clients, teams).await?,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn lobby_team_list<EG>(id: ClientId,
|
|
||||||
client_location: &ClientLocation,
|
|
||||||
clients: &Clients,
|
|
||||||
teams: &Teams<EG>,
|
|
||||||
) -> Result<TeamLobbyList, ShipError>
|
|
||||||
where
|
|
||||||
EG: EntityGateway + Clone + 'static,
|
|
||||||
{
|
|
||||||
Ok(TeamLobbyList {
|
|
||||||
clients: player_team_info_list(id, client_location, clients, teams).await?,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn team_invitation_info(client_id: ClientId, client: &ClientState, team: &TeamEntity) -> TeamInvitationInfo {
|
|
||||||
TeamInvitationInfo {
|
|
||||||
guildcard: client.user.guildcard(),
|
|
||||||
team_id: team.id.0,
|
|
||||||
unknown: [0; 2],
|
|
||||||
team_name: libpso::utf8_to_utf16_array!(team.name, 14),
|
|
||||||
unknown2: 0x00986C84, // TODO: what if we omit this?
|
|
||||||
team_flag: team.team_flag,
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,18 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "quests"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
maps = { workspace = true }
|
|
||||||
|
|
||||||
libpso = { workspace = true }
|
|
||||||
|
|
||||||
async-std = { workspace = true }
|
|
||||||
ages-prs = { workspace = true }
|
|
||||||
log = { workspace = true }
|
|
||||||
byteorder = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
toml = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
@ -1,17 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "room"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
maps = { workspace = true }
|
|
||||||
entity = { workspace = true }
|
|
||||||
quests = { workspace = true }
|
|
||||||
location = { workspace = true }
|
|
||||||
drops = { workspace = true }
|
|
||||||
|
|
||||||
rand = { workspace = true }
|
|
||||||
async-std = { workspace = true }
|
|
||||||
futures = { workspace = true }
|
|
||||||
anyhow = { workspace = true }
|
|
||||||
thiserror = { workspace = true }
|
|
@ -1,9 +1,10 @@
|
|||||||
use libpso::character::character;
|
use libpso::character::character;
|
||||||
use stats::leveltable::CharacterStats;
|
use crate::common::leveltable::CharacterStats;
|
||||||
use entity::character::CharacterEntity;
|
use crate::entity::character::CharacterEntity;
|
||||||
use items::bank::BankState;
|
//use crate::ship::items::{CharacterInventory, CharacterBank};
|
||||||
use items::inventory::InventoryState;
|
use crate::ship::items::bank::BankState;
|
||||||
use entity::item::Meseta;
|
use crate::ship::items::inventory::InventoryState;
|
||||||
|
use crate::entity::item::Meseta;
|
||||||
|
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
@ -53,7 +54,7 @@ impl<'a> CharacterBytesBuilder<'a> {
|
|||||||
let level = self.level.unwrap();
|
let level = self.level.unwrap();
|
||||||
let meseta = self.meseta.unwrap();
|
let meseta = self.meseta.unwrap();
|
||||||
character::Character {
|
character::Character {
|
||||||
name: libpso::util::utf8_to_utf16_array(&character.name),
|
name: libpso::utf8_to_utf16_array!(character.name, 16),
|
||||||
hp: stats.hp,
|
hp: stats.hp,
|
||||||
atp: stats.atp + character.materials.power as u16 * 2,
|
atp: stats.atp + character.materials.power as u16 * 2,
|
||||||
mst: stats.mst + character.materials.mind as u16 * 2,
|
mst: stats.mst + character.materials.mind as u16 * 2,
|
@ -1,11 +1,11 @@
|
|||||||
use libpso::packet::ship::PlayerChat;
|
use libpso::packet::ship::PlayerChat;
|
||||||
use entity::gateway::EntityGateway;
|
use crate::entity::gateway::EntityGateway;
|
||||||
use networking::serverstate::ClientId;
|
use crate::common::serverstate::ClientId;
|
||||||
use crate::{ShipServerState, SendShipPacket};
|
use crate::ship::ship::{ShipServerState, SendShipPacket};
|
||||||
use client::Clients;
|
use crate::ship::client::Clients;
|
||||||
use items::state::ItemState;
|
use crate::ship::items::state::ItemState;
|
||||||
use entity::item::{BankName, BankIdentifier};
|
use crate::entity::item::{BankName, BankIdentifier};
|
||||||
use pktbuilder::message::bank_item_list;
|
use crate::ship::packet::builder::message::bank_item_list;
|
||||||
|
|
||||||
async fn default_bank<'a, EG>(id: ClientId,
|
async fn default_bank<'a, EG>(id: ClientId,
|
||||||
entity_gateway: &mut EG,
|
entity_gateway: &mut EG,
|
@ -6,20 +6,15 @@ use futures::future::BoxFuture;
|
|||||||
use libpso::packet::ship::*;
|
use libpso::packet::ship::*;
|
||||||
use libpso::packet::login::Session;
|
use libpso::packet::login::Session;
|
||||||
|
|
||||||
use networking::serverstate::ClientId;
|
use crate::common::serverstate::ClientId;
|
||||||
use entity::account::{UserAccountEntity, UserSettingsEntity};
|
use crate::entity::account::{UserAccountEntity, UserSettingsEntity};
|
||||||
use entity::character::CharacterEntity;
|
use crate::entity::character::CharacterEntity;
|
||||||
use entity::item;
|
use crate::entity::item;
|
||||||
|
|
||||||
use maps::area::MapArea;
|
use crate::ship::ship::ShipError;
|
||||||
use shops::{WeaponShopItem, ToolShopItem, ArmorShopItem};
|
use crate::ship::items;
|
||||||
|
use crate::ship::map::MapArea;
|
||||||
|
use crate::ship::shops::{WeaponShopItem, ToolShopItem, ArmorShopItem};
|
||||||
#[derive(thiserror::Error, Debug)]
|
|
||||||
pub enum ClientError {
|
|
||||||
#[error("not found {0}")]
|
|
||||||
NotFound(ClientId),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Default)]
|
#[derive(Clone, Default)]
|
||||||
@ -51,7 +46,7 @@ impl Clients {
|
|||||||
.await;
|
.await;
|
||||||
let client = clients
|
let client = clients
|
||||||
.get(&client_id)
|
.get(&client_id)
|
||||||
.ok_or(ClientError::NotFound(client_id))?
|
.ok_or_else(|| ShipError::ClientNotFound(client_id))?
|
||||||
.read()
|
.read()
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
@ -74,14 +69,17 @@ impl Clients {
|
|||||||
for (cindex, client_id) in client_ids.iter().enumerate() {
|
for (cindex, client_id) in client_ids.iter().enumerate() {
|
||||||
let c = clients
|
let c = clients
|
||||||
.get(client_id)
|
.get(client_id)
|
||||||
.ok_or(ClientError::NotFound(*client_id))?
|
.ok_or_else(|| ShipError::ClientNotFound(*client_id))?
|
||||||
.read()
|
.read()
|
||||||
.await;
|
.await;
|
||||||
client_states[cindex].write(c);
|
client_states[cindex].write(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
let client_states = unsafe {
|
let client_states = unsafe {
|
||||||
std::mem::transmute_copy(&client_states)
|
// TODO: this should just be a normal transmute but due to compiler limitations it
|
||||||
|
// does not yet work with const generics
|
||||||
|
// https://github.com/rust-lang/rust/issues/61956
|
||||||
|
std::mem::transmute_copy::<_, [RwLockReadGuard<ClientState>; N]>(&client_states)
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(func(client_states).await)
|
Ok(func(client_states).await)
|
||||||
@ -97,7 +95,7 @@ impl Clients {
|
|||||||
.await;
|
.await;
|
||||||
let mut client = clients
|
let mut client = clients
|
||||||
.get(&client_id)
|
.get(&client_id)
|
||||||
.ok_or(ClientError::NotFound(client_id))?
|
.ok_or_else(|| ShipError::ClientNotFound(client_id))?
|
||||||
.write()
|
.write()
|
||||||
.await;
|
.await;
|
||||||
|
|
@ -2,17 +2,18 @@
|
|||||||
use rand::{Rng};
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use entity::character::SectionID;
|
use crate::entity::character::SectionID;
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use maps::object::{MapObject, MapObjectType, FixedBoxDropType};
|
use crate::ship::map::{MapObject, MapObjectType, FixedBoxDropType};
|
||||||
use crate::rare_drop_table::{RareDropTable, RareDropItem};
|
use crate::ship::drops::rare_drop_table::{RareDropTable, RareDropItem};
|
||||||
use crate::generic_weapon::GenericWeaponTable;
|
use crate::ship::drops::generic_weapon::GenericWeaponTable;
|
||||||
use crate::generic_armor::GenericArmorTable;
|
use crate::ship::drops::generic_armor::GenericArmorTable;
|
||||||
use crate::generic_shield::GenericShieldTable;
|
use crate::ship::drops::generic_shield::GenericShieldTable;
|
||||||
use crate::generic_unit::GenericUnitTable;
|
use crate::ship::drops::generic_unit::GenericUnitTable;
|
||||||
use crate::tool_table::ToolTable;
|
use crate::ship::drops::tool_table::ToolTable;
|
||||||
|
use crate::entity::item::ItemDetail;
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
struct BoxDropRate {
|
struct BoxDropRate {
|
||||||
@ -203,7 +204,7 @@ impl BoxDropTable {
|
|||||||
FixedBoxDropType::Specific(value) => {
|
FixedBoxDropType::Specific(value) => {
|
||||||
let mut buf: [u8; 16] = [0; 16];
|
let mut buf: [u8; 16] = [0; 16];
|
||||||
buf[0..4].copy_from_slice(&u32::to_be_bytes(value));
|
buf[0..4].copy_from_slice(&u32::to_be_bytes(value));
|
||||||
ItemDropType::parse_item_from_bytes(buf)
|
ItemDetail::parse_item_from_bytes(buf)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,14 +1,14 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::Rng;
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
|
|
||||||
use entity::character::SectionID;
|
use crate::entity::item::armor::{ArmorType, Armor};
|
||||||
use entity::item::armor::{ArmorType, Armor};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::map::MapArea;
|
||||||
use maps::area::MapArea;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use stats::items::{armor_stats, ArmorStats};
|
use crate::ship::item_stats::{armor_stats, ArmorStats};
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
@ -1,14 +1,14 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::Rng;
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
|
|
||||||
use entity::item::shield::{ShieldType, Shield};
|
use crate::entity::item::shield::{ShieldType, Shield};
|
||||||
use entity::character::SectionID;
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::map::MapArea;
|
||||||
use maps::area::MapArea;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use stats::items::{shield_stats, ShieldStats};
|
use crate::ship::item_stats::{shield_stats, ShieldStats};
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
@ -1,14 +1,14 @@
|
|||||||
use std::collections::BTreeMap;
|
use std::collections::BTreeMap;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::Rng;
|
use rand::{Rng};
|
||||||
use rand::seq::IteratorRandom;
|
use rand::seq::IteratorRandom;
|
||||||
|
|
||||||
use entity::character::SectionID;
|
use crate::entity::item::unit::{UnitType, Unit, UnitModifier};
|
||||||
use entity::item::unit::{UnitType, Unit, UnitModifier};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::map::MapArea;
|
||||||
use maps::area::MapArea;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use stats::items::{unit_stats, UnitStats};
|
use crate::ship::item_stats::{unit_stats, UnitStats};
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -1,14 +1,14 @@
|
|||||||
use std::collections::{HashMap, BTreeMap};
|
use std::collections::{HashMap, BTreeMap};
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::Rng;
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
use rand::seq::SliceRandom;
|
use rand::seq::SliceRandom;
|
||||||
|
|
||||||
use entity::character::SectionID;
|
use crate::entity::item::weapon::{Weapon, WeaponType, Attribute, WeaponAttribute, WeaponSpecial};
|
||||||
use entity::item::weapon::{Weapon, WeaponType, Attribute, WeaponAttribute, WeaponSpecial};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::map::MapArea;
|
||||||
use maps::area::MapArea;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -5,6 +5,7 @@
|
|||||||
// to their drops
|
// to their drops
|
||||||
|
|
||||||
|
|
||||||
|
mod drop_table;
|
||||||
pub mod rare_drop_table;
|
pub mod rare_drop_table;
|
||||||
mod generic_weapon;
|
mod generic_weapon;
|
||||||
mod generic_armor;
|
mod generic_armor;
|
||||||
@ -21,19 +22,19 @@ use std::io::Read;
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::{Rng, SeedableRng};
|
use rand::{Rng, SeedableRng};
|
||||||
|
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::monster::MonsterType;
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use entity::character::SectionID;
|
use crate::entity::character::SectionID;
|
||||||
use crate::generic_weapon::GenericWeaponTable;
|
use crate::ship::drops::generic_weapon::GenericWeaponTable;
|
||||||
use crate::generic_armor::GenericArmorTable;
|
use crate::ship::drops::generic_armor::GenericArmorTable;
|
||||||
use crate::generic_shield::GenericShieldTable;
|
use crate::ship::drops::generic_shield::GenericShieldTable;
|
||||||
use crate::generic_unit::GenericUnitTable;
|
use crate::ship::drops::generic_unit::GenericUnitTable;
|
||||||
use crate::tool_table::ToolTable;
|
use crate::ship::drops::tool_table::ToolTable;
|
||||||
use crate::rare_drop_table::RareDropTable;
|
use crate::ship::drops::rare_drop_table::RareDropTable;
|
||||||
use crate::box_drop_table::BoxDropTable;
|
use crate::ship::drops::box_drop_table::BoxDropTable;
|
||||||
use maps::object::MapObject;
|
use crate::ship::map::MapObject;
|
||||||
use entity::item::{ItemType, weapon, armor, shield, unit, mag, tool, tech, esweapon};
|
use crate::entity::item::{weapon, armor, shield, unit, mag, tool, tech};
|
||||||
|
|
||||||
|
|
||||||
fn data_file_path(episode: Episode, difficulty: Difficulty, section_id: SectionID, filename: &str) -> PathBuf {
|
fn data_file_path(episode: Episode, difficulty: Difficulty, section_id: SectionID, filename: &str) -> PathBuf {
|
||||||
@ -54,6 +55,18 @@ pub fn load_data_file<T: serde::de::DeserializeOwned>(episode: Episode, difficul
|
|||||||
toml::from_str::<T>(s.as_str()).unwrap()
|
toml::from_str::<T>(s.as_str()).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// this is just copypaste
|
||||||
|
pub fn load_rare_monster_file<T: serde::de::DeserializeOwned>(episode: Episode) -> T {
|
||||||
|
// TODO: where does the rare monster toml file actually live
|
||||||
|
let mut path = PathBuf::from("data/battle_param/");
|
||||||
|
path.push(episode.to_string().to_lowercase() + "_rare_monster.toml");
|
||||||
|
|
||||||
|
let mut f = File::open(path).unwrap();
|
||||||
|
let mut s = String::new();
|
||||||
|
f.read_to_string(&mut s);
|
||||||
|
toml::from_str::<T>(s.as_str()).unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize, Copy, Clone)]
|
#[derive(Debug, Serialize, Deserialize, Copy, Clone)]
|
||||||
pub enum MonsterDropType {
|
pub enum MonsterDropType {
|
||||||
#[serde(rename = "weapon")]
|
#[serde(rename = "weapon")]
|
||||||
@ -89,28 +102,6 @@ pub enum ItemDropType {
|
|||||||
Meseta(u32),
|
Meseta(u32),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ItemDropType {
|
|
||||||
pub fn parse_item_from_bytes(data: [u8; 16]) -> Option<ItemDropType> {
|
|
||||||
let item_type = weapon::WeaponType::parse_type([data[0],data[1],data[2]]).map(ItemType::Weapon)
|
|
||||||
.or_else(|_| armor::ArmorType::parse_type([data[0],data[1],data[2]]).map(ItemType::Armor))
|
|
||||||
.or_else(|_| shield::ShieldType::parse_type([data[0],data[1],data[2]]).map(ItemType::Shield))
|
|
||||||
.or_else(|_| unit::UnitType::parse_type([data[0],data[1],data[2]]).map(ItemType::Unit))
|
|
||||||
.or_else(|_| mag::MagType::parse_type([data[0],data[1],data[2]]).map(ItemType::Mag))
|
|
||||||
.or_else(|_| tool::ToolType::parse_type([data[0],data[1],data[2]]).map(ItemType::Tool))
|
|
||||||
.or_else(|_| esweapon::ESWeaponType::parse_type([data[0],data[1],data[2]]).map(ItemType::ESWeapon)).ok()?;
|
|
||||||
|
|
||||||
match item_type {
|
|
||||||
ItemType::Weapon(_w) => Some(ItemDropType::Weapon(weapon::Weapon::from_bytes(data).ok()?)),
|
|
||||||
ItemType::Armor(_a) => Some(ItemDropType::Armor(armor::Armor::from_bytes(data).ok()?)),
|
|
||||||
ItemType::Shield(_s) => Some(ItemDropType::Shield(shield::Shield::from_bytes(data).ok()?)),
|
|
||||||
ItemType::Unit(_u) => Some(ItemDropType::Unit(unit::Unit::from_bytes(data).ok()?)),
|
|
||||||
ItemType::Mag(_m) => Some(ItemDropType::Mag(mag::Mag::from_bytes(data).ok()?)),
|
|
||||||
ItemType::Tool(_t) => Some(ItemDropType::Tool(tool::Tool::from_bytes(data).ok()?)),
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct ItemDrop {
|
pub struct ItemDrop {
|
||||||
pub map_area: MapArea,
|
pub map_area: MapArea,
|
||||||
@ -121,12 +112,7 @@ pub struct ItemDrop {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
pub trait DropTable {
|
pub struct DropTable {
|
||||||
fn get_drop(&mut self, map_area: &MapArea, monster: &MonsterType) -> Option<ItemDropType>;
|
|
||||||
fn get_box_drop(&mut self, map_area: &MapArea, object: &MapObject) -> Option<ItemDropType>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct StandardDropTable {
|
|
||||||
monster_stats: HashMap<MonsterType, MonsterDropStats>,
|
monster_stats: HashMap<MonsterType, MonsterDropStats>,
|
||||||
rare_table: RareDropTable,
|
rare_table: RareDropTable,
|
||||||
weapon_table: GenericWeaponTable,
|
weapon_table: GenericWeaponTable,
|
||||||
@ -138,12 +124,11 @@ pub struct StandardDropTable {
|
|||||||
rng: rand_chacha::ChaCha20Rng,
|
rng: rand_chacha::ChaCha20Rng,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl StandardDropTable {
|
impl DropTable {
|
||||||
#[allow(clippy::new_ret_no_self)]
|
pub fn new(episode: Episode, difficulty: Difficulty, section_id: SectionID) -> DropTable {
|
||||||
pub fn new(episode: Episode, difficulty: Difficulty, section_id: SectionID) -> Box<dyn DropTable + Send + Sync> {
|
|
||||||
let monster_stats: HashMap<String, MonsterDropStats> = load_data_file(episode, difficulty, section_id, "monster_dar.toml");
|
let monster_stats: HashMap<String, MonsterDropStats> = load_data_file(episode, difficulty, section_id, "monster_dar.toml");
|
||||||
|
|
||||||
Box::new(StandardDropTable {
|
DropTable {
|
||||||
monster_stats: monster_stats.into_iter().map(|(m, s)| (m.parse().unwrap(), s)).collect(),
|
monster_stats: monster_stats.into_iter().map(|(m, s)| (m.parse().unwrap(), s)).collect(),
|
||||||
rare_table: RareDropTable::new(episode, difficulty, section_id),
|
rare_table: RareDropTable::new(episode, difficulty, section_id),
|
||||||
weapon_table: GenericWeaponTable::new(episode, difficulty, section_id),
|
weapon_table: GenericWeaponTable::new(episode, difficulty, section_id),
|
||||||
@ -153,7 +138,7 @@ impl StandardDropTable {
|
|||||||
tool_table: ToolTable::new(episode, difficulty, section_id),
|
tool_table: ToolTable::new(episode, difficulty, section_id),
|
||||||
box_table: BoxDropTable::new(episode, difficulty, section_id),
|
box_table: BoxDropTable::new(episode, difficulty, section_id),
|
||||||
rng: rand_chacha::ChaCha20Rng::from_entropy(),
|
rng: rand_chacha::ChaCha20Rng::from_entropy(),
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn builder() -> DropTableBuilder {
|
pub fn builder() -> DropTableBuilder {
|
||||||
@ -183,10 +168,8 @@ impl StandardDropTable {
|
|||||||
MonsterDropType::None => None,
|
MonsterDropType::None => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
impl DropTable for StandardDropTable {
|
pub fn get_drop(&mut self, map_area: &MapArea, monster: &MonsterType) -> Option<ItemDropType> {
|
||||||
fn get_drop(&mut self, map_area: &MapArea, monster: &MonsterType) -> Option<ItemDropType> {
|
|
||||||
let monster_stat = *self.monster_stats.get(monster)?;
|
let monster_stat = *self.monster_stats.get(monster)?;
|
||||||
|
|
||||||
let drop_anything = self.rng.gen_range(0, 100);
|
let drop_anything = self.rng.gen_range(0, 100);
|
||||||
@ -214,7 +197,7 @@ impl DropTable for StandardDropTable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_box_drop(&mut self, map_area: &MapArea, object: &MapObject) -> Option<ItemDropType> {
|
pub fn get_box_drop(&mut self, map_area: &MapArea, object: &MapObject) -> Option<ItemDropType> {
|
||||||
self.box_table.get_drop(map_area, object, &mut self.rng)
|
self.box_table.get_drop(map_area, object, &mut self.rng)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -261,8 +244,8 @@ impl DropTableBuilder {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn build(self, episode: Episode, difficulty: Difficulty, section_id: SectionID) -> Box<dyn DropTable + Send + Sync> {
|
pub fn build(self, episode: Episode, difficulty: Difficulty, section_id: SectionID) -> DropTable {
|
||||||
Box::new(StandardDropTable {
|
DropTable {
|
||||||
monster_stats: self.monster_stats.unwrap_or_else(|| {
|
monster_stats: self.monster_stats.unwrap_or_else(|| {
|
||||||
let monster_stats: HashMap<String, MonsterDropStats> = load_data_file(episode, difficulty, section_id, "monster_dar.toml");
|
let monster_stats: HashMap<String, MonsterDropStats> = load_data_file(episode, difficulty, section_id, "monster_dar.toml");
|
||||||
monster_stats.into_iter().map(|(m, s)| (m.parse().unwrap(), s)).collect()
|
monster_stats.into_iter().map(|(m, s)| (m.parse().unwrap(), s)).collect()
|
||||||
@ -275,10 +258,11 @@ impl DropTableBuilder {
|
|||||||
tool_table: self.tool_table.unwrap_or_else(|| ToolTable::new(episode, difficulty, section_id)),
|
tool_table: self.tool_table.unwrap_or_else(|| ToolTable::new(episode, difficulty, section_id)),
|
||||||
box_table: self.box_table.unwrap_or_else(|| BoxDropTable::new(episode, difficulty, section_id)),
|
box_table: self.box_table.unwrap_or_else(|| BoxDropTable::new(episode, difficulty, section_id)),
|
||||||
rng: self.rng.unwrap_or_else(rand_chacha::ChaCha20Rng::from_entropy),
|
rng: self.rng.unwrap_or_else(rand_chacha::ChaCha20Rng::from_entropy),
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
@ -1,20 +1,20 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use rand::Rng;
|
use rand::Rng;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use entity::item::weapon::{Weapon, WeaponType};
|
use crate::entity::item::weapon::{Weapon, WeaponType};
|
||||||
use entity::item::armor::{Armor, ArmorType};
|
use crate::entity::item::armor::{Armor, ArmorType};
|
||||||
use entity::item::shield::{Shield, ShieldType};
|
use crate::entity::item::shield::{Shield, ShieldType};
|
||||||
use entity::item::unit::{Unit, UnitType};
|
use crate::entity::item::unit::{Unit, UnitType};
|
||||||
use entity::item::tool::{Tool, ToolType};
|
use crate::entity::item::tool::{Tool, ToolType};
|
||||||
use entity::item::mag::{Mag, MagType};
|
use crate::entity::item::mag::{Mag, MagType};
|
||||||
use entity::character::SectionID;
|
use crate::entity::character::SectionID;
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::monster::MonsterType;
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use crate::generic_weapon::AttributeTable;
|
use crate::ship::drops::generic_weapon::AttributeTable;
|
||||||
use crate::generic_armor::GenericArmorTable;
|
use crate::ship::drops::generic_armor::GenericArmorTable;
|
||||||
use crate::generic_shield::GenericShieldTable;
|
use crate::ship::drops::generic_shield::GenericShieldTable;
|
||||||
|
|
||||||
type ItemParseFn = Box<dyn Fn(&String) -> Option<RareDropItem>>;
|
type ItemParseFn = Box<dyn Fn(&String) -> Option<RareDropItem>>;
|
||||||
|
|
@ -3,11 +3,11 @@ use serde::{Serialize, Deserialize};
|
|||||||
use rand::{Rng};
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
|
|
||||||
use entity::item::tech::{Technique, TechniqueDisk};
|
use crate::entity::item::tech::{Technique, TechniqueDisk};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use entity::character::SectionID;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -1,14 +1,14 @@
|
|||||||
use std::collections::BTreeMap;
|
use std::collections::{BTreeMap};
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use rand::Rng;
|
use rand::{Rng};
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
|
|
||||||
use entity::item::tool::{Tool, ToolType};
|
use crate::entity::item::tool::{Tool, ToolType};
|
||||||
use maps::room::{Difficulty, Episode};
|
use crate::ship::room::{Difficulty, Episode};
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use entity::character::SectionID;
|
use crate::entity::character::SectionID;
|
||||||
use crate::{ItemDropType, load_data_file};
|
use crate::ship::drops::{ItemDropType, load_data_file};
|
||||||
use crate::tech_table::TechniqueTable;
|
use crate::ship::drops::tech_table::TechniqueTable;
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone, enum_utils::FromStr)]
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone, enum_utils::FromStr)]
|
@ -4,13 +4,13 @@ use serde::{Serialize, Deserialize};
|
|||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
|
||||||
use entity::item::weapon::WeaponType;
|
use crate::entity::item::weapon::WeaponType;
|
||||||
use entity::item::armor::ArmorType;
|
use crate::entity::item::armor::ArmorType;
|
||||||
use entity::item::shield::ShieldType;
|
use crate::entity::item::shield::ShieldType;
|
||||||
use entity::item::unit::UnitType;
|
use crate::entity::item::unit::UnitType;
|
||||||
use entity::item::mag::MagType;
|
use crate::entity::item::mag::MagType;
|
||||||
use entity::item::tool::ToolType;
|
use crate::entity::item::tool::ToolType;
|
||||||
use entity::item::tech::Technique;
|
use crate::entity::item::tech::Technique;
|
||||||
|
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
@ -1,6 +1,6 @@
|
|||||||
// TODO: replace various u32s and usizes denoting item amounts for ItemAmount(u32) for consistency
|
// TODO: replace various u32s and usizes denoting item amounts for ItemAmount(u32) for consistency
|
||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use entity::item::{Meseta, ItemNote};
|
use crate::entity::item::{Meseta, ItemNote};
|
||||||
use async_std::sync::Arc;
|
use async_std::sync::Arc;
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
use futures::future::BoxFuture;
|
use futures::future::BoxFuture;
|
||||||
@ -8,33 +8,30 @@ use std::pin::Pin;
|
|||||||
use std::iter::IntoIterator;
|
use std::iter::IntoIterator;
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
|
|
||||||
use entity::character::{CharacterEntity, CharacterEntityId};
|
use libpso::packet::{ship::Message, messages::GameMessage};
|
||||||
use entity::gateway::{EntityGateway, EntityGatewayTransaction};
|
use crate::entity::character::{CharacterEntity, CharacterEntityId};
|
||||||
use entity::item::{ItemDetail, NewItemEntity, TradeId, ItemModifier};
|
use crate::entity::gateway::{EntityGateway, EntityGatewayTransaction};
|
||||||
use entity::item::tool::Tool;
|
use crate::entity::item::{ItemDetail, NewItemEntity, TradeId, ItemModifier};
|
||||||
use entity::room::RoomEntityId;
|
use crate::entity::item::tool::Tool;
|
||||||
use maps::area::MapArea;
|
use crate::entity::room::RoomEntityId;
|
||||||
use crate::state::{ItemStateProxy, ItemStateError, AddItemResult, StackedItemDetail, IndividualItemDetail};
|
use crate::ship::map::MapArea;
|
||||||
use crate::bank::{BankItem, BankItemDetail};
|
use crate::ship::ship::SendShipPacket;
|
||||||
use crate::inventory::{InventoryItem, InventoryItemDetail};
|
use crate::ship::items::state::{ItemStateProxy, ItemStateError, AddItemResult, StackedItemDetail, IndividualItemDetail};
|
||||||
use crate::floor::{FloorItem, FloorItemDetail};
|
use crate::ship::items::bank::{BankItem, BankItemDetail};
|
||||||
use crate::apply_item::{apply_item, ApplyItemAction};
|
use crate::ship::items::inventory::{InventoryItem, InventoryItemDetail};
|
||||||
use shops::ShopItem;
|
use crate::ship::items::floor::{FloorItem, FloorItemDetail};
|
||||||
use drops::{ItemDrop, ItemDropType};
|
use crate::ship::items::apply_item::{apply_item, ApplyItemAction};
|
||||||
use location::AreaClient;
|
use crate::ship::shops::ShopItem;
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::drops::{ItemDrop, ItemDropType};
|
||||||
|
use crate::ship::packet::builder;
|
||||||
|
use crate::ship::location::AreaClient;
|
||||||
|
use crate::ship::monster::MonsterType;
|
||||||
|
|
||||||
pub enum TriggerCreateItem {
|
pub enum TriggerCreateItem {
|
||||||
Yes,
|
Yes,
|
||||||
No
|
No
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub enum CreateItem {
|
|
||||||
Individual(AreaClient, ClientItemId, IndividualItemDetail),
|
|
||||||
Stacked(AreaClient, ClientItemId, Tool, usize),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(super) fn take_item_from_floor<'a, EG, TR>(
|
pub(super) fn take_item_from_floor<'a, EG, TR>(
|
||||||
character_id: CharacterEntityId,
|
character_id: CharacterEntityId,
|
||||||
item_id: ClientItemId
|
item_id: ClientItemId
|
||||||
@ -1147,7 +1144,7 @@ pub(super) fn apply_item_action_packets<'a, EG, TR>(
|
|||||||
character_id: CharacterEntityId,
|
character_id: CharacterEntityId,
|
||||||
area_client: AreaClient,
|
area_client: AreaClient,
|
||||||
) -> impl Fn((ItemStateProxy, TR), ApplyItemAction)
|
) -> impl Fn((ItemStateProxy, TR), ApplyItemAction)
|
||||||
-> BoxFuture<'a, Result<((ItemStateProxy, TR), Vec<CreateItem>), anyhow::Error>>
|
-> BoxFuture<'a, Result<((ItemStateProxy, TR), Vec<SendShipPacket>), anyhow::Error>>
|
||||||
where
|
where
|
||||||
EG: EntityGateway,
|
EG: EntityGateway,
|
||||||
TR: EntityGatewayTransaction<ParentGateway = EG> + 'a,
|
TR: EntityGatewayTransaction<ParentGateway = EG> + 'a,
|
||||||
@ -1164,7 +1161,7 @@ where
|
|||||||
let (inventory_item_detail, create_item) = if item_detail.is_stackable() {
|
let (inventory_item_detail, create_item) = if item_detail.is_stackable() {
|
||||||
let tool = item_detail.as_tool().ok_or_else(|| ItemStateError::NotATool(ClientItemId(0xFFFFFFFF)))?;
|
let tool = item_detail.as_tool().ok_or_else(|| ItemStateError::NotATool(ClientItemId(0xFFFFFFFF)))?;
|
||||||
|
|
||||||
let create_item = CreateItem::Stacked(area_client, item_id, tool, 1);
|
let create_item = builder::message::create_stacked_item(area_client, item_id, &tool, 1).map_err(|_err| ItemStateError::Dummy)?;
|
||||||
let item_detail = StackedItemDetail {
|
let item_detail = StackedItemDetail {
|
||||||
entity_ids: vec![new_item.id],
|
entity_ids: vec![new_item.id],
|
||||||
tool
|
tool
|
||||||
@ -1176,7 +1173,7 @@ where
|
|||||||
entity_id: new_item.id,
|
entity_id: new_item.id,
|
||||||
item: item_detail,
|
item: item_detail,
|
||||||
};
|
};
|
||||||
let create_item = CreateItem::Individual(area_client, item_id, item_detail.clone());
|
let create_item = builder::message::create_individual_item(area_client, item_id, &item_detail).map_err(|_err| ItemStateError::Dummy)?;
|
||||||
(InventoryItemDetail::Individual(item_detail), create_item)
|
(InventoryItemDetail::Individual(item_detail), create_item)
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1190,8 +1187,7 @@ where
|
|||||||
transaction.gateway().set_character_inventory(&character_id, &inventory.as_inventory_entity(&character_id)).await?;
|
transaction.gateway().set_character_inventory(&character_id, &inventory.as_inventory_entity(&character_id)).await?;
|
||||||
item_state.set_inventory(inventory).await;
|
item_state.set_inventory(inventory).await;
|
||||||
|
|
||||||
//vec![SendShipPacket::Message(Message::new(GameMessage::CreateItem(create_item)))]
|
vec![SendShipPacket::Message(Message::new(GameMessage::CreateItem(create_item)))]
|
||||||
vec![create_item]
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
Vec::new()
|
Vec::new()
|
@ -4,15 +4,15 @@ use thiserror::Error;
|
|||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use rand::SeedableRng;
|
use rand::SeedableRng;
|
||||||
use rand::distributions::{WeightedIndex, Distribution};
|
use rand::distributions::{WeightedIndex, Distribution};
|
||||||
use entity::gateway::{EntityGateway, GatewayError};
|
use crate::entity::gateway::{EntityGateway, GatewayError};
|
||||||
use entity::character::{CharacterEntity, TechLevel};
|
use crate::entity::character::{CharacterEntity, TechLevel};
|
||||||
use entity::item::mag::{MagCell, MagCellError};
|
use crate::entity::item::mag::{MagCell, MagCellError};
|
||||||
use entity::item::tool::{Tool, ToolType};
|
use crate::entity::item::tool::{Tool, ToolType};
|
||||||
use entity::item::tech::TechniqueDisk;
|
use crate::entity::item::tech::TechniqueDisk;
|
||||||
use entity::item::{ItemDetail, ItemEntityId};
|
use crate::entity::item::{ItemDetail, ItemEntityId};
|
||||||
use entity::item::weapon::WeaponModifier;
|
use crate::entity::item::weapon::WeaponModifier;
|
||||||
use crate::state::ItemStateProxy;
|
use crate::ship::items::state::ItemStateProxy;
|
||||||
use crate::inventory::InventoryItemDetail;
|
use crate::ship::items::inventory::InventoryItemDetail;
|
||||||
|
|
||||||
|
|
||||||
#[derive(Error, Debug)]
|
#[derive(Error, Debug)]
|
@ -1,15 +1,15 @@
|
|||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use libpso::character::character;
|
use libpso::character::character;
|
||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use entity::item::{Meseta, ItemEntityId, ItemDetail, ItemEntity, BankEntity, BankItemEntity};
|
use crate::entity::item::{Meseta, ItemEntityId, ItemDetail, ItemEntity, BankEntity, BankItemEntity};
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
use async_std::sync::{Arc, Mutex};
|
use async_std::sync::{Arc, Mutex};
|
||||||
|
|
||||||
use entity::character::CharacterEntityId;
|
use crate::entity::character::CharacterEntityId;
|
||||||
use entity::item::BankIdentifier;
|
use crate::entity::item::BankIdentifier;
|
||||||
use crate::state::ItemStateError;
|
use crate::ship::items::state::ItemStateError;
|
||||||
use crate::state::{IndividualItemDetail, StackedItemDetail, AddItemResult};
|
use crate::ship::items::state::{IndividualItemDetail, StackedItemDetail, AddItemResult};
|
||||||
use crate::inventory::{InventoryItem, InventoryItemDetail};
|
use crate::ship::items::inventory::{InventoryItem, InventoryItemDetail};
|
||||||
|
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
@ -323,9 +323,18 @@ impl std::cmp::PartialEq for BankItemDetail {
|
|||||||
|
|
||||||
impl std::cmp::Eq for BankItemDetail {}
|
impl std::cmp::Eq for BankItemDetail {}
|
||||||
|
|
||||||
|
#[allow(clippy::non_canonical_partial_ord_impl)]
|
||||||
impl std::cmp::PartialOrd for BankItemDetail {
|
impl std::cmp::PartialOrd for BankItemDetail {
|
||||||
fn partial_cmp(&self, other: &BankItemDetail) -> Option<std::cmp::Ordering> {
|
fn partial_cmp(&self, other: &BankItemDetail) -> Option<std::cmp::Ordering> {
|
||||||
Some(self.cmp(other))
|
let mut self_bytes = [0u8; 4];
|
||||||
|
let mut other_bytes = [0u8; 4];
|
||||||
|
self_bytes.copy_from_slice(&self.as_client_bytes()[0..4]);
|
||||||
|
other_bytes.copy_from_slice(&other.as_client_bytes()[0..4]);
|
||||||
|
|
||||||
|
let self_value = u32::from_be_bytes(self_bytes);
|
||||||
|
let other_value = u32::from_be_bytes(other_bytes);
|
||||||
|
|
||||||
|
self_value.partial_cmp(&other_value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -336,8 +345,8 @@ impl std::cmp::Ord for BankItemDetail {
|
|||||||
self_bytes.copy_from_slice(&self.as_client_bytes()[0..4]);
|
self_bytes.copy_from_slice(&self.as_client_bytes()[0..4]);
|
||||||
other_bytes.copy_from_slice(&other.as_client_bytes()[0..4]);
|
other_bytes.copy_from_slice(&other.as_client_bytes()[0..4]);
|
||||||
|
|
||||||
let self_value = u32::from_be_bytes(self_bytes);
|
let self_value = u32::from_le_bytes(self_bytes);
|
||||||
let other_value = u32::from_be_bytes(other_bytes);
|
let other_value = u32::from_le_bytes(other_bytes);
|
||||||
|
|
||||||
self_value.cmp(&other_value)
|
self_value.cmp(&other_value)
|
||||||
}
|
}
|
||||||
@ -352,9 +361,10 @@ impl std::cmp::PartialEq for BankItem {
|
|||||||
|
|
||||||
impl std::cmp::Eq for BankItem {}
|
impl std::cmp::Eq for BankItem {}
|
||||||
|
|
||||||
|
#[allow(clippy::non_canonical_partial_ord_impl)]
|
||||||
impl std::cmp::PartialOrd for BankItem {
|
impl std::cmp::PartialOrd for BankItem {
|
||||||
fn partial_cmp(&self, other: &BankItem) -> Option<std::cmp::Ordering> {
|
fn partial_cmp(&self, other: &BankItem) -> Option<std::cmp::Ordering> {
|
||||||
Some(self.cmp(other))
|
self.item.partial_cmp(&other.item)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -1,14 +1,14 @@
|
|||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use entity::item::{Meseta, ItemEntityId, ItemDetail};
|
use crate::entity::item::{Meseta, ItemEntityId, ItemDetail};
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
|
|
||||||
use maps::area::MapArea;
|
use crate::ship::map::MapArea;
|
||||||
use entity::character::CharacterEntityId;
|
use crate::entity::character::CharacterEntityId;
|
||||||
use entity::item::mag::Mag;
|
use crate::entity::item::mag::Mag;
|
||||||
|
|
||||||
use crate::state::ItemStateError;
|
use crate::ship::items::state::ItemStateError;
|
||||||
use crate::state::{IndividualItemDetail, StackedItemDetail};
|
use crate::ship::items::state::{IndividualItemDetail, StackedItemDetail};
|
||||||
use crate::inventory::{InventoryItem, InventoryItemDetail};
|
use crate::ship::items::inventory::{InventoryItem, InventoryItemDetail};
|
||||||
|
|
||||||
pub enum FloorType {
|
pub enum FloorType {
|
||||||
Local,
|
Local,
|
@ -1,18 +1,18 @@
|
|||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use libpso::character::character;
|
use libpso::character::character;
|
||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use entity::item::{Meseta, ItemEntityId, ItemDetail, ItemEntity, InventoryEntity, InventoryItemEntity, EquippedEntity};
|
use crate::entity::item::{Meseta, ItemEntityId, ItemDetail, ItemEntity, InventoryEntity, InventoryItemEntity, EquippedEntity};
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
use async_std::sync::{Arc, Mutex};
|
use async_std::sync::{Arc, Mutex};
|
||||||
|
|
||||||
use entity::character::CharacterEntityId;
|
use crate::entity::character::CharacterEntityId;
|
||||||
use entity::item::tool::ToolType;
|
use crate::entity::item::tool::ToolType;
|
||||||
use entity::item::mag::Mag;
|
use crate::entity::item::mag::Mag;
|
||||||
use entity::item::weapon::Weapon;
|
use crate::entity::item::weapon::Weapon;
|
||||||
use shops::{ShopItem, ArmorShopItem, ToolShopItem, WeaponShopItem};
|
use crate::ship::shops::{ShopItem, ArmorShopItem, ToolShopItem, WeaponShopItem};
|
||||||
use crate::state::ItemStateError;
|
use crate::ship::items::state::ItemStateError;
|
||||||
use crate::state::{IndividualItemDetail, StackedItemDetail, AddItemResult};
|
use crate::ship::items::state::{IndividualItemDetail, StackedItemDetail, AddItemResult};
|
||||||
use crate::floor::{FloorItem, FloorItemDetail};
|
use crate::ship::items::floor::{FloorItem, FloorItemDetail};
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub enum InventoryItemDetail {
|
pub enum InventoryItemDetail {
|
1380
src/ship/items/manager.rs
Normal file
1380
src/ship/items/manager.rs
Normal file
File diff suppressed because it is too large
Load Diff
@ -1,5 +1,3 @@
|
|||||||
#![feature(extract_if)]
|
|
||||||
|
|
||||||
pub mod state;
|
pub mod state;
|
||||||
pub mod actions;
|
pub mod actions;
|
||||||
pub mod apply_item;
|
pub mod apply_item;
|
||||||
@ -8,7 +6,6 @@ pub mod inventory;
|
|||||||
pub mod floor;
|
pub mod floor;
|
||||||
pub mod bank;
|
pub mod bank;
|
||||||
pub mod tasks;
|
pub mod tasks;
|
||||||
pub mod trade;
|
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq, serde::Serialize, serde::Deserialize, derive_more::Display)]
|
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq, serde::Serialize, serde::Deserialize, derive_more::Display)]
|
||||||
pub struct ClientItemId(pub u32);
|
pub struct ClientItemId(pub u32);
|
@ -4,18 +4,18 @@ use async_std::sync::{Arc, RwLock, Mutex};
|
|||||||
use futures::stream::{FuturesOrdered, StreamExt};
|
use futures::stream::{FuturesOrdered, StreamExt};
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
|
|
||||||
use entity::gateway::{EntityGateway, GatewayError};
|
use crate::entity::gateway::{EntityGateway, GatewayError};
|
||||||
use entity::character::{CharacterEntity, CharacterEntityId};
|
use crate::entity::character::{CharacterEntity, CharacterEntityId};
|
||||||
use entity::item::{ItemEntityId, ItemDetail, ItemEntity, InventoryItemEntity, BankItemEntity, BankIdentifier};
|
use crate::entity::item::{ItemEntityId, ItemDetail, ItemEntity, InventoryItemEntity, BankItemEntity, BankIdentifier};
|
||||||
use entity::item::tool::Tool;
|
use crate::entity::item::tool::Tool;
|
||||||
use entity::item::weapon::Weapon;
|
use crate::entity::item::weapon::Weapon;
|
||||||
use entity::item::mag::Mag;
|
use crate::entity::item::mag::Mag;
|
||||||
use drops::ItemDrop;
|
use crate::ship::drops::ItemDrop;
|
||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use crate::inventory::{Inventory, InventoryItem, InventoryItemDetail, InventoryError, InventoryState};
|
use crate::ship::items::inventory::{Inventory, InventoryItem, InventoryItemDetail, InventoryError, InventoryState};
|
||||||
use crate::floor::{FloorState, FloorItem, LocalFloor, SharedFloor, FloorType};
|
use crate::ship::items::floor::{FloorState, FloorItem, LocalFloor, SharedFloor, FloorType};
|
||||||
use crate::bank::{Bank, BankState, BankItem, BankItemDetail, BankError};
|
use crate::ship::items::bank::{Bank, BankState, BankItem, BankItemDetail, BankError};
|
||||||
use location::{AreaClient, RoomId};
|
use crate::ship::location::{AreaClient, RoomId};
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum ItemStateError {
|
pub enum ItemStateError {
|
||||||
@ -50,7 +50,7 @@ pub enum ItemStateError {
|
|||||||
#[error("stacked item")]
|
#[error("stacked item")]
|
||||||
StackedItemError(Vec<ItemEntity>),
|
StackedItemError(Vec<ItemEntity>),
|
||||||
#[error("apply item {0}")]
|
#[error("apply item {0}")]
|
||||||
ApplyItemError(#[from] crate::apply_item::ApplyItemError),
|
ApplyItemError(#[from] crate::ship::items::apply_item::ApplyItemError),
|
||||||
#[error("item is not a mag {0}")]
|
#[error("item is not a mag {0}")]
|
||||||
NotAMag(ClientItemId),
|
NotAMag(ClientItemId),
|
||||||
#[error("item is not mag food {0}")]
|
#[error("item is not mag food {0}")]
|
@ -1,23 +1,24 @@
|
|||||||
use futures::future::BoxFuture;
|
use futures::future::BoxFuture;
|
||||||
use crate::ClientItemId;
|
use crate::ship::items::ClientItemId;
|
||||||
use entity::item::Meseta;
|
use crate::entity::item::Meseta;
|
||||||
|
|
||||||
use maps::area::MapArea;
|
use crate::ship::ship::SendShipPacket;
|
||||||
use entity::character::{CharacterEntity, CharacterEntityId};
|
use crate::ship::map::MapArea;
|
||||||
use entity::gateway::{EntityGateway, EntityGatewayTransaction};
|
use crate::entity::character::{CharacterEntity, CharacterEntityId};
|
||||||
use entity::item::ItemModifier;
|
use crate::entity::gateway::{EntityGateway, EntityGatewayTransaction};
|
||||||
use entity::room::RoomEntityId;
|
use crate::entity::item::ItemModifier;
|
||||||
use crate::state::{ItemState, ItemStateProxy, IndividualItemDetail};
|
use crate::entity::room::RoomEntityId;
|
||||||
use crate::itemstateaction::{ItemStateAction, ItemAction};
|
use crate::ship::items::state::{ItemState, ItemStateProxy, IndividualItemDetail};
|
||||||
use crate::inventory::InventoryItem;
|
use crate::ship::items::itemstateaction::{ItemStateAction, ItemAction};
|
||||||
use crate::floor::FloorItem;
|
use crate::ship::items::inventory::InventoryItem;
|
||||||
use shops::ShopItem;
|
use crate::ship::items::floor::FloorItem;
|
||||||
use crate::trade::TradeItem;
|
use crate::ship::shops::ShopItem;
|
||||||
use location::AreaClient;
|
use crate::ship::trade::TradeItem;
|
||||||
use drops::ItemDrop;
|
use crate::ship::location::AreaClient;
|
||||||
use maps::monster::MonsterType;
|
use crate::ship::drops::ItemDrop;
|
||||||
|
use crate::ship::monster::MonsterType;
|
||||||
|
|
||||||
use crate::actions;
|
use crate::ship::items::actions;
|
||||||
|
|
||||||
pub fn pick_up_item<'a, EG>(
|
pub fn pick_up_item<'a, EG>(
|
||||||
item_state: &'a mut ItemState,
|
item_state: &'a mut ItemState,
|
||||||
@ -277,7 +278,7 @@ pub fn use_item<'a, EG> (
|
|||||||
area_client: AreaClient,
|
area_client: AreaClient,
|
||||||
item_id: &'a ClientItemId,
|
item_id: &'a ClientItemId,
|
||||||
amount: u32,
|
amount: u32,
|
||||||
) -> BoxFuture<'a, Result<Vec<actions::CreateItem>, anyhow::Error>>
|
) -> BoxFuture<'a, Result<Vec<SendShipPacket>, anyhow::Error>>
|
||||||
where
|
where
|
||||||
EG: EntityGateway + 'static,
|
EG: EntityGateway + 'static,
|
||||||
{
|
{
|
||||||
@ -371,8 +372,6 @@ where
|
|||||||
Ok((transaction, result))
|
Ok((transaction, result))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::type_complexity)]
|
|
||||||
pub fn trade_items<'a, EG> (
|
pub fn trade_items<'a, EG> (
|
||||||
item_state: &'a mut ItemState,
|
item_state: &'a mut ItemState,
|
||||||
entity_gateway: &'a mut EG,
|
entity_gateway: &'a mut EG,
|
@ -2,7 +2,7 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::time::SystemTime;
|
use std::time::SystemTime;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
use networking::serverstate::ClientId;
|
use crate::common::serverstate::ClientId;
|
||||||
|
|
||||||
use async_std::sync::{Arc, RwLock};
|
use async_std::sync::{Arc, RwLock};
|
||||||
use futures::{stream, StreamExt};
|
use futures::{stream, StreamExt};
|
@ -2,7 +2,7 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
use crate::room::Episode;
|
use crate::ship::room::Episode;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, PartialEq, Eq, Serialize, Deserialize)]
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, Serialize, Deserialize)]
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user