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-04-20 08:44:29 +00:00
|
|
|
models::{Body, Character, NewCharacter, Stats, StatsJoinData},
|
|
|
|
schema,
|
2020-05-09 15:41:25 +00:00
|
|
|
};
|
|
|
|
use crate::comp;
|
|
|
|
use common::character::{Character as CharacterData, CharacterItem, MAX_CHARACTERS_PER_PLAYER};
|
|
|
|
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-05-12 23:58:15 +00:00
|
|
|
pub fn load_character_data(character_id: i32, db_dir: &str) -> Result<comp::Stats, Error> {
|
2020-05-11 10:06:53 +00:00
|
|
|
let (character_data, body_data, stats_data) = schema::character::dsl::character
|
|
|
|
.filter(schema::character::id.eq(character_id))
|
|
|
|
.inner_join(schema::body::table)
|
|
|
|
.inner_join(schema::stats::table)
|
2020-05-12 23:58:15 +00:00
|
|
|
.first::<(Character, Body, Stats)>(&establish_connection(db_dir))?;
|
2020-05-11 10:06:53 +00:00
|
|
|
|
|
|
|
Ok(comp::Stats::from(StatsJoinData {
|
|
|
|
alias: &character_data.alias,
|
|
|
|
body: &comp::Body::from(&body_data),
|
|
|
|
stats: &stats_data,
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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-04-20 08:44:29 +00:00
|
|
|
use schema::{body, character, character::dsl::*, 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,
|
|
|
|
race: body_data.race as i16,
|
|
|
|
body_type: body_data.body_type as i16,
|
|
|
|
hair_style: body_data.hair_style as i16,
|
|
|
|
beard: body_data.beard as i16,
|
|
|
|
eyebrows: body_data.eyebrows as i16,
|
|
|
|
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-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-12 23:58:15 +00:00
|
|
|
diesel::delete(character.filter(id.eq(character_id))).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(()),
|
|
|
|
}
|
|
|
|
}
|