/* This file is part of the OdinMS Maple Story Server Copyright (C) 2008 Patrick Huy Matthias Butz Jan Christian Meyer Copyleft (L) 2016 - 2019 RonanLana (HeavenMS) This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation version 3 as published by the Free Software Foundation. You may not use, modify or distribute this program under any other version of the GNU Affero General Public License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package client.processor.npc; import client.Character; import client.Client; import client.autoban.AutobanFactory; import client.inventory.Inventory; import client.inventory.InventoryType; import client.inventory.Item; import client.inventory.ItemFactory; import client.inventory.manipulator.InventoryManipulator; import client.inventory.manipulator.KarmaManipulator; import config.YamlConfig; import constants.id.ItemId; import constants.inventory.ItemConstants; import net.server.channel.Channel; import server.DueyPackage; import server.ItemInformationProvider; import server.Trade; import tools.DatabaseConnection; import tools.FilePrinter; import tools.PacketCreator; import tools.Pair; import java.sql.*; import java.util.Calendar; import java.util.Collections; import java.util.LinkedList; import java.util.List; /** * @author RonanLana - synchronization of Duey modules */ public class DueyProcessor { public enum Actions { TOSERVER_RECV_ITEM(0x00), TOSERVER_SEND_ITEM(0x02), TOSERVER_CLAIM_PACKAGE(0x04), TOSERVER_REMOVE_PACKAGE(0x05), TOSERVER_CLOSE_DUEY(0x07), TOCLIENT_OPEN_DUEY(0x08), TOCLIENT_SEND_ENABLE_ACTIONS(0x09), TOCLIENT_SEND_NOT_ENOUGH_MESOS(0x0A), TOCLIENT_SEND_INCORRECT_REQUEST(0x0B), TOCLIENT_SEND_NAME_DOES_NOT_EXIST(0x0C), TOCLIENT_SEND_SAMEACC_ERROR(0x0D), TOCLIENT_SEND_RECEIVER_STORAGE_FULL(0x0E), TOCLIENT_SEND_RECEIVER_UNABLE_TO_RECV(0x0F), TOCLIENT_SEND_RECEIVER_STORAGE_WITH_UNIQUE(0x10), TOCLIENT_SEND_MESO_LIMIT(0x11), TOCLIENT_SEND_SUCCESSFULLY_SENT(0x12), TOCLIENT_RECV_UNKNOWN_ERROR(0x13), TOCLIENT_RECV_ENABLE_ACTIONS(0x14), TOCLIENT_RECV_NO_FREE_SLOTS(0x15), TOCLIENT_RECV_RECEIVER_WITH_UNIQUE(0x16), TOCLIENT_RECV_SUCCESSFUL_MSG(0x17), TOCLIENT_RECV_PACKAGE_MSG(0x1B); final byte code; Actions(int code) { this.code = (byte) code; } public byte getCode() { return code; } } private static Pair getAccountCharacterIdFromCNAME(String name) { Pair ids = null; try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("SELECT id,accountid FROM characters WHERE name = ?")) { ps.setString(1, name); try (ResultSet rs = ps.executeQuery()) { if (rs.next()) { ids = new Pair<>(rs.getInt("accountid"), rs.getInt("id")); } } } catch (SQLException e) { e.printStackTrace(); } return ids; } private static void showDueyNotification(Client c, Character player) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("SELECT SenderName, Type FROM dueypackages WHERE ReceiverId = ? AND Checked = 1 ORDER BY Type DESC")) { ps.setInt(1, player.getId()); try (ResultSet rs = ps.executeQuery()) { if (rs.next()) { try (PreparedStatement ps2 = con.prepareStatement("UPDATE dueypackages SET Checked = 0 where ReceiverId = ?")) { ps2.setInt(1, player.getId()); ps2.executeUpdate(); c.sendPacket(PacketCreator.sendDueyParcelReceived(rs.getString("SenderName"), rs.getInt("Type") == 1)); } } } } catch (SQLException e) { e.printStackTrace(); } } private static void deletePackageFromInventoryDB(Connection con, int packageId) throws SQLException { ItemFactory.DUEY.saveItems(new LinkedList<>(), packageId, con); } private static void removePackageFromDB(int packageId) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("DELETE FROM dueypackages WHERE PackageId = ?")) { ps.setInt(1, packageId); ps.executeUpdate(); deletePackageFromInventoryDB(con, packageId); } catch (SQLException e) { e.printStackTrace(); } } private static DueyPackage getPackageFromDB(ResultSet rs) { try { int packageId = rs.getInt("PackageId"); List> dueyItems = ItemFactory.DUEY.loadItems(packageId, false); DueyPackage dueypack; if (!dueyItems.isEmpty()) { // in a duey package there's only one item dueypack = new DueyPackage(packageId, dueyItems.get(0).getLeft()); } else { dueypack = new DueyPackage(packageId); } dueypack.setSender(rs.getString("SenderName")); dueypack.setMesos(rs.getInt("Mesos")); dueypack.setSentTime(rs.getTimestamp("TimeStamp"), rs.getBoolean("Type")); dueypack.setMessage(rs.getString("Message")); return dueypack; } catch (SQLException sqle) { sqle.printStackTrace(); return null; } } private static List loadPackages(Character chr) { List packages = new LinkedList<>(); try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("SELECT * FROM dueypackages dp WHERE ReceiverId = ?")) { ps.setInt(1, chr.getId()); try (ResultSet rs = ps.executeQuery()) { while (rs.next()) { DueyPackage dueypack = getPackageFromDB(rs); if (dueypack == null) { continue; } packages.add(dueypack); } } } catch (SQLException e) { e.printStackTrace(); } return packages; } private static int createPackage(int mesos, String message, String sender, int toCid, boolean quick) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("INSERT INTO `dueypackages` (ReceiverId, SenderName, Mesos, TimeStamp, Message, Type, Checked) VALUES (?, ?, ?, ?, ?, ?, 1)", Statement.RETURN_GENERATED_KEYS)) { ps.setInt(1, toCid); ps.setString(2, sender); ps.setInt(3, mesos); ps.setTimestamp(4, new Timestamp(System.currentTimeMillis())); ps.setString(5, message); ps.setInt(6, quick ? 1 : 0); int updateRows = ps.executeUpdate(); if (updateRows < 1) { FilePrinter.printError(FilePrinter.INSERT_CHAR, "Error trying to create package [mesos: " + mesos + ", " + sender + ", quick: " + quick + ", to CharacterId: " + toCid + "]"); return -1; } final int packageId; try (ResultSet rs = ps.getGeneratedKeys()) { if (rs.next()) { packageId = rs.getInt(1); } else { FilePrinter.printError(FilePrinter.INSERT_CHAR, "Failed inserting package [mesos: " + mesos + ", " + sender + ", quick: " + quick + ", to CharacterId: " + toCid + "]"); return -1; } } return packageId; } catch (SQLException sqle) { sqle.printStackTrace(); } return -1; } private static boolean insertPackageItem(int packageId, Item item) { Pair dueyItem = new Pair<>(item, InventoryType.getByType(item.getItemType())); try (Connection con = DatabaseConnection.getConnection()) { ItemFactory.DUEY.saveItems(Collections.singletonList(dueyItem), packageId, con); return true; } catch (SQLException sqle) { sqle.printStackTrace(); } return false; } private static int addPackageItemFromInventory(int packageId, Client c, byte invTypeId, short itemPos, short amount) { if (invTypeId > 0) { ItemInformationProvider ii = ItemInformationProvider.getInstance(); InventoryType invType = InventoryType.getByType(invTypeId); Inventory inv = c.getPlayer().getInventory(invType); Item item; inv.lockInventory(); try { item = inv.getItem(itemPos); if (item != null && item.getQuantity() >= amount) { if (item.isUntradeable() || ii.isUnmerchable(item.getItemId())) { return -1; } if (ItemConstants.isRechargeable(item.getItemId())) { InventoryManipulator.removeFromSlot(c, invType, itemPos, item.getQuantity(), true); } else { InventoryManipulator.removeFromSlot(c, invType, itemPos, amount, true, false); } item = item.copy(); } else { return -2; } } finally { inv.unlockInventory(); } KarmaManipulator.toggleKarmaFlagToUntradeable(item); item.setQuantity(amount); if (!insertPackageItem(packageId, item)) { return 1; } } return 0; } public static void dueySendItem(Client c, byte invTypeId, short itemPos, short amount, int sendMesos, String sendMessage, String recipient, boolean quick) { if (c.tryacquireClient()) { try { int fee = Trade.getFee(sendMesos); if (!quick) { fee += 5000; } else if (!c.getPlayer().haveItem(ItemId.QUICK_DELIVERY_TICKET)) { AutobanFactory.PACKET_EDIT.alert(c.getPlayer(), c.getPlayer().getName() + " tried to packet edit with Quick Delivery on duey."); FilePrinter.printError(FilePrinter.EXPLOITS + c.getPlayer().getName() + ".txt", c.getPlayer().getName() + " tried to use duey with Quick Delivery, mesos " + sendMesos + " and amount " + amount); c.disconnect(true, false); return; } long finalcost = (long) sendMesos + fee; if (finalcost < 0 || finalcost > Integer.MAX_VALUE || (amount < 1 && sendMesos == 0)) { AutobanFactory.PACKET_EDIT.alert(c.getPlayer(), c.getPlayer().getName() + " tried to packet edit with duey."); FilePrinter.printError(FilePrinter.EXPLOITS + c.getPlayer().getName() + ".txt", c.getPlayer().getName() + " tried to use duey with mesos " + sendMesos + " and amount " + amount); c.disconnect(true, false); return; } Pair accIdCid; if (c.getPlayer().getMeso() >= finalcost) { accIdCid = getAccountCharacterIdFromCNAME(recipient); int recipientAccId = accIdCid.getLeft(); if (recipientAccId != -1) { if (recipientAccId == c.getAccID()) { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_SAMEACC_ERROR.getCode())); return; } } else { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_NAME_DOES_NOT_EXIST.getCode())); return; } } else { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_NOT_ENOUGH_MESOS.getCode())); return; } int recipientCid = accIdCid.getRight(); if (recipientCid == -1) { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_NAME_DOES_NOT_EXIST.getCode())); return; } if (quick) { InventoryManipulator.removeById(c, InventoryType.CASH, ItemId.QUICK_DELIVERY_TICKET, (short) 1, false, false); } int packageId = createPackage(sendMesos, sendMessage, c.getPlayer().getName(), recipientCid, quick); if (packageId == -1) { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_ENABLE_ACTIONS.getCode())); return; } c.getPlayer().gainMeso((int) -finalcost, false); int res = addPackageItemFromInventory(packageId, c, invTypeId, itemPos, amount); if (res == 0) { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_SUCCESSFULLY_SENT.getCode())); } else if (res > 0) { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_ENABLE_ACTIONS.getCode())); } else { c.sendPacket(PacketCreator.sendDueyMSG(DueyProcessor.Actions.TOCLIENT_SEND_INCORRECT_REQUEST.getCode())); } Client rClient = null; int channel = c.getWorldServer().find(recipient); if (channel > -1) { Channel rcserv = c.getWorldServer().getChannel(channel); if (rcserv != null) { Character rChr = rcserv.getPlayerStorage().getCharacterByName(recipient); if (rChr != null) { rClient = rChr.getClient(); } } } if (rClient != null && rClient.isLoggedIn() && !rClient.getPlayer().isAwayFromWorld()) { showDueyNotification(rClient, rClient.getPlayer()); } } finally { c.releaseClient(); } } } public static void dueyRemovePackage(Client c, int packageid, boolean playerRemove) { if (c.tryacquireClient()) { try { removePackageFromDB(packageid); c.sendPacket(PacketCreator.removeItemFromDuey(playerRemove, packageid)); } finally { c.releaseClient(); } } } public static void dueyClaimPackage(Client c, int packageId) { if (c.tryacquireClient()) { try { try { DueyPackage dp = null; try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("SELECT * FROM dueypackages dp WHERE PackageId = ?")) { ps.setInt(1, packageId); try (ResultSet rs = ps.executeQuery()) { if (rs.next()) { dp = getPackageFromDB(rs); } } } if (dp == null) { c.sendPacket(PacketCreator.sendDueyMSG(Actions.TOCLIENT_RECV_UNKNOWN_ERROR.getCode())); FilePrinter.printError(FilePrinter.EXPLOITS + c.getPlayer().getName() + ".txt", c.getPlayer().getName() + " tried to receive package from duey with id " + packageId); return; } if (dp.isDeliveringTime()) { c.sendPacket(PacketCreator.sendDueyMSG(Actions.TOCLIENT_RECV_UNKNOWN_ERROR.getCode())); return; } Item dpItem = dp.getItem(); if (dpItem != null) { if (!c.getPlayer().canHoldMeso(dp.getMesos())) { c.sendPacket(PacketCreator.sendDueyMSG(Actions.TOCLIENT_RECV_UNKNOWN_ERROR.getCode())); return; } if (!InventoryManipulator.checkSpace(c, dpItem.getItemId(), dpItem.getQuantity(), dpItem.getOwner())) { int itemid = dpItem.getItemId(); if (ItemInformationProvider.getInstance().isPickupRestricted(itemid) && c.getPlayer().getInventory(ItemConstants.getInventoryType(itemid)).findById(itemid) != null) { c.sendPacket(PacketCreator.sendDueyMSG(Actions.TOCLIENT_RECV_RECEIVER_WITH_UNIQUE.getCode())); } else { c.sendPacket(PacketCreator.sendDueyMSG(Actions.TOCLIENT_RECV_NO_FREE_SLOTS.getCode())); } return; } else { InventoryManipulator.addFromDrop(c, dpItem, false); } } c.getPlayer().gainMeso(dp.getMesos(), false); dueyRemovePackage(c, packageId, false); } catch (SQLException e) { e.printStackTrace(); } } finally { c.releaseClient(); } } } public static void dueySendTalk(Client c, boolean quickDelivery) { if (c.tryacquireClient()) { try { long timeNow = System.currentTimeMillis(); if (timeNow - c.getPlayer().getNpcCooldown() < YamlConfig.config.server.BLOCK_NPC_RACE_CONDT) { c.sendPacket(PacketCreator.enableActions()); return; } c.getPlayer().setNpcCooldown(timeNow); if (quickDelivery) { c.sendPacket(PacketCreator.sendDuey(0x1A, null)); } else { c.sendPacket(PacketCreator.sendDuey(0x8, loadPackages(c.getPlayer()))); } } finally { c.releaseClient(); } } } public static void dueyCreatePackage(Item item, int mesos, String sender, int recipientCid) { int packageId = createPackage(mesos, null, sender, recipientCid, false); if (packageId != -1) { insertPackageItem(packageId, item); } } public static void runDueyExpireSchedule() { Calendar c = Calendar.getInstance(); c.add(Calendar.DATE, -30); final Timestamp ts = new Timestamp(c.getTime().getTime()); try (Connection con = DatabaseConnection.getConnection()) { List toRemove = new LinkedList<>(); try (PreparedStatement ps = con.prepareStatement("SELECT `PackageId` FROM dueypackages WHERE `TimeStamp` < ?")) { ps.setTimestamp(1, ts); try (ResultSet rs = ps.executeQuery()) { while (rs.next()) { toRemove.add(rs.getInt("PackageId")); } } } for (Integer pid : toRemove) { removePackageFromDB(pid); } try (PreparedStatement ps = con.prepareStatement("DELETE FROM dueypackages WHERE `TimeStamp` < ?")) { ps.setTimestamp(1, ts); ps.executeUpdate(); } } catch (SQLException e) { e.printStackTrace(); } } }