OfflineTradersTable.java 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. /*
  2. * Copyright © 2004-2019 L2J Server
  3. *
  4. * This file is part of L2J Server.
  5. *
  6. * L2J Server is free software: you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation, either version 3 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * L2J Server is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. */
  19. package com.l2jserver.gameserver.data.sql.impl;
  20. import java.util.Calendar;
  21. import java.util.logging.Level;
  22. import java.util.logging.Logger;
  23. import com.l2jserver.gameserver.config.Config;
  24. import com.l2jserver.commons.database.ConnectionFactory;
  25. import com.l2jserver.gameserver.LoginServerThread;
  26. import com.l2jserver.gameserver.enums.PrivateStoreType;
  27. import com.l2jserver.gameserver.model.L2ManufactureItem;
  28. import com.l2jserver.gameserver.model.L2World;
  29. import com.l2jserver.gameserver.model.TradeItem;
  30. import com.l2jserver.gameserver.model.actor.instance.L2PcInstance;
  31. import com.l2jserver.gameserver.network.L2GameClient;
  32. import com.l2jserver.gameserver.network.L2GameClient.GameClientState;
  33. public class OfflineTradersTable {
  34. private static final Logger LOGGER = Logger.getLogger(OfflineTradersTable.class.getName());
  35. private static final String SAVE_OFFLINE_STATUS = "INSERT INTO character_offline_trade (`charId`,`time`,`type`,`title`) VALUES (?,?,?,?)";
  36. private static final String SAVE_ITEMS = "INSERT INTO character_offline_trade_items (`charId`,`item`,`count`,`price`) VALUES (?,?,?,?)";
  37. private static final String CLEAR_OFFLINE_TABLE = "DELETE FROM character_offline_trade";
  38. private static final String CLEAR_OFFLINE_TABLE_ITEMS = "DELETE FROM character_offline_trade_items";
  39. private static final String LOAD_OFFLINE_STATUS = "SELECT * FROM character_offline_trade";
  40. private static final String LOAD_OFFLINE_ITEMS = "SELECT * FROM character_offline_trade_items WHERE charId = ?";
  41. public void storeOffliners() {
  42. try (var con = ConnectionFactory.getInstance().getConnection();
  43. var stm1 = con.prepareStatement(CLEAR_OFFLINE_TABLE);
  44. var stm2 = con.prepareStatement(CLEAR_OFFLINE_TABLE_ITEMS);
  45. var stm3 = con.prepareStatement(SAVE_OFFLINE_STATUS);
  46. var stm_items = con.prepareStatement(SAVE_ITEMS)) {
  47. stm1.execute();
  48. stm2.execute();
  49. con.setAutoCommit(false); // avoid halfway done
  50. for (L2PcInstance pc : L2World.getInstance().getPlayers()) {
  51. try {
  52. if ((pc.getPrivateStoreType() != PrivateStoreType.NONE) && pc.isInOfflineMode()) {
  53. stm3.setInt(1, pc.getObjectId()); // Char Id
  54. stm3.setLong(2, pc.getOfflineStartTime());
  55. stm3.setInt(3, pc.getPrivateStoreType().getId()); // store type
  56. String title = null;
  57. switch (pc.getPrivateStoreType()) {
  58. case BUY:
  59. if (!Config.OFFLINE_TRADE_ENABLE) {
  60. continue;
  61. }
  62. title = pc.getBuyList().getTitle();
  63. for (TradeItem i : pc.getBuyList().getItems()) {
  64. stm_items.setInt(1, pc.getObjectId());
  65. stm_items.setInt(2, i.getItem().getId());
  66. stm_items.setLong(3, i.getCount());
  67. stm_items.setLong(4, i.getPrice());
  68. stm_items.executeUpdate();
  69. stm_items.clearParameters();
  70. }
  71. break;
  72. case SELL:
  73. case PACKAGE_SELL:
  74. if (!Config.OFFLINE_TRADE_ENABLE) {
  75. continue;
  76. }
  77. title = pc.getSellList().getTitle();
  78. for (TradeItem i : pc.getSellList().getItems()) {
  79. stm_items.setInt(1, pc.getObjectId());
  80. stm_items.setInt(2, i.getObjectId());
  81. stm_items.setLong(3, i.getCount());
  82. stm_items.setLong(4, i.getPrice());
  83. stm_items.executeUpdate();
  84. stm_items.clearParameters();
  85. }
  86. break;
  87. case MANUFACTURE:
  88. if (!Config.OFFLINE_CRAFT_ENABLE) {
  89. continue;
  90. }
  91. title = pc.getStoreName();
  92. for (L2ManufactureItem i : pc.getManufactureItems().values()) {
  93. stm_items.setInt(1, pc.getObjectId());
  94. stm_items.setInt(2, i.getRecipeId());
  95. stm_items.setLong(3, 0);
  96. stm_items.setLong(4, i.getCost());
  97. stm_items.executeUpdate();
  98. stm_items.clearParameters();
  99. }
  100. }
  101. stm3.setString(4, title);
  102. stm3.executeUpdate();
  103. stm3.clearParameters();
  104. con.commit(); // flush
  105. }
  106. } catch (Exception e) {
  107. LOGGER.log(Level.WARNING, getClass().getSimpleName() + ": Error while saving offline trader: " + pc.getObjectId() + " " + e, e);
  108. }
  109. }
  110. LOGGER.info(getClass().getSimpleName() + ": Offline traders stored.");
  111. } catch (Exception e) {
  112. LOGGER.log(Level.WARNING, getClass().getSimpleName() + ": Error while saving offline traders: " + e, e);
  113. }
  114. }
  115. public void restoreOfflineTraders() {
  116. LOGGER.info(getClass().getSimpleName() + ": Loading offline traders...");
  117. int nTraders = 0;
  118. try (var con = ConnectionFactory.getInstance().getConnection();
  119. var stm = con.createStatement();
  120. var rs = stm.executeQuery(LOAD_OFFLINE_STATUS)) {
  121. while (rs.next()) {
  122. long time = rs.getLong("time");
  123. if (Config.OFFLINE_MAX_DAYS > 0) {
  124. Calendar cal = Calendar.getInstance();
  125. cal.setTimeInMillis(time);
  126. cal.add(Calendar.DAY_OF_YEAR, Config.OFFLINE_MAX_DAYS);
  127. if (cal.getTimeInMillis() <= System.currentTimeMillis()) {
  128. continue;
  129. }
  130. }
  131. PrivateStoreType type = PrivateStoreType.findById(rs.getInt("type"));
  132. if (type == null) {
  133. LOGGER.warning(getClass().getSimpleName() + ": PrivateStoreType with id " + rs.getInt("type") + " could not be found.");
  134. continue;
  135. }
  136. if (type == PrivateStoreType.NONE) {
  137. continue;
  138. }
  139. L2PcInstance player = null;
  140. try {
  141. L2GameClient client = new L2GameClient(null);
  142. client.setDetached(true);
  143. player = L2PcInstance.load(rs.getInt("charId"));
  144. client.setActiveChar(player);
  145. player.setOnlineStatus(true, false);
  146. client.setAccountName(player.getAccountNamePlayer());
  147. L2World.getInstance().addPlayerToWorld(player);
  148. client.setState(GameClientState.IN_GAME);
  149. player.setClient(client);
  150. player.setOfflineStartTime(time);
  151. player.spawnMe(player.getX(), player.getY(), player.getZ());
  152. LoginServerThread.getInstance().addGameServerLogin(player.getAccountName(), client);
  153. try (var stm_items = con.prepareStatement(LOAD_OFFLINE_ITEMS)) {
  154. stm_items.setInt(1, player.getObjectId());
  155. try (var items = stm_items.executeQuery()) {
  156. switch (type) {
  157. case BUY:
  158. while (items.next()) {
  159. if (player.getBuyList().addItemByItemId(items.getInt(2), items.getLong(3), items.getLong(4)) == null) {
  160. throw new NullPointerException();
  161. }
  162. }
  163. player.getBuyList().setTitle(rs.getString("title"));
  164. break;
  165. case SELL:
  166. case PACKAGE_SELL:
  167. while (items.next()) {
  168. if (player.getSellList().addItem(items.getInt(2), items.getLong(3), items.getLong(4)) == null) {
  169. throw new NullPointerException();
  170. }
  171. }
  172. player.getSellList().setTitle(rs.getString("title"));
  173. player.getSellList().setPackaged(type == PrivateStoreType.PACKAGE_SELL);
  174. break;
  175. case MANUFACTURE:
  176. while (items.next()) {
  177. player.getManufactureItems().put(items.getInt(2), new L2ManufactureItem(items.getInt(2), items.getLong(4)));
  178. }
  179. player.setStoreName(rs.getString("title"));
  180. break;
  181. }
  182. }
  183. }
  184. player.sitDown();
  185. if (Config.OFFLINE_SET_NAME_COLOR) {
  186. player.getAppearance().setNameColor(Config.OFFLINE_NAME_COLOR);
  187. }
  188. player.setPrivateStoreType(type);
  189. player.setOnlineStatus(true, true);
  190. player.restoreEffects();
  191. player.broadcastUserInfo();
  192. nTraders++;
  193. } catch (Exception e) {
  194. LOGGER.log(Level.WARNING, getClass().getSimpleName() + ": Error loading trader: " + player, e);
  195. if (player != null) {
  196. player.deleteMe();
  197. }
  198. }
  199. }
  200. LOGGER.info(getClass().getSimpleName() + ": Loaded: " + nTraders + " offline trader(s)");
  201. try (var stm1 = con.createStatement()) {
  202. stm1.execute(CLEAR_OFFLINE_TABLE);
  203. stm1.execute(CLEAR_OFFLINE_TABLE_ITEMS);
  204. }
  205. } catch (Exception e) {
  206. LOGGER.log(Level.WARNING, getClass().getSimpleName() + ": Error while loading offline traders: ", e);
  207. }
  208. }
  209. /**
  210. * Gets the single instance of OfflineTradersTable.
  211. * @return single instance of OfflineTradersTable
  212. */
  213. public static OfflineTradersTable getInstance() {
  214. return SingletonHolder._instance;
  215. }
  216. private static class SingletonHolder {
  217. protected static final OfflineTradersTable _instance = new OfflineTradersTable();
  218. }
  219. }