2020-05-09 15:41:25 +00:00
|
|
|
extern crate diesel;
|
|
|
|
|
|
|
|
use super::{
|
2020-04-20 08:44:29 +00:00
|
|
|
error::Error,
|
2020-05-09 15:41:25 +00:00
|
|
|
establish_connection,
|
2020-06-01 21:34:52 +00:00
|
|
|
models::{
|
|
|
|
Body, Character, Inventory, InventoryUpdate, NewCharacter, Stats, StatsJoinData,
|
|
|
|
StatsUpdate,
|
|
|
|
},
|
2020-04-20 08:44:29 +00:00
|
|
|
schema,
|
2020-05-09 15:41:25 +00:00
|
|
|
};
|
|
|
|
use crate::comp;
|
|
|
|
use common::character::{Character as CharacterData, CharacterItem, MAX_CHARACTERS_PER_PLAYER};
|
2020-06-01 21:34:52 +00:00
|
|
|
use crossbeam::channel;
|
2020-05-09 15:41:25 +00:00
|
|
|
use diesel::prelude::*;
|
|
|
|
|
|
|
|
type CharacterListResult = Result<Vec<CharacterItem>, Error>;
|
|
|
|
|
2020-05-11 10:06:53 +00:00
|
|
|
/// Load stored data for a character.
|
|
|
|
///
|
|
|
|
/// After first logging in, and after a character is selected, we fetch this
|
|
|
|
/// data for the purpose of inserting their persisted data for the entity.
|
2020-06-01 21:34:52 +00:00
|
|
|
pub fn load_character_data(
|
|
|
|
character_id: i32,
|
|
|
|
db_dir: &str,
|
|
|
|
) -> Result<(comp::Stats, comp::Inventory), Error> {
|
|
|
|
let connection = establish_connection(db_dir);
|
|
|
|
|
|
|
|
let (character_data, body_data, stats_data, maybe_inventory) =
|
|
|
|
schema::character::dsl::character
|
|
|
|
.filter(schema::character::id.eq(character_id))
|
|
|
|
.inner_join(schema::body::table)
|
|
|
|
.inner_join(schema::stats::table)
|
|
|
|
.left_join(schema::inventory::table)
|
|
|
|
.first::<(Character, Body, Stats, Option<Inventory>)>(&connection)?;
|
2020-05-11 10:06:53 +00:00
|
|
|
|
2020-06-01 21:34:52 +00:00
|
|
|
Ok((
|
|
|
|
comp::Stats::from(StatsJoinData {
|
|
|
|
alias: &character_data.alias,
|
|
|
|
body: &comp::Body::from(&body_data),
|
|
|
|
stats: &stats_data,
|
|
|
|
}),
|
|
|
|
maybe_inventory.map_or_else(
|
|
|
|
|| {
|
|
|
|
// If no inventory record was found for the character, create it now
|
|
|
|
let row = Inventory::from((character_data.id, comp::Inventory::default()));
|
|
|
|
|
|
|
|
if let Err(error) = diesel::insert_into(schema::inventory::table)
|
|
|
|
.values(&row)
|
|
|
|
.execute(&connection)
|
|
|
|
{
|
|
|
|
log::warn!(
|
|
|
|
"Failed to create an inventory record for character {}: {}",
|
|
|
|
&character_data.id,
|
|
|
|
error
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
comp::Inventory::default()
|
|
|
|
},
|
|
|
|
|inv| comp::Inventory::from(inv),
|
|
|
|
),
|
|
|
|
))
|
2020-05-11 10:06:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Loads a list of characters belonging to the player. This data is a small
|
|
|
|
/// subset of the character's data, and is used to render the character and
|
|
|
|
/// their level in the character list.
|
|
|
|
///
|
|
|
|
/// In the event that a join fails, for a character (i.e. they lack an entry for
|
|
|
|
/// stats, body, etc...) the character is skipped, and no entry will be
|
|
|
|
/// returned.
|
2020-05-12 23:58:15 +00:00
|
|
|
pub fn load_character_list(player_uuid: &str, db_dir: &str) -> CharacterListResult {
|
2020-04-20 08:44:29 +00:00
|
|
|
let data: Vec<(Character, Body, Stats)> = schema::character::dsl::character
|
|
|
|
.filter(schema::character::player_uuid.eq(player_uuid))
|
|
|
|
.order(schema::character::id.desc())
|
|
|
|
.inner_join(schema::body::table)
|
|
|
|
.inner_join(schema::stats::table)
|
2020-05-12 23:58:15 +00:00
|
|
|
.load::<(Character, Body, Stats)>(&establish_connection(db_dir))?;
|
2020-05-09 15:41:25 +00:00
|
|
|
|
|
|
|
Ok(data
|
|
|
|
.iter()
|
2020-04-20 08:44:29 +00:00
|
|
|
.map(|(character_data, body_data, stats_data)| {
|
|
|
|
let character = CharacterData::from(character_data);
|
|
|
|
let body = comp::Body::from(body_data);
|
2020-05-11 10:06:53 +00:00
|
|
|
let level = stats_data.level as usize;
|
2020-04-20 08:44:29 +00:00
|
|
|
|
|
|
|
CharacterItem {
|
|
|
|
character,
|
|
|
|
body,
|
2020-05-11 10:06:53 +00:00
|
|
|
level,
|
2020-04-20 08:44:29 +00:00
|
|
|
}
|
2020-05-09 15:41:25 +00:00
|
|
|
})
|
|
|
|
.collect())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new character with provided comp::Character and comp::Body data.
|
2020-05-11 10:06:53 +00:00
|
|
|
///
|
2020-04-20 08:44:29 +00:00
|
|
|
/// Note that sqlite does not support returning the inserted data after a
|
2020-05-09 15:41:25 +00:00
|
|
|
/// successful insert. To workaround, we wrap this in a transaction which
|
|
|
|
/// inserts, queries for the newly created chaacter id, then uses the character
|
|
|
|
/// id for insertion of the `body` table entry
|
|
|
|
pub fn create_character(
|
|
|
|
uuid: &str,
|
2020-04-20 08:44:29 +00:00
|
|
|
character_alias: String,
|
|
|
|
character_tool: Option<String>,
|
2020-05-09 15:41:25 +00:00
|
|
|
body: &comp::Body,
|
2020-05-12 23:58:15 +00:00
|
|
|
db_dir: &str,
|
2020-05-09 15:41:25 +00:00
|
|
|
) -> CharacterListResult {
|
2020-05-12 23:58:15 +00:00
|
|
|
check_character_limit(uuid, db_dir)?;
|
2020-05-09 15:41:25 +00:00
|
|
|
|
2020-05-12 23:58:15 +00:00
|
|
|
let connection = establish_connection(db_dir);
|
2020-05-09 15:41:25 +00:00
|
|
|
|
|
|
|
connection.transaction::<_, diesel::result::Error, _>(|| {
|
2020-06-01 21:34:52 +00:00
|
|
|
use schema::{body, character, character::dsl::*, inventory, stats};
|
2020-05-09 15:41:25 +00:00
|
|
|
|
|
|
|
match body {
|
|
|
|
comp::Body::Humanoid(body_data) => {
|
2020-04-20 08:44:29 +00:00
|
|
|
let new_character = NewCharacter {
|
|
|
|
player_uuid: uuid,
|
|
|
|
alias: &character_alias,
|
|
|
|
tool: character_tool.as_deref(),
|
|
|
|
};
|
|
|
|
|
2020-05-09 15:41:25 +00:00
|
|
|
diesel::insert_into(character::table)
|
|
|
|
.values(&new_character)
|
|
|
|
.execute(&connection)?;
|
|
|
|
|
|
|
|
let inserted_character = character
|
|
|
|
.filter(player_uuid.eq(uuid))
|
|
|
|
.order(id.desc())
|
|
|
|
.first::<Character>(&connection)?;
|
|
|
|
|
|
|
|
let new_body = Body {
|
|
|
|
character_id: inserted_character.id as i32,
|
2020-05-29 18:23:00 +00:00
|
|
|
species: body_data.species as i16,
|
2020-05-09 15:41:25 +00:00
|
|
|
body_type: body_data.body_type as i16,
|
|
|
|
hair_style: body_data.hair_style as i16,
|
|
|
|
beard: body_data.beard as i16,
|
2020-05-29 18:23:00 +00:00
|
|
|
eyes: body_data.eyes as i16,
|
2020-05-09 15:41:25 +00:00
|
|
|
accessory: body_data.accessory as i16,
|
|
|
|
hair_color: body_data.hair_color as i16,
|
|
|
|
skin: body_data.skin as i16,
|
|
|
|
eye_color: body_data.eye_color as i16,
|
|
|
|
};
|
|
|
|
|
|
|
|
diesel::insert_into(body::table)
|
|
|
|
.values(&new_body)
|
|
|
|
.execute(&connection)?;
|
2020-04-20 08:44:29 +00:00
|
|
|
|
|
|
|
let default_stats = comp::Stats::new(String::from(new_character.alias), *body);
|
|
|
|
|
|
|
|
// Insert some default stats
|
|
|
|
let new_stats = Stats {
|
|
|
|
character_id: inserted_character.id as i32,
|
|
|
|
level: default_stats.level.level() as i32,
|
|
|
|
exp: default_stats.exp.current() as i32,
|
|
|
|
endurance: default_stats.endurance as i32,
|
|
|
|
fitness: default_stats.fitness as i32,
|
|
|
|
willpower: default_stats.willpower as i32,
|
|
|
|
};
|
|
|
|
|
|
|
|
diesel::insert_into(stats::table)
|
|
|
|
.values(&new_stats)
|
|
|
|
.execute(&connection)?;
|
2020-06-01 21:34:52 +00:00
|
|
|
|
|
|
|
// Default inventory
|
|
|
|
let inventory =
|
|
|
|
Inventory::from((inserted_character.id, comp::Inventory::default()));
|
|
|
|
|
|
|
|
diesel::insert_into(inventory::table)
|
|
|
|
.values(&inventory)
|
|
|
|
.execute(&connection)?;
|
2020-05-09 15:41:25 +00:00
|
|
|
},
|
|
|
|
_ => log::warn!("Creating non-humanoid characters is not supported."),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
})?;
|
|
|
|
|
2020-05-12 23:58:15 +00:00
|
|
|
load_character_list(uuid, db_dir)
|
2020-05-09 15:41:25 +00:00
|
|
|
}
|
|
|
|
|
2020-05-11 10:06:53 +00:00
|
|
|
/// Delete a character. Returns the updated character list.
|
2020-05-12 23:58:15 +00:00
|
|
|
pub fn delete_character(uuid: &str, character_id: i32, db_dir: &str) -> CharacterListResult {
|
2020-05-09 15:41:25 +00:00
|
|
|
use schema::character::dsl::*;
|
|
|
|
|
2020-05-30 18:31:31 +00:00
|
|
|
diesel::delete(
|
|
|
|
character
|
|
|
|
.filter(id.eq(character_id))
|
|
|
|
.filter(player_uuid.eq(uuid)),
|
|
|
|
)
|
|
|
|
.execute(&establish_connection(db_dir))?;
|
2020-05-09 15:41:25 +00:00
|
|
|
|
2020-05-12 23:58:15 +00:00
|
|
|
load_character_list(uuid, db_dir)
|
2020-05-09 15:41:25 +00:00
|
|
|
}
|
|
|
|
|
2020-05-12 23:58:15 +00:00
|
|
|
fn check_character_limit(uuid: &str, db_dir: &str) -> Result<(), Error> {
|
2020-05-09 15:41:25 +00:00
|
|
|
use diesel::dsl::count_star;
|
|
|
|
use schema::character::dsl::*;
|
|
|
|
|
|
|
|
let character_count = character
|
|
|
|
.select(count_star())
|
|
|
|
.filter(player_uuid.eq(uuid))
|
2020-05-12 23:58:15 +00:00
|
|
|
.load::<i64>(&establish_connection(db_dir))?;
|
2020-05-09 15:41:25 +00:00
|
|
|
|
|
|
|
match character_count.first() {
|
|
|
|
Some(count) => {
|
|
|
|
if count < &(MAX_CHARACTERS_PER_PLAYER as i64) {
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(Error::CharacterLimitReached)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => Ok(()),
|
|
|
|
}
|
|
|
|
}
|
2020-06-01 21:34:52 +00:00
|
|
|
|
|
|
|
pub type CharacterUpdateData = (StatsUpdate, InventoryUpdate);
|
|
|
|
|
|
|
|
pub struct CharacterUpdater {
|
|
|
|
update_tx: Option<channel::Sender<Vec<(i32, CharacterUpdateData)>>>,
|
|
|
|
handle: Option<std::thread::JoinHandle<()>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CharacterUpdater {
|
|
|
|
pub fn new(db_dir: String) -> Self {
|
|
|
|
let (update_tx, update_rx) = channel::unbounded::<Vec<(i32, CharacterUpdateData)>>();
|
|
|
|
let handle = std::thread::spawn(move || {
|
|
|
|
while let Ok(updates) = update_rx.recv() {
|
|
|
|
batch_update(updates.into_iter(), &db_dir);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
Self {
|
|
|
|
update_tx: Some(update_tx),
|
|
|
|
handle: Some(handle),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn batch_update<'a>(
|
|
|
|
&self,
|
|
|
|
updates: impl Iterator<Item = (i32, &'a comp::Stats, &'a comp::Inventory)>,
|
|
|
|
) {
|
|
|
|
let updates = updates
|
|
|
|
.map(|(id, stats, inventory)| {
|
|
|
|
(
|
|
|
|
id,
|
|
|
|
(StatsUpdate::from(stats), InventoryUpdate::from(inventory)),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
if let Err(err) = self.update_tx.as_ref().unwrap().send(updates) {
|
|
|
|
log::error!("Could not send stats updates: {:?}", err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn update(&self, character_id: i32, stats: &comp::Stats, inventory: &comp::Inventory) {
|
|
|
|
self.batch_update(std::iter::once((character_id, stats, inventory)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn batch_update(updates: impl Iterator<Item = (i32, CharacterUpdateData)>, db_dir: &str) {
|
|
|
|
let connection = establish_connection(db_dir);
|
|
|
|
|
|
|
|
if let Err(err) = connection.transaction::<_, diesel::result::Error, _>(|| {
|
|
|
|
updates.for_each(|(character_id, (stats_update, inventory_update))| {
|
|
|
|
update(character_id, &stats_update, &inventory_update, &connection)
|
|
|
|
});
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}) {
|
|
|
|
log::error!("Error during stats batch update transaction: {:?}", err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn update(
|
|
|
|
character_id: i32,
|
|
|
|
stats: &StatsUpdate,
|
|
|
|
inventory: &InventoryUpdate,
|
|
|
|
connection: &SqliteConnection,
|
|
|
|
) {
|
|
|
|
if let Err(error) =
|
|
|
|
diesel::update(schema::stats::table.filter(schema::stats::character_id.eq(character_id)))
|
|
|
|
.set(stats)
|
|
|
|
.execute(connection)
|
|
|
|
{
|
|
|
|
log::warn!(
|
|
|
|
"Failed to update stats for character: {:?}: {:?}",
|
|
|
|
character_id,
|
|
|
|
error
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Err(error) = diesel::update(
|
|
|
|
schema::inventory::table.filter(schema::inventory::character_id.eq(character_id)),
|
|
|
|
)
|
|
|
|
.set(inventory)
|
|
|
|
.execute(connection)
|
|
|
|
{
|
|
|
|
log::warn!(
|
|
|
|
"Failed to update inventory for character: {:?}: {:?}",
|
|
|
|
character_id,
|
|
|
|
error
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for CharacterUpdater {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
drop(self.update_tx.take());
|
|
|
|
if let Err(err) = self.handle.take().unwrap().join() {
|
|
|
|
log::error!("Error from joining character update thread: {:?}", err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|