//! # Implementing new commands. //! To implement a new command, add an instance of `ChatCommand` to //! `CHAT_COMMANDS` and provide a handler function. use crate::{ settings::{BanRecord, EditableSetting}, Server, SpawnPoint, StateExt, }; use assets::AssetExt; use authc::Uuid; use chrono::{NaiveTime, Timelike}; use common::{ assets, cmd::{ChatCommand, BUFF_PACK, BUFF_PARSER, CHAT_COMMANDS, CHAT_SHORTCUTS}, comp::{ self, aura::{Aura, AuraKind, AuraTarget}, buff::{Buff, BuffCategory, BuffData, BuffKind, BuffSource}, inventory::item::{tool::AbilityMap, MaterialStatManifest, Quality}, invite::InviteKind, ChatType, Inventory, Item, LightEmitter, WaypointArea, }, depot, effect::Effect, event::{EventBus, ServerEvent}, npc::{self, get_npc_name}, resources::{PlayerPhysicsSettings, TimeOfDay}, terrain::{Block, BlockKind, SpriteKind, TerrainChunkSize}, uid::Uid, vol::RectVolSize, Damage, DamageSource, Explosion, LoadoutBuilder, RadiusEffect, }; use common_net::{ msg::{DisconnectReason, Notification, PlayerListUpdate, ServerGeneral}, sync::WorldSyncExt, }; use common_state::{BuildAreaError, BuildAreas}; use core::{convert::TryFrom, ops::Not, time::Duration}; use hashbrown::{HashMap, HashSet}; use rand::Rng; use specs::{Builder, Entity as EcsEntity, Join, WorldExt}; use vek::*; use wiring::{Circuit, Wire, WiringAction, WiringActionEffect, WiringElement}; use world::util::Sampler; use crate::{client::Client, login_provider::LoginProvider, wiring}; use scan_fmt::{scan_fmt, scan_fmt_some}; use tracing::{error, info, warn}; pub trait ChatCommandExt { fn execute(&self, server: &mut Server, entity: EcsEntity, args: String); } impl ChatCommandExt for ChatCommand { #[allow(clippy::needless_return)] // TODO: Pending review in #587 fn execute(&self, server: &mut Server, entity: EcsEntity, args: String) { if self.needs_admin() && !server.entity_is_admin(entity) { server.notify_client( entity, ServerGeneral::server_msg( ChatType::CommandError, format!("You don't have permission to use '/{}'.", self.keyword()), ), ); } else if let Err(err) = get_handler(self)(server, entity, entity, args, &self) { server.notify_client( entity, ServerGeneral::server_msg(ChatType::CommandError, err), ); } } } type CmdResult = Result; /// Handler function called when the command is executed. /// # Arguments /// * `&mut Server` - the `Server` instance executing the command. /// * `EcsEntity` - an `Entity` corresponding to the player that invoked the /// command. /// * `EcsEntity` - an `Entity` for the player on whom the command is invoked. /// This differs from the previous argument when using /sudo /// * `String` - a `String` containing the part of the command after the /// keyword. /// * `&ChatCommand` - the command to execute with the above arguments. /// Handler functions must parse arguments from the the given `String` /// (`scan_fmt!` is included for this purpose). /// /// # Returns /// /// A `Result` that is `Ok` if the command went smoothly, and `Err` if it /// failed; on failure, the string is sent to the client who initiated the /// command. type CommandHandler = fn(&mut Server, EcsEntity, EcsEntity, String, &ChatCommand) -> CmdResult<()>; fn get_handler(cmd: &ChatCommand) -> CommandHandler { match cmd { ChatCommand::Adminify => handle_adminify, ChatCommand::Airship => handle_spawn_airship, ChatCommand::Alias => handle_alias, ChatCommand::ApplyBuff => handle_apply_buff, ChatCommand::Ban => handle_ban, ChatCommand::Build => handle_build, ChatCommand::BuildAreaAdd => handle_build_area_add, ChatCommand::BuildAreaList => handle_build_area_list, ChatCommand::BuildAreaRemove => handle_build_area_remove, ChatCommand::Campfire => handle_spawn_campfire, ChatCommand::DebugColumn => handle_debug_column, ChatCommand::DisconnectAllPlayers => handle_disconnect_all_players, ChatCommand::DropAll => handle_drop_all, ChatCommand::Dummy => handle_spawn_training_dummy, ChatCommand::Explosion => handle_explosion, ChatCommand::Faction => handle_faction, ChatCommand::GiveItem => handle_give_item, ChatCommand::Goto => handle_goto, ChatCommand::Group => handle_group, ChatCommand::GroupInvite => handle_group_invite, ChatCommand::GroupKick => handle_group_kick, ChatCommand::GroupLeave => handle_group_leave, ChatCommand::GroupPromote => handle_group_promote, ChatCommand::Health => handle_health, ChatCommand::Help => handle_help, ChatCommand::Home => handle_home, ChatCommand::JoinFaction => handle_join_faction, ChatCommand::Jump => handle_jump, ChatCommand::Kick => handle_kick, ChatCommand::Kill => handle_kill, ChatCommand::KillNpcs => handle_kill_npcs, ChatCommand::Kit => handle_kit, ChatCommand::Lantern => handle_lantern, ChatCommand::Light => handle_light, ChatCommand::MakeBlock => handle_make_block, ChatCommand::MakeSprite => handle_make_sprite, ChatCommand::Motd => handle_motd, ChatCommand::Object => handle_object, ChatCommand::PermitBuild => handle_permit_build, ChatCommand::Players => handle_players, ChatCommand::Region => handle_region, ChatCommand::RemoveLights => handle_remove_lights, ChatCommand::RevokeBuild => handle_revoke_build, ChatCommand::RevokeBuildAll => handle_revoke_build_all, ChatCommand::Safezone => handle_safezone, ChatCommand::Say => handle_say, ChatCommand::ServerPhysics => handle_server_physics, ChatCommand::SetMotd => handle_set_motd, ChatCommand::Site => handle_site, ChatCommand::SkillPoint => handle_skill_point, ChatCommand::Spawn => handle_spawn, ChatCommand::Sudo => handle_sudo, ChatCommand::Tell => handle_tell, ChatCommand::Time => handle_time, ChatCommand::Tp => handle_tp, ChatCommand::Unban => handle_unban, ChatCommand::Version => handle_version, ChatCommand::Waypoint => handle_waypoint, ChatCommand::Wiring => handle_spawn_wiring, ChatCommand::Whitelist => handle_whitelist, ChatCommand::World => handle_world, } } // Fallibly get position of entity with the given descriptor (used for error // message). fn position(server: &Server, entity: EcsEntity, descriptor: &str) -> CmdResult { server .state .ecs() .read_storage::() .get(entity) .copied() .ok_or_else(|| format!("Cannot get position for {:?}!", descriptor)) } fn position_mut( server: &mut Server, entity: EcsEntity, descriptor: &str, f: impl for<'a> FnOnce(&'a mut comp::Pos) -> T, ) -> CmdResult { let mut pos_storage = server.state.ecs_mut().write_storage::(); pos_storage .get_mut(entity) .map(f) .ok_or_else(|| format!("Cannot get position for {:?}!", descriptor)) } fn insert_or_replace_component( server: &mut Server, entity: EcsEntity, component: C, descriptor: &str, ) -> CmdResult<()> { server .state .ecs_mut() .write_storage() .insert(entity, component) .and(Ok(())) .map_err(|_| format!("Entity {:?} is dead!", descriptor)) } // Fallibly get uid of entity with the given descriptor (used for error // message). fn uid(server: &Server, target: EcsEntity, descriptor: &str) -> CmdResult { server .state .ecs() .read_storage::() .get(target) .copied() .ok_or_else(|| format!("Cannot get uid for {:?}", descriptor)) } fn area(server: &mut Server, area_name: &str) -> CmdResult>> { server .state .mut_resource::() .area_names() .get(area_name) .copied() .ok_or_else(|| format!("Area name not found: {}", area_name)) } // Prevent use through sudo. fn no_sudo(client: EcsEntity, target: EcsEntity) -> CmdResult<()> { if client == target { Ok(()) } else { // This happens when [ab]using /sudo Err("It's rude to impersonate people".into()) } } // Prevent application to hardcoded administrators. fn verify_not_hardcoded_admin(server: &mut Server, uuid: Uuid, reason: &str) -> CmdResult<()> { server .editable_settings() .admins .contains(&uuid) .not() .then_some(()) .ok_or_else(|| reason.into()) } fn find_alias(ecs: &specs::World, alias: &str) -> CmdResult<(EcsEntity, Uuid)> { (&ecs.entities(), &ecs.read_storage::()) .join() .find(|(_, player)| player.alias == alias) .map(|(entity, player)| (entity, player.uuid())) .ok_or_else(|| format!("Player {:?} not found!", alias)) } fn find_uuid(ecs: &specs::World, uuid: Uuid) -> CmdResult { (&ecs.entities(), &ecs.read_storage::()) .join() .find(|(_, player)| player.uuid() == uuid) .map(|(entity, _)| entity) .ok_or_else(|| format!("Player with UUID {:?} not found!", uuid)) } fn find_username(server: &mut Server, username: &str) -> CmdResult { server .state .mut_resource::() .username_to_uuid(username) .map_err(|_| format!("Unable to determine UUID for username \"{}\"", username)) } fn handle_drop_all( server: &mut Server, _client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let pos = position(server, target, "target")?; let mut items = Vec::new(); if let Some(mut inventory) = server .state .ecs() .write_storage::() .get_mut(target) { items = inventory.drain().collect(); } let mut rng = rand::thread_rng(); let item_to_place = items .into_iter() .filter(|i| !matches!(i.quality(), Quality::Debug)); for item in item_to_place { let vel = Vec3::new(rng.gen_range(-0.1..0.1), rng.gen_range(-0.1..0.1), 0.5); server .state .create_object(Default::default(), comp::object::Body::Pouch) .with(comp::Pos(Vec3::new( pos.0.x + rng.gen_range(5.0..10.0), pos.0.y + rng.gen_range(5.0..10.0), pos.0.z + 5.0, ))) .with(item) .with(comp::Vel(vel)) .build(); } Ok(()) } #[allow(clippy::useless_conversion)] // TODO: Pending review in #587 fn handle_give_item( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(item_name), give_amount_opt) = scan_fmt_some!(&args, &action.arg_fmt(), String, u32) { let give_amount = give_amount_opt.unwrap_or(1); if let Ok(item) = Item::new_from_asset(&item_name.replace('/', ".").replace("\\", ".")) { let mut item: Item = item; let mut res = Ok(()); if let Ok(()) = item.set_amount(give_amount.min(2000)) { server .state .ecs() .write_storage::() .get_mut(target) .map(|mut inv| { // NOTE: Deliberately ignores items that couldn't be pushed. if inv.push(item).is_err() { res = Err(format!( "Player inventory full. Gave 0 of {} items.", give_amount )); } }); } else { let ability_map = server.state.ecs().read_resource::(); let msm = server.state.ecs().read_resource::(); // This item can't stack. Give each item in a loop. server .state .ecs() .write_storage::() .get_mut(target) .map(|mut inv| { for i in 0..give_amount { // NOTE: Deliberately ignores items that couldn't be pushed. if inv.push(item.duplicate(&ability_map, &msm)).is_err() { res = Err(format!( "Player inventory full. Gave {} of {} items.", i, give_amount )); break; } } }); } insert_or_replace_component( server, target, comp::InventoryUpdate::new(comp::InventoryUpdateEvent::Given), "target", )?; res } else { Err(format!("Invalid item: {}", item_name)) } } else { Err(action.help_string()) } } fn handle_make_block( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(block_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { if let Ok(bk) = BlockKind::try_from(block_name.as_str()) { let pos = position(server, target, "target")?; server.state.set_block( pos.0.map(|e| e.floor() as i32), Block::new(bk, Rgb::broadcast(255)), ); Ok(()) } else { Err(format!("Invalid block kind: {}", block_name)) } } else { Err(action.help_string()) } } fn handle_make_sprite( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(sprite_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { if let Ok(sk) = SpriteKind::try_from(sprite_name.as_str()) { let pos = position(server, target, "target")?; let pos = pos.0.map(|e| e.floor() as i32); let new_block = server .state .get_block(pos) // TODO: Make more principled. .unwrap_or_else(|| Block::air(SpriteKind::Empty)) .with_sprite(sk); server.state.set_block(pos, new_block); Ok(()) } else { Err(format!("Invalid sprite kind: {}", sprite_name)) } } else { Err(action.help_string()) } } fn handle_motd( server: &mut Server, client: EcsEntity, _target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, (*server.editable_settings().server_description).clone(), ), ); Ok(()) } fn handle_set_motd( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let data_dir = server.data_dir(); match scan_fmt!(&args, &action.arg_fmt(), String) { Ok(msg) => { server .editable_settings_mut() .server_description .edit(data_dir.as_ref(), |d| **d = msg.clone()); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Server description set to \"{}\"", msg), ), ); }, Err(_) => { server .editable_settings_mut() .server_description .edit(data_dir.as_ref(), |d| d.clear()); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, "Removed server description".to_string(), ), ); }, } Ok(()) } fn handle_jump( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok((x, y, z)) = scan_fmt!(&args, &action.arg_fmt(), f32, f32, f32) { position_mut(server, target, "target", |current_pos| { current_pos.0 += Vec3::new(x, y, z) })?; insert_or_replace_component(server, target, comp::ForceUpdate, "target") } else { Err(action.help_string()) } } fn handle_goto( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok((x, y, z)) = scan_fmt!(&args, &action.arg_fmt(), f32, f32, f32) { position_mut(server, target, "target", |current_pos| { current_pos.0 = Vec3::new(x, y, z) })?; insert_or_replace_component(server, target, comp::ForceUpdate, "target") } else { Err(action.help_string()) } } /// TODO: Add autocompletion if possible (might require modifying enum to handle /// dynamic values). fn handle_site( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { #[cfg(feature = "worldgen")] if let Ok(dest_name) = scan_fmt!(&args, &action.arg_fmt(), String) { let site = server .world .civs() .sites() .find(|site| { site.site_tmp .map_or(false, |id| server.index.sites[id].name() == dest_name) }) .ok_or_else(|| "Site not found".to_string())?; let site_pos = server .world .find_lowest_accessible_pos(server.index.as_index_ref(), site.center); position_mut(server, target, "target", |current_pos| { current_pos.0 = site_pos })?; insert_or_replace_component(server, target, comp::ForceUpdate, "target") } else { Err(action.help_string()) } #[cfg(not(feature = "worldgen"))] Ok(()) } fn handle_home( server: &mut Server, _client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let home_pos = server.state.mut_resource::().0; let time = *server.state.mut_resource::(); position_mut(server, target, "target", |current_pos| { current_pos.0 = home_pos })?; insert_or_replace_component( server, target, comp::Waypoint::temp_new(home_pos, time), "target", )?; insert_or_replace_component(server, target, comp::ForceUpdate, "target") } fn handle_kill( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let reason = if client == target { comp::HealthSource::Suicide } else if let Some(uid) = server.state.read_storage::().get(client) { comp::HealthSource::Damage { kind: DamageSource::Other, by: Some(*uid), } } else { comp::HealthSource::Command }; server .state .ecs_mut() .write_storage::() .get_mut(target) .map(|mut h| h.set_to(0, reason)); Ok(()) } fn handle_time( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { const DAY: u64 = 86400; let time_in_seconds = server.state.mut_resource::().0; let current_day = time_in_seconds as u64 / DAY; let day_start = (current_day * DAY) as f64; // Find the next occurence of the given time in the day/night cycle let next_cycle = |time| { let new_time = day_start + time; new_time + if new_time < time_in_seconds { DAY as f64 } else { 0.0 } }; let time = scan_fmt_some!(&args, &action.arg_fmt(), String); let new_time = match time.as_deref() { Some("midnight") => { next_cycle(NaiveTime::from_hms(0, 0, 0).num_seconds_from_midnight() as f64) }, Some("night") => { next_cycle(NaiveTime::from_hms(20, 0, 0).num_seconds_from_midnight() as f64) }, Some("dawn") => next_cycle(NaiveTime::from_hms(5, 0, 0).num_seconds_from_midnight() as f64), Some("morning") => { next_cycle(NaiveTime::from_hms(8, 0, 0).num_seconds_from_midnight() as f64) }, Some("day") => next_cycle(NaiveTime::from_hms(10, 0, 0).num_seconds_from_midnight() as f64), Some("noon") => { next_cycle(NaiveTime::from_hms(12, 0, 0).num_seconds_from_midnight() as f64) }, Some("dusk") => { next_cycle(NaiveTime::from_hms(17, 0, 0).num_seconds_from_midnight() as f64) }, Some(n) => match n.parse() { Ok(n) => n, Err(_) => match NaiveTime::parse_from_str(n, "%H:%M") { // Relative to current day Ok(time) => next_cycle(time.num_seconds_from_midnight() as f64), // Accept `u12345`, seconds since midnight day 0 Err(_) => match n .get(1..) .filter(|_| n.starts_with('u')) .and_then(|n| n.trim_start_matches('u').parse::().ok()) { // Absolute time (i.e: since in-game epoch) Some(n) => n as f64, None => { return Err(format!("{:?} is not a valid time.", n)); }, }, }, }, None => { // Would this ever change? Perhaps in a few hundred thousand years some // game archeologists of the future will resurrect the best game of all // time which, obviously, would be Veloren. By that time, the inescapable // laws of thermodynamics will mean that the earth's rotation period // would be slower. Of course, a few hundred thousand years is enough // for the circadian rhythm of human biology to have shifted to account // accordingly. When booting up Veloren for the first time in 337,241 // years, they might feel a touch of anguish at the fact that their // earth days and the days within the game do not neatly divide into // one-another. Understandably, they'll want to change this. Who // wouldn't? It would be like turning the TV volume up to an odd number // or having a slightly untuned radio (assuming they haven't begun // broadcasting information directly into their brains). Totally // unacceptable. No, the correct and proper thing to do would be to // release a retroactive definitive edition DLC for $99 with the very // welcome addition of shorter day periods and a complementary // 'developer commentary' mode created by digging up the long-decayed // skeletons of the Veloren team, measuring various attributes of their // jawlines, and using them to recreate their voices. But how to go about // this Herculean task? This code is jibberish! The last of the core Rust // dev team died exactly 337,194 years ago! Rust is now a long-forgotten // dialect of the ancient ones, lost to the sands of time. Ashes to ashes, // dust to dust. When all hope is lost, one particularly intrepid // post-human hominid exployed by the 'Veloren Revival Corp' (no doubt we // still won't have gotted rid of this blasted 'capitalism' thing by then) // might notice, after years of searching, a particularly curious // inscription within the code. The letters `D`, `A`, `Y`. Curious! She // consults the post-human hominid scholars of the old. Care to empathise // with her shock when she discovers that these symbols, as alien as they // may seem, correspond exactly to the word `ⓕя𝐢ᵇᵇ𝔩E`, the word for // 'day' in the post-human hominid language, which is of course universal. // Imagine also her surprise when, after much further translating, she // finds a comment predicting her very existence and her struggle to // decode this great mystery. Rejoyce! The Veloren Revival Corp. may now // persist with their great Ultimate Edition DLC because the day period // might now be changed because they have found the constant that controls // it! Everybody was henceforth happy until the end of time. // // This one's for you, xMac ;) let current_time = NaiveTime::from_num_seconds_from_midnight_opt( // Wraps around back to 0s if it exceeds 24 hours (24 hours = 86400s) (time_in_seconds as u64 % DAY) as u32, 0, ); let msg = match current_time { Some(time) => format!("It is {}", time.format("%H:%M").to_string()), None => String::from("Unknown Time"), }; server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, msg), ); return Ok(()); }, }; server.state.mut_resource::().0 = new_time; // Update all clients with the new TimeOfDay (without this they would have to // wait for the next 100th tick to receive the update). let mut tod_lazymsg = None; let clients = server.state.ecs().read_storage::(); for client in (&clients).join() { let msg = tod_lazymsg .unwrap_or_else(|| client.prepare(ServerGeneral::TimeOfDay(TimeOfDay(new_time)))); let _ = client.send_prepared(&msg); tod_lazymsg = Some(msg); } if let Some(new_time) = NaiveTime::from_num_seconds_from_midnight_opt(((new_time as u64) % 86400) as u32, 0) { server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Time changed to: {}", new_time.format("%H:%M").to_string(),), ), ); } Ok(()) } fn handle_health( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok(hp) = scan_fmt!(&args, &action.arg_fmt(), u32) { if let Some(mut health) = server .state .ecs() .write_storage::() .get_mut(target) { health.set_to(hp * 10, comp::HealthSource::Command); Ok(()) } else { Err("You have no health".into()) } } else { Err("You must specify health amount!".into()) } } fn handle_alias( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok(alias) = scan_fmt!(&args, &action.arg_fmt(), String) { // Prevent silly aliases comp::Player::alias_validate(&alias).map_err(|e| e.to_string())?; let old_alias_optional = server .state .ecs_mut() .write_storage::() .get_mut(target) .map(|mut player| std::mem::replace(&mut player.alias, alias)); // Update name on client player lists let ecs = server.state.ecs(); if let (Some(uid), Some(player), Some(old_alias)) = ( ecs.read_storage::().get(target), ecs.read_storage::().get(target), old_alias_optional, ) { let msg = ServerGeneral::PlayerListUpdate(PlayerListUpdate::Alias( *uid, player.alias.clone(), )); server.state.notify_players(msg); // Announce alias change if target has a Body. if ecs.read_storage::().get(target).is_some() { server.state.notify_players(ServerGeneral::server_msg( ChatType::CommandInfo, format!("{} is now known as {}.", old_alias, player.alias), )); } } if client != target { // Notify target that an admin changed the alias due to /sudo server.notify_client( target, ServerGeneral::server_msg(ChatType::CommandInfo, "An admin changed your alias."), ); } Ok(()) } else { Err(action.help_string()) } } fn handle_tp( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let player = if let Some(alias) = scan_fmt_some!(&args, &action.arg_fmt(), String) { find_alias(server.state.ecs(), &alias)?.0 } else if client != target { client } else { return Err(action.help_string()); }; let player_pos = position(server, player, "player")?; position_mut(server, target, "target", |target_pos| { *target_pos = player_pos })?; insert_or_replace_component(server, target, comp::ForceUpdate, "target") } fn handle_spawn( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { match scan_fmt_some!(&args, &action.arg_fmt(), String, npc::NpcBody, u32, bool) { (Some(opt_align), Some(npc::NpcBody(id, mut body)), opt_amount, opt_ai) => { let uid = uid(server, target, "target")?; let alignment = parse_alignment(uid, &opt_align)?; let amount = opt_amount.filter(|x| *x > 0).unwrap_or(1).min(50); let ai = opt_ai.unwrap_or(true); let pos = position(server, target, "target")?; let agent = if let comp::Alignment::Owned(_) | comp::Alignment::Npc = alignment { comp::Agent::default() } else { comp::Agent::default().with_patrol_origin(pos.0) }; for _ in 0..amount { let vel = Vec3::new( rand::thread_rng().gen_range(-2.0..3.0), rand::thread_rng().gen_range(-2.0..3.0), 10.0, ); let body = body(); let loadout = LoadoutBuilder::build_loadout(body, None, None, None).build(); let inventory = Inventory::new_with_loadout(loadout); let mut entity_base = server .state .create_npc( pos, comp::Stats::new(get_npc_name(id, npc::BodyType::from_body(body))), comp::SkillSet::default(), comp::Health::new(body, 1), comp::Poise::new(body), inventory, body, ) .with(comp::Vel(vel)) .with(comp::MountState::Unmounted) .with(alignment); if ai { entity_base = entity_base.with(agent.clone()); } let new_entity = entity_base.build(); // Add to group system if a pet if matches!(alignment, comp::Alignment::Owned { .. }) { let state = server.state(); let clients = state.ecs().read_storage::(); let uids = state.ecs().read_storage::(); let mut group_manager = state.ecs().write_resource::(); group_manager.new_pet( new_entity, target, &mut state.ecs().write_storage(), &state.ecs().entities(), &state.ecs().read_storage(), &uids, &mut |entity, group_change| { clients .get(entity) .and_then(|c| { group_change .try_map(|e| uids.get(e).copied()) .map(|g| (g, c)) }) .map(|(g, c)| { c.send_fallible(ServerGeneral::GroupUpdate(g)); }); }, ); } else if let Some(group) = match alignment { comp::Alignment::Wild => None, comp::Alignment::Passive => None, comp::Alignment::Enemy => Some(comp::group::ENEMY), comp::Alignment::Npc | comp::Alignment::Tame => Some(comp::group::NPC), comp::Alignment::Owned(_) => unreachable!(), } { insert_or_replace_component(server, new_entity, group, "new entity")?; } if let Some(uid) = server.state.ecs().uid_from_entity(new_entity) { server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Spawned entity with ID: {}", uid), ), ); } } server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Spawned {} entities", amount), ), ); Ok(()) }, _ => Err(action.help_string()), } } fn handle_spawn_training_dummy( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let pos = position(server, target, "target")?; let vel = Vec3::new( rand::thread_rng().gen_range(-2.0..3.0), rand::thread_rng().gen_range(-2.0..3.0), 10.0, ); let body = comp::Body::Object(comp::object::Body::TrainingDummy); let stats = comp::Stats::new("Training Dummy".to_string()); let skill_set = comp::SkillSet::default(); let health = comp::Health::new(body, 0); let poise = comp::Poise::new(body); server .state .create_npc( pos, stats, skill_set, health, poise, Inventory::new_empty(), body, ) .with(comp::Vel(vel)) .with(comp::MountState::Unmounted) .build(); server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Spawned a training dummy"), ); Ok(()) } fn handle_spawn_airship( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let angle = scan_fmt!(&args, &action.arg_fmt(), f32).ok(); let mut pos = position(server, target, "target")?; pos.0.z += 50.0; const DESTINATION_RADIUS: f32 = 2000.0; let angle = angle.map(|a| a * std::f32::consts::PI / 180.0); let destination = angle.map(|a| { pos.0 + Vec3::new( DESTINATION_RADIUS * a.cos(), DESTINATION_RADIUS * a.sin(), 200.0, ) }); let mut builder = server .state .create_ship(pos, comp::ship::Body::DefaultAirship, true) .with(LightEmitter { col: Rgb::new(1.0, 0.65, 0.2), strength: 2.0, flicker: 1.0, animated: true, }); if let Some(pos) = destination { builder = builder.with(comp::Agent::default().with_destination(pos)) } builder.build(); server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Spawned an airship"), ); Ok(()) } fn handle_spawn_campfire( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let pos = position(server, target, "target")?; server .state .create_object(pos, comp::object::Body::CampfireLit) .with(LightEmitter { col: Rgb::new(1.0, 0.65, 0.2), strength: 2.0, flicker: 1.0, animated: true, }) .with(WaypointArea::default()) .with(comp::Auras::new(vec![Aura::new( AuraKind::Buff { kind: BuffKind::CampfireHeal, data: BuffData::new(0.02, Some(Duration::from_secs(1))), category: BuffCategory::Natural, source: BuffSource::World, }, 5.0, None, AuraTarget::All, )])) .build(); server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Spawned a campfire"), ); Ok(()) } fn handle_safezone( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let range = scan_fmt_some!(&args, &action.arg_fmt(), f32); let pos = position(server, target, "target")?; server.state.create_safezone(range, pos).build(); server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Spawned a safe zone"), ); Ok(()) } fn handle_permit_build( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(area_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let bb_id = area(server, &area_name)?; let mut can_build = server.state.ecs().write_storage::(); let entry = can_build .entry(target) .map_err(|_| "Cannot find target entity!".to_string())?; let mut comp_can_build = entry.or_insert(comp::CanBuild { enabled: false, build_areas: HashSet::new(), }); comp_can_build.build_areas.insert(bb_id); drop(can_build); if client != target { server.notify_client( target, ServerGeneral::server_msg( ChatType::CommandInfo, format!("You are now permitted to build in {}", area_name), ), ); } server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Permission to build in {} granted", area_name), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_revoke_build( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(area_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let bb_id = area(server, &area_name)?; let mut can_build = server.state.ecs_mut().write_storage::(); if let Some(mut comp_can_build) = can_build.get_mut(target) { comp_can_build.build_areas.retain(|&x| x != bb_id); drop(can_build); if client != target { server.notify_client( target, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Your permission to build in {} has been revoked", area_name), ), ); } server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Permission to build in {} revoked", area_name), ), ); Ok(()) } else { Err("You do not have permission to build.".into()) } } else { Err(action.help_string()) } } fn handle_revoke_build_all( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let ecs = server.state.ecs(); ecs.write_storage::().remove(target); if client != target { server.notify_client( target, ServerGeneral::server_msg( ChatType::CommandInfo, "Your build permissions have been revoked.", ), ); } server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "All build permissions revoked"), ); Ok(()) } fn handle_players( server: &mut Server, client: EcsEntity, _target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let ecs = server.state.ecs(); let entity_tuples = ( &ecs.entities(), &ecs.read_storage::(), &ecs.read_storage::(), ); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, entity_tuples.join().fold( format!("{} online players:", entity_tuples.join().count()), |s, (_, player, stat)| format!("{}\n[{}]{}", s, player.alias, stat.name,), ), ), ); Ok(()) } fn handle_build( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { if let Some(mut can_build) = server .state .ecs() .write_storage::() .get_mut(target) { let toggle_string = if can_build.enabled { "off" } else { "on" }; let chat_msg = ServerGeneral::server_msg( ChatType::CommandInfo, format!("Toggled {:?} build mode!", toggle_string), ); can_build.enabled ^= true; if client != target { server.notify_client(target, chat_msg.clone()); } server.notify_client(client, chat_msg); Ok(()) } else { Err("You do not have permission to build.".into()) } } fn handle_build_area_add( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(area_name), Some(xlo), Some(xhi), Some(ylo), Some(yhi), Some(zlo), Some(zhi)) = scan_fmt_some!( &args, &action.arg_fmt(), String, i32, i32, i32, i32, i32, i32 ) { let build_areas = server.state.mut_resource::(); let msg = ServerGeneral::server_msg( ChatType::CommandInfo, format!("Created build zone {}", area_name), ); build_areas .insert(area_name, Aabb { min: Vec3::new(xlo, ylo, zlo), max: Vec3::new(xhi, yhi, zhi), }) .map_err(|area_name| format!("Build zone {} already exists!", area_name))?; server.notify_client(client, msg); Ok(()) } else { Err(action.help_string()) } } fn handle_build_area_list( server: &mut Server, client: EcsEntity, _target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let build_areas = server.state.mut_resource::(); let msg = ServerGeneral::server_msg( ChatType::CommandInfo, build_areas.area_names().iter().fold( "Build Areas:".to_string(), |acc, (area_name, bb_id)| { if let Some(aabb) = build_areas.areas().get(*bb_id) { format!("{}\n{}: {} to {}", acc, area_name, aabb.min, aabb.max) } else { acc } }, ), ); server.notify_client(client, msg); Ok(()) } fn handle_build_area_remove( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(area_name) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let build_areas = server.state.mut_resource::(); build_areas.remove(&area_name).map_err(|err| match err { BuildAreaError::Reserved => format!( "Build area is reserved and cannot be removed: {}", area_name ), BuildAreaError::NotFound => format!("No such build area {}", area_name), })?; server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Removed build zone {}", area_name), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_help( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(cmd) = scan_fmt_some!(&args, &action.arg_fmt(), ChatCommand) { server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, cmd.help_string()), ) } else { let mut message = String::new(); for cmd in CHAT_COMMANDS.iter() { if !cmd.needs_admin() || server.entity_is_admin(client) { message += &cmd.help_string(); message += "\n"; } } message += "Additionally, you can use the following shortcuts:"; for (k, v) in CHAT_SHORTCUTS.iter() { message += &format!(" /{} => /{}", k, v.keyword()); } server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, message), ) } Ok(()) } fn parse_alignment(owner: Uid, alignment: &str) -> CmdResult { match alignment { "wild" => Ok(comp::Alignment::Wild), "enemy" => Ok(comp::Alignment::Enemy), "npc" => Ok(comp::Alignment::Npc), "pet" => Ok(comp::Alignment::Owned(owner)), _ => Err(format!("Invalid alignment: {:?}", alignment)), } } fn handle_kill_npcs( server: &mut Server, client: EcsEntity, _target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let ecs = server.state.ecs(); let mut healths = ecs.write_storage::(); let players = ecs.read_storage::(); let mut count = 0; for (mut health, ()) in (&mut healths, !&players).join() { count += 1; health.set_to(0, comp::HealthSource::Command); } let text = if count > 0 { format!("Destroyed {} NPCs.", count) } else { "No NPCs on server.".to_string() }; server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, text), ); Ok(()) } fn handle_kit( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let kit_name = scan_fmt!(&args, &action.arg_fmt(), String); if let Ok(name) = kit_name { if let Ok(kits) = common::cmd::KitManifest::load("server.manifests.kits") { let kits = kits.read(); if let Some(kit) = kits.0.get(&name) { if let (Some(mut target_inventory), mut target_inv_update) = ( server .state() .ecs() .write_storage::() .get_mut(target), server.state.ecs().write_storage::(), ) { for (item_id, quantity) in kit.iter() { if let Ok(mut item) = comp::Item::new_from_asset(item_id) { let _ = item.set_amount(*quantity); let (_, _) = ( target_inventory.push(item), target_inv_update.insert( target, comp::InventoryUpdate::new(comp::InventoryUpdateEvent::Debug), ), ); } } Ok(()) } else { Err("Could not get inventory".to_string()) } } else { Err(format!("Kit '{}' not found", name)) } } else { Err("Could not load manifest file 'server.manifests.kits'".to_string()) } } else { Err(action.help_string()) } } #[allow(clippy::float_cmp)] // TODO: Pending review in #587 #[allow(clippy::needless_return)] // TODO: Pending review in #587 #[allow(clippy::useless_format)] // TODO: Pending review in #587 fn handle_object( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let obj_type = scan_fmt!(&args, &action.arg_fmt(), String); let pos = position(server, target, "target")?; let ori = server .state .ecs() .read_storage::() .get(target) .copied() .ok_or_else(|| "Cannot get orientation for target".to_string())?; /*let builder = server.state .create_object(pos, ori, obj_type) .with(ori);*/ let obj_str_res = obj_type.as_ref().map(String::as_str); if let Some(obj_type) = comp::object::ALL_OBJECTS .iter() .find(|o| Ok(o.to_string()) == obj_str_res) { server .state .create_object(pos, *obj_type) .with( comp::Ori::from_unnormalized_vec( // converts player orientation into a 90° rotation for the object by using // the axis with the highest value { let look_dir = ori.look_dir(); look_dir.map(|e| { if e.abs() == look_dir.map(|e| e.abs()).reduce_partial_max() { e } else { 0.0 } }) }, ) .unwrap_or_default(), ) .build(); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Spawned: {}", obj_str_res.unwrap_or("")), ), ); Ok(()) } else { Err("Object not found!".into()) } } #[allow(clippy::useless_format)] // TODO: Pending review in #587 fn handle_light( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let (opt_r, opt_g, opt_b, opt_x, opt_y, opt_z, opt_s) = scan_fmt_some!(&args, &action.arg_fmt(), f32, f32, f32, f32, f32, f32, f32); let mut light_emitter = comp::LightEmitter::default(); let mut light_offset_opt = None; if let (Some(r), Some(g), Some(b)) = (opt_r, opt_g, opt_b) { if r < 0.0 || g < 0.0 || b < 0.0 { return Err("cr, cg and cb values mustn't be negative.".into()); } let r = r.max(0.0).min(1.0); let g = g.max(0.0).min(1.0); let b = b.max(0.0).min(1.0); light_emitter.col = Rgb::new(r, g, b) }; if let (Some(x), Some(y), Some(z)) = (opt_x, opt_y, opt_z) { light_offset_opt = Some(comp::LightAnimation { offset: Vec3::new(x, y, z), col: light_emitter.col, strength: 0.0, }) }; if let Some(s) = opt_s { light_emitter.strength = s.max(0.0) }; let pos = position(server, target, "target")?; let builder = server .state .ecs_mut() .create_entity_synced() .with(pos) .with(comp::ForceUpdate) .with(light_emitter); if let Some(light_offset) = light_offset_opt { builder.with(light_offset).build(); } else { builder.build(); } server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Spawned object."), ); Ok(()) } #[allow(clippy::useless_conversion)] // TODO: Pending review in #587 fn handle_lantern( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(s), r, g, b) = scan_fmt_some!(&args, &action.arg_fmt(), f32, f32, f32, f32) { if let Some(mut light) = server .state .ecs() .write_storage::() .get_mut(target) { light.strength = s.max(0.1).min(10.0); if let (Some(r), Some(g), Some(b)) = (r, g, b) { light.col = ( r.max(0.0).min(1.0), g.max(0.0).min(1.0), b.max(0.0).min(1.0), ) .into(); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, "You adjusted flame strength and color.", ), ) } else { server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, "You adjusted flame strength.", ), ) } Ok(()) } else { Err("Please equip a lantern first".into()) } } else { Err(action.help_string()) } } fn handle_explosion( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let power = scan_fmt!(&args, &action.arg_fmt(), f32).unwrap_or(8.0); const MIN_POWER: f32 = 0.0; const MAX_POWER: f32 = 512.0; if power > MAX_POWER { return Err(format!( "Explosion power mustn't be more than {:?}.", MAX_POWER )); } else if power <= 0.0 { return Err(format!( "Explosion power must be more than {:?}.", MIN_POWER )); } let pos = position(server, target, "target")?; let owner = server .state .ecs() .read_storage::() .get(target) .copied(); server .state .mut_resource::>() .emit_now(ServerEvent::Explosion { pos: pos.0, explosion: Explosion { effects: vec![ RadiusEffect::Entity(Effect::Damage(Damage { source: DamageSource::Explosion, value: 100.0 * power, })), RadiusEffect::TerrainDestruction(power), ], radius: 3.0 * power, reagent: None, }, owner, }); Ok(()) } fn handle_waypoint( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { let pos = position(server, target, "target")?; let time = *server.state.mut_resource::(); insert_or_replace_component( server, target, comp::Waypoint::temp_new(pos.0, time), "target", )?; server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Waypoint saved!"), ); server.notify_client( target, ServerGeneral::Notification(Notification::WaypointSaved), ); Ok(()) } fn handle_spawn_wiring( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { // Obviously it is a WIP - use it for debug let mut pos = position(server, target, "target")?; pos.0.x += 3.0; let mut outputs1 = HashMap::new(); outputs1.insert(String::from("color"), wiring::OutputFormula::OnDeath { value: 1.0, radius: 30.0, }); let builder1 = server .state .create_wiring(pos, comp::object::Body::Coins, WiringElement { actions: vec![WiringAction { formula: wiring::OutputFormula::Constant { value: 1.0 }, threshold: 1.0, effects: vec![WiringActionEffect::SetLight { r: wiring::OutputFormula::Input { name: String::from("color"), }, g: wiring::OutputFormula::Input { name: String::from("color"), }, b: wiring::OutputFormula::Input { name: String::from("color"), }, }], }], inputs: HashMap::new(), outputs: outputs1, }) .with(comp::Sticky); let ent1 = builder1.build(); pos.0.x += 3.0; let builder2 = server .state .create_wiring(pos, comp::object::Body::Coins, WiringElement { actions: vec![ WiringAction { formula: wiring::OutputFormula::Input { name: String::from("color"), }, threshold: 1.0, effects: vec![WiringActionEffect::SpawnProjectile { constr: comp::ProjectileConstructor::Arrow { damage: 1.0, energy_regen: 0.0, knockback: 0.0, }, }], }, WiringAction { formula: wiring::OutputFormula::Constant { value: 1.0 }, threshold: 1.0, effects: vec![WiringActionEffect::SetLight { r: wiring::OutputFormula::Input { name: String::from("color"), }, g: wiring::OutputFormula::Input { name: String::from("color"), }, b: wiring::OutputFormula::Input { name: String::from("color"), }, }], }, ], inputs: HashMap::new(), outputs: HashMap::new(), }); let ent2 = builder2.build(); pos.0.x += 3.0; let builder3 = server .state .create_wiring(pos, comp::object::Body::TrainingDummy, WiringElement { actions: vec![], inputs: HashMap::new(), outputs: HashMap::new(), }) .with(Circuit { wires: vec![Wire { input_entity: ent1, input_field: String::from("color"), output_entity: ent2, output_field: String::from("color"), }], }); builder3.build(); server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, "Wire"), ); Ok(()) } #[allow(clippy::useless_conversion)] // TODO: Pending review in #587 fn handle_adminify( server: &mut Server, _client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok(alias) = scan_fmt!(&args, &action.arg_fmt(), String) { let (player, uuid) = find_alias(server.state.ecs(), &alias)?; let uid = uid(server, player, "player")?; verify_not_hardcoded_admin( server, uuid, "Admins specified in server configuration files cannot be de-adminified.", )?; let is_admin = if server .state .read_component_copied::(player) .is_some() { server .state .ecs() .write_storage::() .remove(player); false } else { server .state .ecs() .write_storage() .insert(player, comp::Admin) .is_ok() }; // Update player list so the player shows up as admin in client chat. let msg = ServerGeneral::PlayerListUpdate(PlayerListUpdate::Admin(uid, is_admin)); server.state.notify_players(msg); Ok(()) } else { Err(action.help_string()) } } #[allow(clippy::useless_conversion)] // TODO: Pending review in #587 #[allow(clippy::useless_format)] // TODO: Pending review in #587 fn handle_tell( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; if let (Some(alias), message_opt) = scan_fmt_some!(&args, &action.arg_fmt(), String, String) { let ecs = server.state.ecs(); let player = find_alias(ecs, &alias)?.0; if player == target { return Err("You can't /tell yourself.".into()); } let target_uid = uid(server, target, "target")?; let player_uid = uid(server, player, "player")?; let mode = comp::ChatMode::Tell(player_uid); insert_or_replace_component(server, target, mode.clone(), "target")?; let msg = message_opt.unwrap_or_else(|| format!("{} wants to talk to you.", alias)); server.state.send_chat(mode.new_message(target_uid, msg)); server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } else { Err(action.help_string()) } } fn handle_faction( server: &mut Server, client: EcsEntity, target: EcsEntity, msg: String, _action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; let factions = server.state.ecs().read_storage(); if let Some(comp::Faction(faction)) = factions.get(target) { let mode = comp::ChatMode::Faction(faction.to_string()); drop(factions); insert_or_replace_component(server, target, mode.clone(), "target")?; if !msg.is_empty() { if let Some(uid) = server.state.ecs().read_storage().get(target) { server.state.send_chat(mode.new_message(*uid, msg)); } } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } else { Err("Please join a faction with /join_faction".into()) } } fn handle_group( server: &mut Server, client: EcsEntity, target: EcsEntity, msg: String, _action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; let groups = server.state.ecs().read_storage::(); if let Some(group) = groups.get(target) { let mode = comp::ChatMode::Group(*group); drop(groups); insert_or_replace_component(server, target, mode.clone(), "target")?; if !msg.is_empty() { if let Some(uid) = server.state.ecs().read_storage().get(target) { server.state.send_chat(mode.new_message(*uid, msg)); } } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } else { Err("Please create a group first".into()) } } fn handle_group_invite( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Some(target_alias) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let target_player = find_alias(server.state.ecs(), &target_alias)?.0; let uid = uid(server, target_player, "player")?; server .state .mut_resource::>() .emit_now(ServerEvent::InitiateInvite(target, uid, InviteKind::Group)); if client != target { server.notify_client( target, ServerGeneral::server_msg( ChatType::CommandInfo, format!("{} has been invited to your group.", target_alias), ), ); } server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Invited {} to the group.", target_alias), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_group_kick( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { // Checking if leader is already done in group_manip if let Some(target_alias) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let target_player = find_alias(server.state.ecs(), &target_alias)?.0; let uid = uid(server, target_player, "player")?; server .state .mut_resource::>() .emit_now(ServerEvent::GroupManip(target, comp::GroupManip::Kick(uid))); Ok(()) } else { Err(action.help_string()) } } fn handle_group_leave( server: &mut Server, _client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { server .state .mut_resource::>() .emit_now(ServerEvent::GroupManip(target, comp::GroupManip::Leave)); Ok(()) } fn handle_group_promote( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { // Checking if leader is already done in group_manip if let Some(target_alias) = scan_fmt_some!(&args, &action.arg_fmt(), String) { let target_player = find_alias(server.state.ecs(), &target_alias)?.0; let uid = uid(server, target_player, "player")?; server .state .mut_resource::>() .emit_now(ServerEvent::GroupManip( target, comp::GroupManip::AssignLeader(uid), )); Ok(()) } else { Err(action.help_string()) } } fn handle_region( server: &mut Server, client: EcsEntity, target: EcsEntity, msg: String, _action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; let mode = comp::ChatMode::Region; insert_or_replace_component(server, target, mode.clone(), "target")?; if !msg.is_empty() { if let Some(uid) = server.state.ecs().read_storage().get(target) { server.state.send_chat(mode.new_message(*uid, msg)); } } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } fn handle_say( server: &mut Server, client: EcsEntity, target: EcsEntity, msg: String, _action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; let mode = comp::ChatMode::Say; insert_or_replace_component(server, target, mode.clone(), "target")?; if !msg.is_empty() { if let Some(uid) = server.state.ecs().read_storage().get(target) { server.state.send_chat(mode.new_message(*uid, msg)); } } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } fn handle_world( server: &mut Server, client: EcsEntity, target: EcsEntity, msg: String, _action: &ChatCommand, ) -> CmdResult<()> { no_sudo(client, target)?; let mode = comp::ChatMode::World; insert_or_replace_component(server, target, mode.clone(), "target")?; if !msg.is_empty() { if let Some(uid) = server.state.ecs().read_storage().get(target) { server.state.send_chat(mode.new_message(*uid, msg)); } } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } fn handle_join_faction( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let players = server.state.ecs().read_storage::(); if let Some(alias) = players.get(target).map(|player| player.alias.clone()) { drop(players); let (faction_leave, mode) = if let Ok(faction) = scan_fmt!(&args, &action.arg_fmt(), String) { let mode = comp::ChatMode::Faction(faction.clone()); insert_or_replace_component(server, target, mode.clone(), "target")?; let faction_join = server .state .ecs() .write_storage() .insert(target, comp::Faction(faction.clone())) .ok() .flatten() .map(|f| f.0); server.state.send_chat( ChatType::FactionMeta(faction.clone()) .chat_msg(format!("[{}] joined faction ({})", alias, faction)), ); (faction_join, mode) } else { let mode = comp::ChatMode::default(); insert_or_replace_component(server, target, mode.clone(), "target")?; let faction_leave = server .state .ecs() .write_storage() .remove(target) .map(|comp::Faction(f)| f); (faction_leave, mode) }; if let Some(faction) = faction_leave { server.state.send_chat( ChatType::FactionMeta(faction.clone()) .chat_msg(format!("[{}] left faction ({})", alias, faction)), ); } server.notify_client(target, ServerGeneral::ChatMode(mode)); Ok(()) } else { Err("Could not find your player alias".into()) } } #[cfg(not(feature = "worldgen"))] fn handle_debug_column( server: &mut Server, client: EcsEntity, target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { Err("Unsupported without worldgen enabled".into()) } #[cfg(feature = "worldgen")] fn handle_debug_column( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let sim = server.world.sim(); let sampler = server.world.sample_columns(); let wpos = if let Ok((x, y)) = scan_fmt!(&args, &action.arg_fmt(), i32, i32) { Vec2::new(x, y) } else { let pos = position(server, target, "target")?; // FIXME: Deal with overflow, if needed. pos.0.xy().map(|x| x as i32) }; let msg_generator = || { let alt = sim.get_interpolated(wpos, |chunk| chunk.alt)?; let basement = sim.get_interpolated(wpos, |chunk| chunk.basement)?; let water_alt = sim.get_interpolated(wpos, |chunk| chunk.water_alt)?; let chaos = sim.get_interpolated(wpos, |chunk| chunk.chaos)?; let temp = sim.get_interpolated(wpos, |chunk| chunk.temp)?; let humidity = sim.get_interpolated(wpos, |chunk| chunk.humidity)?; let rockiness = sim.get_interpolated(wpos, |chunk| chunk.rockiness)?; let tree_density = sim.get_interpolated(wpos, |chunk| chunk.tree_density)?; let spawn_rate = sim.get_interpolated(wpos, |chunk| chunk.spawn_rate)?; let chunk_pos = wpos.map2(TerrainChunkSize::RECT_SIZE, |e, sz: u32| e / sz as i32); let chunk = sim.get(chunk_pos)?; let col = sampler.get((wpos, server.index.as_index_ref()))?; let gradient = sim.get_gradient_approx(chunk_pos)?; let downhill = chunk.downhill; let river = &chunk.river; let flux = chunk.flux; Some(format!( r#"wpos: {:?} alt {:?} ({:?}) water_alt {:?} ({:?}) basement {:?} river {:?} gradient {:?} downhill {:?} chaos {:?} flux {:?} temp {:?} humidity {:?} rockiness {:?} tree_density {:?} spawn_rate {:?} "#, wpos, alt, col.alt, water_alt, col.water_level, basement, river, gradient, downhill, chaos, flux, temp, humidity, rockiness, tree_density, spawn_rate )) }; if let Some(s) = msg_generator() { server.notify_client(client, ServerGeneral::server_msg(ChatType::CommandInfo, s)); Ok(()) } else { Err("Not a pregenerated chunk.".into()) } } fn handle_disconnect_all_players( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, _action: &ChatCommand, ) -> CmdResult<()> { if args != *"confirm" { return Err( "Please run the command again with the second argument of \"confirm\" to confirm that \ you really want to disconnect all players from the server" .to_string(), ); } let ecs = server.state.ecs(); let players = &ecs.read_storage::(); // TODO: This logging and verification of admin commands would be better moved // to a more generic method used for auditing -all- admin commands. let player_name; if let Some(player) = players.get(client) { player_name = &*player.alias; } else { warn!( "Failed to get player name for admin who used /disconnect_all_players - ignoring \ command." ); return Err("You do not exist, so you cannot use this command".to_string()); } info!( "Disconnecting all clients due to admin command from {}", player_name ); server.disconnect_all_clients_requested = true; Ok(()) } fn handle_skill_point( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(a_skill_tree), Some(sp), a_alias) = scan_fmt_some!(&args, &action.arg_fmt(), String, u16, String) { let skill_tree = parse_skill_tree(&a_skill_tree)?; let player = a_alias .map(|alias| find_alias(server.state.ecs(), &alias).map(|(target, _)| target)) .unwrap_or(Ok(target))?; if let Some(mut skill_set) = server .state .ecs_mut() .write_storage::() .get_mut(player) { skill_set.add_skill_points(skill_tree, sp); Ok(()) } else { Err("Player has no stats!".into()) } } else { Err(action.help_string()) } } fn parse_skill_tree(skill_tree: &str) -> CmdResult { use comp::{item::tool::ToolKind, skills::SkillGroupKind}; match skill_tree { "general" => Ok(SkillGroupKind::General), "sword" => Ok(SkillGroupKind::Weapon(ToolKind::Sword)), "axe" => Ok(SkillGroupKind::Weapon(ToolKind::Axe)), "hammer" => Ok(SkillGroupKind::Weapon(ToolKind::Hammer)), "bow" => Ok(SkillGroupKind::Weapon(ToolKind::Bow)), "staff" => Ok(SkillGroupKind::Weapon(ToolKind::Staff)), "sceptre" => Ok(SkillGroupKind::Weapon(ToolKind::Sceptre)), _ => Err(format!("{} is not a skill group!", skill_tree)), } } fn handle_remove_lights( server: &mut Server, client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { let opt_radius = scan_fmt_some!(&args, &action.arg_fmt(), f32); let player_pos = position(server, target, "target")?; let mut to_delete = vec![]; let ecs = server.state.ecs(); for (entity, pos, _, _, _) in ( &ecs.entities(), &ecs.read_storage::(), &ecs.read_storage::(), !&ecs.read_storage::(), !&ecs.read_storage::(), ) .join() { if opt_radius .map(|r| pos.0.distance(player_pos.0) < r) .unwrap_or(true) { to_delete.push(entity); } } let size = to_delete.len(); for entity in to_delete { if let Err(e) = server.state.delete_entity_recorded(entity) { error!(?e, "Failed to delete light: {:?}", e); } } server.notify_client( client, ServerGeneral::server_msg(ChatType::CommandInfo, format!("Removed {} lights!", size)), ); Ok(()) } fn handle_sudo( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(player_alias), Some(cmd), cmd_args) = scan_fmt_some!(&args, &action.arg_fmt(), String, String, String) { let cmd_args = cmd_args.unwrap_or_else(|| String::from("")); if let Ok(action) = cmd.parse() { let ecs = server.state.ecs(); let (entity, uuid) = find_alias(ecs, &player_alias)?; verify_not_hardcoded_admin( server, uuid, "Cannot sudo admins specified in server configuration files.", )?; get_handler(&action)(server, client, entity, cmd_args, &action) } else { Err(format!("Unknown command: /{}", cmd)) } } else { Err(action.help_string()) } } fn handle_version( server: &mut Server, client: EcsEntity, _target: EcsEntity, _args: String, _action: &ChatCommand, ) -> CmdResult<()> { server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!( "Server is running {}[{}]", common::util::GIT_HASH.to_string(), common::util::GIT_DATE.to_string(), ), ), ); Ok(()) } fn handle_whitelist( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok((whitelist_action, username)) = scan_fmt!(&args, &action.arg_fmt(), String, String) { if whitelist_action.eq_ignore_ascii_case("add") { let uuid = find_username(server, &username)?; server .editable_settings_mut() .whitelist .edit(server.data_dir().as_ref(), |w| w.insert(uuid)); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("\"{}\" added to whitelist", username), ), ); Ok(()) } else if whitelist_action.eq_ignore_ascii_case("remove") { let uuid = find_username(server, &username)?; server .editable_settings_mut() .whitelist .edit(server.data_dir().as_ref(), |w| w.remove(&uuid)); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("\"{}\" removed from whitelist", username), ), ); Ok(()) } else { Err(action.help_string()) } } else { Err(action.help_string()) } } fn kick_player( server: &mut Server, (target_player, uuid): (EcsEntity, Uuid), reason: &str, ) -> CmdResult<()> { verify_not_hardcoded_admin( server, uuid, "Cannot kick admins specified in server configuration files.", )?; server.notify_client( target_player, ServerGeneral::Disconnect(DisconnectReason::Kicked(reason.to_string())), ); server .state .mut_resource::>() .emit_now(ServerEvent::ClientDisconnect( target_player, common::comp::DisconnectReason::Kicked, )); Ok(()) } fn handle_kick( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(target_alias), reason_opt) = scan_fmt_some!(&args, &action.arg_fmt(), String, String) { let reason = reason_opt.unwrap_or_default(); let ecs = server.state.ecs(); let target_player = find_alias(ecs, &target_alias)?; kick_player(server, target_player, &reason)?; server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!( "Kicked {} from the server with reason: {}", target_alias, reason ), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_ban( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(username), reason_opt) = scan_fmt_some!(&args, &action.arg_fmt(), String, String) { let reason = reason_opt.unwrap_or_default(); let uuid = find_username(server, &username)?; if server.editable_settings().banlist.contains_key(&uuid) { Err(format!("{} is already on the banlist", username)) } else { server .editable_settings_mut() .banlist .edit(server.data_dir().as_ref(), |b| { b.insert(uuid, BanRecord { username_when_banned: username.clone(), reason: reason.clone(), }); }); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("Added {} to the banlist with reason: {}", username, reason), ), ); // If the player is online kick them (this may fail if the player is a hardcoded // admin; we don't care about that case because hardcoded admins can log on even // if they're on the ban list). let ecs = server.state.ecs(); if let Ok(target_player) = find_uuid(ecs, uuid) { let _ = kick_player(server, (target_player, uuid), &reason); } Ok(()) } } else { Err(action.help_string()) } } fn handle_unban( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let Ok(username) = scan_fmt!(&args, &action.arg_fmt(), String) { let uuid = find_username(server, &username)?; server .editable_settings_mut() .banlist .edit(server.data_dir().as_ref(), |b| { b.remove(&uuid); }); server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!("{} was successfully unbanned", username), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_server_physics( server: &mut Server, client: EcsEntity, _target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(username), enabled_opt) = scan_fmt_some!(&args, &action.arg_fmt(), String, bool) { let uuid = find_username(server, &username)?; let server_force = enabled_opt.unwrap_or(true); let mut player_physics_settings = server.state.ecs().write_resource::(); let entry = player_physics_settings.settings.entry(uuid).or_default(); entry.server_force = server_force; server.notify_client( client, ServerGeneral::server_msg( ChatType::CommandInfo, format!( "Updated physics settings for {} ({}): {:?}", username, uuid, entry ), ), ); Ok(()) } else { Err(action.help_string()) } } fn handle_apply_buff( server: &mut Server, _client: EcsEntity, target: EcsEntity, args: String, action: &ChatCommand, ) -> CmdResult<()> { if let (Some(buff), strength, duration) = scan_fmt_some!(&args, &action.arg_fmt(), String, f32, f64) { let strength = strength.unwrap_or(0.01); let duration = Duration::from_secs_f64(duration.unwrap_or(1.0)); let buffdata = BuffData::new(strength, Some(duration)); if buff != "all" { cast_buff(&buff, buffdata, server, target) } else { for kind in BUFF_PACK.iter() { cast_buff(kind, buffdata, server, target)?; } Ok(()) } } else { Err(action.help_string()) } } fn cast_buff(kind: &str, data: BuffData, server: &mut Server, target: EcsEntity) -> CmdResult<()> { if let Some(buffkind) = parse_buffkind(kind) { let ecs = &server.state.ecs(); let mut buffs_all = ecs.write_storage::(); if let Some(mut buffs) = buffs_all.get_mut(target) { buffs.insert(Buff::new(buffkind, data, vec![], BuffSource::Command)); } Ok(()) } else { Err(format!("unknown buff: {}", kind)) } } fn parse_buffkind(buff: &str) -> Option { BUFF_PARSER.get(buff).copied() }