You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

275 lines
9.9 KiB

3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
  1. /* TODO:
  2. 1. test to check if sjs/lame/limiter drop with Some() kill counter enabled
  3. 2. test to make sure other items drop with None kill counter
  4. 3. test kill counters get incremented per kill
  5. 4. test unsealing item:
  6. - client item id does not change
  7. - unsealed item no longer has kill counter
  8. 5. test reject unsealing item if not enough kills (can this even happen?)
  9. */
  10. use elseware::common::serverstate::{ClientId, ServerState};
  11. use elseware::entity::gateway::{EntityGateway, InMemoryGateway};
  12. use elseware::ship::ship::{ShipServerState, RecvShipPacket, SendShipPacket};
  13. use elseware::entity::character::SectionID;
  14. use elseware::ship::room::Difficulty;
  15. use elseware::ship::monster::MonsterType;
  16. use elseware::entity::item;
  17. use libpso::packet::ship::*;
  18. use libpso::packet::messages::*;
  19. #[path = "common.rs"]
  20. mod common;
  21. use common::*;
  22. #[async_std::test]
  23. async fn test_sjs_drops_with_kill_counter() {
  24. let mut entity_gateway = InMemoryGateway::default();
  25. let (_user1, mut char1) = new_user_character_with_sid(&mut entity_gateway, "a1", "a", SectionID::Skyly).await;
  26. char1.exp = 80000000;
  27. entity_gateway.save_character(&char1).await.unwrap();
  28. let mut ship = Box::new(ShipServerState::builder()
  29. .gateway(entity_gateway.clone())
  30. .build());
  31. log_in_char(&mut ship, ClientId(1), "a1", "a").await;
  32. join_lobby(&mut ship, ClientId(1)).await;
  33. create_ep2_room_with_difficulty(&mut ship, ClientId(1), "room", "", Difficulty::Ultimate).await;
  34. let room = ship.blocks.0[0].rooms[0].as_mut().unwrap();
  35. room.toggle_redbox_mode(); // enable redbox mode
  36. let gigue_id = room.maps.get_enemy_id_by_monster_type(MonsterType::GiGue).unwrap();
  37. let packets = ship.handle(ClientId(1), &RecvShipPacket::DirectMessage(DirectMessage::new(0, GameMessage::RequestItem(RequestItem {
  38. client: 0,
  39. target: 0,
  40. map_area: 9, // seaside
  41. pt_index: 55, // gigue ? (taken from ingame logs)
  42. enemy_id: gigue_id,
  43. x: 0.0,
  44. y: 0.0,
  45. z: 0.0,
  46. })))).await.unwrap().collect::<Vec<_>>(); // this should return 1 packet (ItemDrop)?
  47. assert!(packets.len() == 1);
  48. match &packets[0].1 {
  49. SendShipPacket::Message(Message {msg: GameMessage::ItemDrop(item_drop)}) => {
  50. assert_eq!(item_drop.item_bytes[10], 0x80)
  51. }
  52. _ => panic!("SJS didn't drop with the expected value! attr[2] should be 0x80 (128) for 0 kills")
  53. }
  54. }
  55. #[async_std::test]
  56. async fn test_other_weapons_drop_without_kill_counter() {
  57. let mut entity_gateway = InMemoryGateway::default();
  58. let (_user1, mut char1) = new_user_character_with_sid(&mut entity_gateway, "a1", "a", SectionID::Skyly).await;
  59. char1.exp = 80000000;
  60. entity_gateway.save_character(&char1).await.unwrap();
  61. let mut ship = Box::new(ShipServerState::builder()
  62. .gateway(entity_gateway.clone())
  63. .build());
  64. log_in_char(&mut ship, ClientId(1), "a1", "a").await;
  65. join_lobby(&mut ship, ClientId(1)).await;
  66. create_ep2_room_with_difficulty(&mut ship, ClientId(1), "room", "", Difficulty::Ultimate).await;
  67. let room = ship.blocks.0[0].rooms[0].as_mut().unwrap();
  68. room.toggle_redbox_mode(); // enable redbox mode
  69. let enemy_id = room.maps.get_enemy_id_by_monster_type(MonsterType::Hildebear).unwrap();
  70. let packets = ship.handle(ClientId(1), &RecvShipPacket::DirectMessage(DirectMessage::new(0, GameMessage::RequestItem(RequestItem {
  71. client: 0,
  72. target: 0,
  73. map_area: 1, // temple alpha
  74. pt_index: 0, // TODO: this is going to break if pt_index ever gets properly used
  75. enemy_id: enemy_id,
  76. x: 0.0,
  77. y: 0.0,
  78. z: 0.0,
  79. })))).await.unwrap().collect::<Vec<_>>();
  80. assert!(packets.len() == 1);
  81. match &packets[0].1 {
  82. SendShipPacket::Message(Message {msg: GameMessage::ItemDrop(item_drop)}) => {
  83. assert_ne!(item_drop.item_bytes[10], 0x80)
  84. }
  85. _ => panic!("Weapon didn't drop with the expected value! attr[2] should be less than 0x80 (128) because it shouldn't have a kill counter!")
  86. }
  87. }
  88. #[async_std::test]
  89. async fn test_all_equipped_kill_counters_increase_per_kill() {
  90. let mut entity_gateway = InMemoryGateway::default();
  91. let (_user1, char1) = new_user_character(&mut entity_gateway, "a1", "a", 1).await;
  92. let mut ship = Box::new(ShipServerState::builder()
  93. .gateway(entity_gateway.clone())
  94. .build());
  95. let mut p1_inv = Vec::new();
  96. p1_inv.push(entity_gateway.create_item(
  97. item::NewItemEntity {
  98. item: item::ItemDetail::Weapon(
  99. item::weapon::Weapon {
  100. weapon: item::weapon::WeaponType::SealedJSword,
  101. grind: 0,
  102. special: None,
  103. attrs: [None,
  104. None,
  105. None,],
  106. tekked: true,
  107. kills: Some(0),
  108. }
  109. ),
  110. }).await.unwrap());
  111. p1_inv.push(entity_gateway.create_item(
  112. item::NewItemEntity {
  113. item: item::ItemDetail::Unit(
  114. item::unit::Unit {
  115. unit: item::unit::UnitType::Limiter,
  116. modifier: None,
  117. kills: Some(0),
  118. }
  119. ),
  120. }).await.unwrap());
  121. let equipped = item::EquippedEntity {
  122. weapon: Some(p1_inv[0].id),
  123. armor: None,
  124. shield: None,
  125. unit: [Some(p1_inv[1].id), None, None, None],
  126. mag: None,
  127. };
  128. entity_gateway.set_character_equips(&char1.id, &equipped).await.unwrap();
  129. entity_gateway.set_character_inventory(&char1.id, &item::InventoryEntity::new(p1_inv)).await.unwrap();
  130. log_in_char(&mut ship, ClientId(1), "a1", "a").await;
  131. join_lobby(&mut ship, ClientId(1)).await;
  132. create_room(&mut ship, ClientId(1), "room", "").await;
  133. let enemy_id = {
  134. let room = ship.blocks.0[0].rooms[0].as_ref().unwrap();
  135. let enemy_id = (0..).filter_map(|i| {
  136. room.maps.enemy_by_id(i).ok().and_then(|enemy| {
  137. if enemy.monster == MonsterType::Booma {
  138. Some(i)
  139. }
  140. else {
  141. None
  142. }
  143. })
  144. }).next().unwrap();
  145. enemy_id
  146. };
  147. ship.handle(ClientId(1), &RecvShipPacket::Message(Message::new(GameMessage::KillMonster(KillMonster{
  148. client: enemy_id as u8,
  149. target: 16,
  150. map_area: 1,
  151. data: [8,0],
  152. })))).await.unwrap().for_each(drop);
  153. let equipped_items = entity_gateway.get_character_equips(&char1.id).await.unwrap();
  154. let inventory = entity_gateway.get_character_inventory(&char1.id).await.unwrap();
  155. let w = inventory.items.iter().find(|x| x.individual().unwrap().id == equipped_items.weapon.unwrap()).unwrap().individual().unwrap();
  156. let u = inventory.items.iter().find(|x| x.individual().unwrap().id == equipped_items.unit[0].unwrap()).unwrap().individual().unwrap();
  157. assert!(w.item.as_client_bytes()[11] == 1);
  158. assert!(u.item.as_client_bytes()[11] == 1);
  159. }
  160. #[async_std::test]
  161. async fn test_non_equipped_kill_counter_does_not_increase() {
  162. let mut entity_gateway = InMemoryGateway::default();
  163. let (_user1, char1) = new_user_character(&mut entity_gateway, "a1", "a", 1).await;
  164. let mut ship = Box::new(ShipServerState::builder()
  165. .gateway(entity_gateway.clone())
  166. .build());
  167. let mut p1_inv = Vec::new();
  168. p1_inv.push(entity_gateway.create_item(
  169. item::NewItemEntity {
  170. item: item::ItemDetail::Weapon(
  171. item::weapon::Weapon {
  172. weapon: item::weapon::WeaponType::SealedJSword,
  173. grind: 0,
  174. special: None,
  175. attrs: [None,
  176. None,
  177. None,],
  178. tekked: true,
  179. kills: Some(0),
  180. }
  181. ),
  182. }).await.unwrap());
  183. p1_inv.push(entity_gateway.create_item(
  184. item::NewItemEntity {
  185. item: item::ItemDetail::Unit(
  186. item::unit::Unit {
  187. unit: item::unit::UnitType::Limiter,
  188. modifier: None,
  189. kills: Some(0),
  190. }
  191. ),
  192. }).await.unwrap());
  193. let equipped = item::EquippedEntity {
  194. weapon: Some(p1_inv[0].id),
  195. armor: None,
  196. shield: None,
  197. unit: [None; 4],
  198. mag: None,
  199. };
  200. entity_gateway.set_character_equips(&char1.id, &equipped).await.unwrap();
  201. entity_gateway.set_character_inventory(&char1.id, &item::InventoryEntity::new(p1_inv)).await.unwrap();
  202. log_in_char(&mut ship, ClientId(1), "a1", "a").await;
  203. join_lobby(&mut ship, ClientId(1)).await;
  204. create_room(&mut ship, ClientId(1), "room", "").await;
  205. let enemy_id = {
  206. let room = ship.blocks.0[0].rooms[0].as_ref().unwrap();
  207. let enemy_id = (0..).filter_map(|i| {
  208. room.maps.enemy_by_id(i).ok().and_then(|enemy| {
  209. if enemy.monster == MonsterType::Booma {
  210. Some(i)
  211. }
  212. else {
  213. None
  214. }
  215. })
  216. }).next().unwrap();
  217. enemy_id
  218. };
  219. ship.handle(ClientId(1), &RecvShipPacket::Message(Message::new(GameMessage::KillMonster(KillMonster{
  220. client: enemy_id as u8,
  221. target: 16,
  222. map_area: 1,
  223. data: [8,0],
  224. })))).await.unwrap().for_each(drop);
  225. let equipped_items = entity_gateway.get_character_equips(&char1.id).await.unwrap();
  226. let inventory = entity_gateway.get_character_inventory(&char1.id).await.unwrap();
  227. let w = inventory.items.iter().find(|x| x.individual().unwrap().id == equipped_items.weapon.unwrap()).unwrap().individual().unwrap();
  228. let u = inventory.items.iter().find(|x| x.individual().unwrap().id == item::ItemEntityId(2)).unwrap().individual().unwrap();
  229. assert!(w.item.as_client_bytes()[11] == 1);
  230. assert!(u.item.as_client_bytes()[11] == 0);
  231. }