2021-02-13 23:32:55 +00:00
|
|
|
use crate::Server;
|
2021-02-11 19:35:36 +00:00
|
|
|
use common::{
|
2021-02-13 23:32:55 +00:00
|
|
|
comp::inventory::Inventory,
|
|
|
|
trade::{PendingTrade, TradeAction, TradeId, TradeResult, Trades},
|
2021-02-11 19:35:36 +00:00
|
|
|
};
|
|
|
|
use common_net::{msg::ServerGeneral, sync::WorldSyncExt};
|
2021-02-13 23:32:55 +00:00
|
|
|
use hashbrown::hash_map::Entry;
|
2021-02-11 19:35:36 +00:00
|
|
|
use specs::{world::WorldExt, Entity as EcsEntity};
|
2021-02-12 20:47:45 +00:00
|
|
|
use std::cmp::Ordering;
|
2021-02-13 23:32:55 +00:00
|
|
|
use tracing::{error, trace};
|
2021-02-11 19:35:36 +00:00
|
|
|
|
2021-02-12 20:47:45 +00:00
|
|
|
/// Invoked when the trade UI is up, handling item changes, accepts, etc
|
2021-02-12 02:53:25 +00:00
|
|
|
pub fn handle_process_trade_action(
|
|
|
|
server: &mut Server,
|
|
|
|
entity: EcsEntity,
|
2021-02-13 23:32:55 +00:00
|
|
|
trade_id: TradeId,
|
|
|
|
action: TradeAction,
|
2021-02-12 02:53:25 +00:00
|
|
|
) {
|
2021-02-11 19:35:36 +00:00
|
|
|
if let Some(uid) = server.state.ecs().uid_from_entity(entity) {
|
|
|
|
let mut trades = server.state.ecs().write_resource::<Trades>();
|
2021-02-13 23:32:55 +00:00
|
|
|
if let TradeAction::Decline = action {
|
2021-02-11 19:35:36 +00:00
|
|
|
let to_notify = trades.decline_trade(trade_id, uid);
|
|
|
|
to_notify
|
|
|
|
.and_then(|u| server.state.ecs().entity_from_uid(u.0))
|
2021-02-12 23:09:18 +00:00
|
|
|
.map(|e| {
|
|
|
|
server.notify_client(e, ServerGeneral::FinishedTrade(TradeResult::Declined))
|
|
|
|
});
|
2021-02-11 19:35:36 +00:00
|
|
|
} else {
|
2021-02-12 10:51:32 +00:00
|
|
|
if let Some(inv) = server.state.ecs().read_component::<Inventory>().get(entity) {
|
2021-02-13 23:32:55 +00:00
|
|
|
trades.process_trade_action(trade_id, uid, action, inv);
|
2021-02-12 10:51:32 +00:00
|
|
|
}
|
2021-02-13 23:32:55 +00:00
|
|
|
if let Entry::Occupied(entry) = trades.trades.entry(trade_id) {
|
|
|
|
let parties = entry.get().parties;
|
|
|
|
let msg = if entry.get().should_commit() {
|
|
|
|
let result = commit_trade(server.state.ecs(), entry.get());
|
|
|
|
entry.remove();
|
|
|
|
ServerGeneral::FinishedTrade(result)
|
|
|
|
} else {
|
|
|
|
ServerGeneral::UpdatePendingTrade(trade_id, entry.get().clone())
|
|
|
|
};
|
2021-02-12 10:51:32 +00:00
|
|
|
// send the updated state to both parties
|
2021-02-13 23:32:55 +00:00
|
|
|
for party in parties.iter() {
|
2021-02-12 23:09:18 +00:00
|
|
|
server
|
|
|
|
.state
|
|
|
|
.ecs()
|
|
|
|
.entity_from_uid(party.0)
|
|
|
|
.map(|e| server.notify_client(e, msg.clone()));
|
2021-02-11 19:35:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-12 20:47:45 +00:00
|
|
|
|
2021-02-12 23:09:18 +00:00
|
|
|
/// Commit a trade that both parties have agreed to, modifying their respective
|
|
|
|
/// inventories
|
2021-02-12 20:47:45 +00:00
|
|
|
fn commit_trade(ecs: &specs::World, trade: &PendingTrade) -> TradeResult {
|
2021-02-13 23:32:55 +00:00
|
|
|
let mut entities = Vec::new();
|
|
|
|
for party in trade.parties.iter() {
|
|
|
|
match ecs.entity_from_uid(party.0) {
|
2021-02-12 20:47:45 +00:00
|
|
|
Some(entity) => entities.push(entity),
|
|
|
|
None => return TradeResult::Declined,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let mut inventories = ecs.write_component::<Inventory>();
|
2021-02-13 23:32:55 +00:00
|
|
|
for entity in entities.iter() {
|
|
|
|
if inventories.get_mut(*entity).is_none() {
|
2021-02-12 20:47:45 +00:00
|
|
|
return TradeResult::Declined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let invmsg = "inventories.get_mut(entities[who]).is_none() should have returned already";
|
|
|
|
trace!("committing trade: {:?}", trade);
|
2021-02-12 23:09:18 +00:00
|
|
|
// Compute the net change in slots of each player during the trade, to detect
|
|
|
|
// out-of-space-ness before transferring any items
|
2021-02-12 20:47:45 +00:00
|
|
|
let mut delta_slots: [isize; 2] = [0, 0];
|
|
|
|
for who in [0, 1].iter().cloned() {
|
|
|
|
for (slot, quantity) in trade.offers[who].iter() {
|
|
|
|
let inventory = inventories.get_mut(entities[who]).expect(invmsg);
|
|
|
|
let item = match inventory.get(*slot) {
|
|
|
|
Some(item) => item,
|
|
|
|
None => {
|
2021-02-12 23:09:18 +00:00
|
|
|
error!(
|
|
|
|
"PendingTrade invariant violation in trade {:?}: slots offered in a trade \
|
|
|
|
should be non-empty",
|
|
|
|
trade
|
|
|
|
);
|
2021-02-12 20:47:45 +00:00
|
|
|
return TradeResult::Declined;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
match item.amount().cmp(&quantity) {
|
|
|
|
Ordering::Less => {
|
2021-02-12 23:09:18 +00:00
|
|
|
error!(
|
|
|
|
"PendingTrade invariant violation in trade {:?}: party {} offered more of \
|
|
|
|
an item than they have",
|
|
|
|
trade, who
|
|
|
|
);
|
2021-02-12 20:47:45 +00:00
|
|
|
return TradeResult::Declined;
|
|
|
|
},
|
|
|
|
Ordering::Equal => {
|
|
|
|
delta_slots[who] -= 1; // exact, removes the whole stack
|
2021-02-12 23:09:18 +00:00
|
|
|
delta_slots[1 - who] += 1; // overapproximation, assumes the stack won't merge
|
2021-02-12 20:47:45 +00:00
|
|
|
},
|
|
|
|
Ordering::Greater => {
|
2021-02-13 23:32:55 +00:00
|
|
|
// No change to delta_slots[who], since they have leftovers
|
2021-02-12 23:09:18 +00:00
|
|
|
delta_slots[1 - who] += 1; // overapproximation, assumes the stack won't merge
|
2021-02-12 20:47:45 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
trace!("delta_slots: {:?}", delta_slots);
|
|
|
|
for who in [0, 1].iter().cloned() {
|
2021-02-12 23:09:18 +00:00
|
|
|
// Inventories should never exceed 2^{63} slots, so the usize -> isize
|
|
|
|
// conversions here should be safe
|
2021-02-12 20:47:45 +00:00
|
|
|
let inv = inventories.get_mut(entities[who]).expect(invmsg);
|
|
|
|
if inv.populated_slots() as isize + delta_slots[who] > inv.capacity() as isize {
|
|
|
|
return TradeResult::NotEnoughSpace;
|
|
|
|
}
|
|
|
|
}
|
2021-02-13 23:32:55 +00:00
|
|
|
let mut items = [Vec::new(), Vec::new()];
|
2021-02-12 20:47:45 +00:00
|
|
|
for who in [0, 1].iter().cloned() {
|
|
|
|
for (slot, quantity) in trade.offers[who].iter() {
|
2021-02-13 23:32:55 +00:00
|
|
|
// Take the items one by one, to benefit from Inventory's stack handling
|
2021-02-12 20:47:45 +00:00
|
|
|
for _ in 0..*quantity {
|
2021-02-12 23:09:18 +00:00
|
|
|
inventories
|
|
|
|
.get_mut(entities[who])
|
|
|
|
.expect(invmsg)
|
|
|
|
.take(*slot)
|
|
|
|
.map(|item| items[who].push(item));
|
2021-02-12 20:47:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for who in [0, 1].iter().cloned() {
|
2021-02-12 23:09:18 +00:00
|
|
|
if let Err(leftovers) = inventories
|
|
|
|
.get_mut(entities[1 - who])
|
|
|
|
.expect(invmsg)
|
|
|
|
.push_all(items[who].drain(..))
|
|
|
|
{
|
2021-02-13 23:32:55 +00:00
|
|
|
// This should only happen if the arithmetic above for delta_slots says there's
|
2021-02-12 23:09:18 +00:00
|
|
|
// enough space and there isn't (i.e. underapproximates)
|
2021-02-13 23:32:55 +00:00
|
|
|
panic!(
|
|
|
|
"Not enough space for all the items, leftovers are {:?}",
|
2021-02-12 23:09:18 +00:00
|
|
|
leftovers
|
|
|
|
);
|
2021-02-12 20:47:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
TradeResult::Completed
|
|
|
|
}
|