/* This file is part of the OdinMS Maple Story Server Copyright (C) 2008 Patrick Huy Matthias Butz Jan Christian Meyer 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 net.server.world; import client.BuddyList; import client.BuddyList.BuddyAddResult; import client.BuddyList.BuddyOperation; import client.BuddylistEntry; import client.MapleCharacter; import client.MapleFamily; import config.YamlConfig; import constants.game.GameConstants; import net.server.PlayerStorage; import net.server.Server; import net.server.audit.LockCollector; import net.server.audit.locks.*; import net.server.audit.locks.factory.MonitoredReadLockFactory; import net.server.audit.locks.factory.MonitoredReentrantLockFactory; import net.server.audit.locks.factory.MonitoredWriteLockFactory; import net.server.channel.Channel; import net.server.channel.CharacterIdChannelPair; import net.server.coordinator.matchchecker.MapleMatchCheckerCoordinator; import net.server.coordinator.partysearch.MaplePartySearchCoordinator; import net.server.coordinator.world.MapleInviteCoordinator; import net.server.coordinator.world.MapleInviteCoordinator.InviteResult; import net.server.coordinator.world.MapleInviteCoordinator.InviteType; import net.server.guild.MapleGuild; import net.server.guild.MapleGuildCharacter; import net.server.guild.MapleGuildSummary; import net.server.services.BaseService; import net.server.services.ServicesManager; import net.server.services.type.WorldServices; import net.server.task.*; import scripting.event.EventInstanceManager; import server.MapleStorage; import server.TimerManager; import server.maps.*; import tools.DatabaseConnection; import tools.MaplePacketCreator; import tools.Pair; import tools.packets.Fishing; import java.sql.*; import java.util.*; import java.util.Map.Entry; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.atomic.AtomicInteger; /** * * @author kevintjuh93 * @author Ronan - thread-oriented (world schedules + guild queue + marriages + party chars) */ public class World { private int id, flag, exprate, droprate, bossdroprate, mesorate, questrate, travelrate, fishingrate; private String eventmsg; private List channels = new ArrayList<>(); private Map pnpcStep = new HashMap<>(); private Map pnpcPodium = new HashMap<>(); private Map messengers = new HashMap<>(); private AtomicInteger runningMessengerId = new AtomicInteger(); private Map families = new LinkedHashMap<>(); private Map relationships = new HashMap<>(); private Map> relationshipCouples = new HashMap<>(); private Map gsStore = new HashMap<>(); private PlayerStorage players = new PlayerStorage(); private ServicesManager services = new ServicesManager(WorldServices.SAVE_CHARACTER); private MapleMatchCheckerCoordinator matchChecker = new MapleMatchCheckerCoordinator(); private MaplePartySearchCoordinator partySearch = new MaplePartySearchCoordinator(); private final MonitoredReentrantReadWriteLock chnLock = new MonitoredReentrantReadWriteLock(MonitoredLockType.WORLD_CHANNELS, true); private MonitoredReadLock chnRLock = MonitoredReadLockFactory.createLock(chnLock); private MonitoredWriteLock chnWLock = MonitoredWriteLockFactory.createLock(chnLock); private Map> accountChars = new HashMap<>(); private Map accountStorages = new HashMap<>(); private MonitoredReentrantLock accountCharsLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_CHARS, true); private Set queuedGuilds = new HashSet<>(); private Map, Pair>> queuedMarriages = new HashMap<>(); private Map> marriageGuests = new ConcurrentHashMap<>(); private Map partyChars = new HashMap<>(); private Map parties = new HashMap<>(); private AtomicInteger runningPartyId = new AtomicInteger(); private MonitoredReentrantLock partyLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_PARTY, true); private Map owlSearched = new LinkedHashMap<>(); private List> cashItemBought = new ArrayList<>(9); private final MonitoredReentrantReadWriteLock suggestLock = new MonitoredReentrantReadWriteLock(MonitoredLockType.WORLD_SUGGEST, true); private MonitoredReadLock suggestRLock = MonitoredReadLockFactory.createLock(suggestLock); private MonitoredWriteLock suggestWLock = MonitoredWriteLockFactory.createLock(suggestLock); private Map disabledServerMessages = new HashMap<>(); // reuse owl lock private MonitoredReentrantLock srvMessagesLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_SRVMESSAGES); private ScheduledFuture srvMessagesSchedule; private MonitoredReentrantLock activePetsLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_PETS, true); private Map activePets = new LinkedHashMap<>(); private ScheduledFuture petsSchedule; private long petUpdate; private MonitoredReentrantLock activeMountsLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_MOUNTS, true); private Map activeMounts = new LinkedHashMap<>(); private ScheduledFuture mountsSchedule; private long mountUpdate; private MonitoredReentrantLock activePlayerShopsLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_PSHOPS, true); private Map activePlayerShops = new LinkedHashMap<>(); private MonitoredReentrantLock activeMerchantsLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_MERCHS, true); private Map> activeMerchants = new LinkedHashMap<>(); private ScheduledFuture merchantSchedule; private long merchantUpdate; private Map registeredTimedMapObjects = new LinkedHashMap<>(); private ScheduledFuture timedMapObjectsSchedule; private MonitoredReentrantLock timedMapObjectLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.WORLD_MAPOBJS, true); private Map fishingAttempters = Collections.synchronizedMap(new WeakHashMap()); private ScheduledFuture charactersSchedule; private ScheduledFuture marriagesSchedule; private ScheduledFuture mapOwnershipSchedule; private ScheduledFuture fishingSchedule; private ScheduledFuture partySearchSchedule; private ScheduledFuture timeoutSchedule; public World(int world, int flag, String eventmsg, int exprate, int droprate, int bossdroprate, int mesorate, int questrate, int travelrate, int fishingrate) { this.id = world; this.flag = flag; this.eventmsg = eventmsg; this.exprate = exprate; this.droprate = droprate; this.bossdroprate = bossdroprate; this.mesorate = mesorate; this.questrate = questrate; this.travelrate = travelrate; this.fishingrate = fishingrate; runningPartyId.set(1000000001); // partyid must not clash with charid to solve update item looting issues, found thanks to Vcoc runningMessengerId.set(1); petUpdate = Server.getInstance().getCurrentTime(); mountUpdate = petUpdate; for (int i = 0; i < 9; i++) { cashItemBought.add(new LinkedHashMap()); } TimerManager tman = TimerManager.getInstance(); petsSchedule = tman.register(new PetFullnessTask(this), 60 * 1000, 60 * 1000); srvMessagesSchedule = tman.register(new ServerMessageTask(this), 10 * 1000, 10 * 1000); mountsSchedule = tman.register(new MountTirednessTask(this), 60 * 1000, 60 * 1000); merchantSchedule = tman.register(new HiredMerchantTask(this), 10 * 60 * 1000, 10 * 60 * 1000); timedMapObjectsSchedule = tman.register(new TimedMapObjectTask(this), 60 * 1000, 60 * 1000); charactersSchedule = tman.register(new CharacterAutosaverTask(this), 60 * 60 * 1000, 60 * 60 * 1000); marriagesSchedule = tman.register(new WeddingReservationTask(this), YamlConfig.config.server.WEDDING_RESERVATION_INTERVAL * 60 * 1000, YamlConfig.config.server.WEDDING_RESERVATION_INTERVAL * 60 * 1000); mapOwnershipSchedule = tman.register(new MapOwnershipTask(this), 20 * 1000, 20 * 1000); fishingSchedule = tman.register(new FishingTask(this), 10 * 1000, 10 * 1000); partySearchSchedule = tman.register(new PartySearchTask(this), 10 * 1000, 10 * 1000); timeoutSchedule = tman.register(new TimeoutTask(this), 10 * 1000, 10 * 1000); if(YamlConfig.config.server.USE_FAMILY_SYSTEM) { long timeLeft = Server.getTimeLeftForNextDay(); FamilyDailyResetTask.resetEntitlementUsage(this); tman.register(new FamilyDailyResetTask(this), 24 * 60 * 60 * 1000, timeLeft); } } public int getChannelsSize() { chnRLock.lock(); try { return channels.size(); } finally { chnRLock.unlock(); } } public List getChannels() { chnRLock.lock(); try { return new ArrayList<>(channels); } finally { chnRLock.unlock(); } } public Channel getChannel(int channel) { chnRLock.lock(); try { try { return channels.get(channel - 1); } catch (IndexOutOfBoundsException e) { return null; } } finally { chnRLock.unlock(); } } public boolean addChannel(Channel channel) { chnWLock.lock(); try { if (channel.getId() == channels.size() + 1) { channels.add(channel); return true; } else { return false; } } finally { chnWLock.unlock(); } } public int removeChannel() { Channel ch; int chIdx; chnRLock.lock(); try { chIdx = channels.size() - 1; if(chIdx < 0) { return -1; } ch = channels.get(chIdx); } finally { chnRLock.unlock(); } if(ch == null || !ch.canUninstall()) { return -1; } chnWLock.lock(); try { if(chIdx == channels.size() - 1) { channels.remove(chIdx); } else { return -1; } } finally { chnWLock.unlock(); } ch.shutdown(); return ch.getId(); } public boolean canUninstall() { if(players.getSize() > 0) return false; for(Channel ch : this.getChannels()) { if(!ch.canUninstall()) { return false; } } return true; } public void setFlag(byte b) { this.flag = b; } public int getFlag() { return flag; } public String getEventMessage() { return eventmsg; } public int getExpRate() { return exprate; } public void setExpRate(int exp) { Collection list = getPlayerStorage().getAllCharacters(); for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.revertWorldRates(); } this.exprate = exp; for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.setWorldRates(); } } public int getDropRate() { return droprate; } public void setDropRate(int drop) { Collection list = getPlayerStorage().getAllCharacters(); for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.revertWorldRates(); } this.droprate = drop; for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.setWorldRates(); } } public int getBossDropRate() { // boss rate concept thanks to Lapeiro return bossdroprate; } public void setBossDropRate(int bossdrop) { bossdroprate = bossdrop; } public int getMesoRate() { return mesorate; } public void setMesoRate(int meso) { Collection list = getPlayerStorage().getAllCharacters(); for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.revertWorldRates(); } this.mesorate = meso; for(MapleCharacter chr : list) { if(!chr.isLoggedin()) continue; chr.setWorldRates(); } } public int getQuestRate() { return questrate; } public void setQuestRate(int quest) { this.questrate = quest; } public int getTravelRate() { return travelrate; } public void setTravelRate(int travel) { this.travelrate = travel; } public int getTransportationTime(int travelTime) { return (int) Math.ceil(travelTime / travelrate); } public int getFishingRate() { return fishingrate; } public void setFishingRate(int quest) { this.fishingrate = quest; } public void loadAccountCharactersView(Integer accountId, List chars) { SortedMap charsMap = new TreeMap<>(); for(MapleCharacter chr : chars) { charsMap.put(chr.getId(), chr); } accountCharsLock.lock(); // accountCharsLock should be used after server's lgnWLock for compliance try { accountChars.put(accountId, charsMap); } finally { accountCharsLock.unlock(); } } public void registerAccountCharacterView(Integer accountId, MapleCharacter chr) { accountCharsLock.lock(); try { accountChars.get(accountId).put(chr.getId(), chr); } finally { accountCharsLock.unlock(); } } public void unregisterAccountCharacterView(Integer accountId, Integer chrId) { accountCharsLock.lock(); try { accountChars.get(accountId).remove(chrId); } finally { accountCharsLock.unlock(); } } public void clearAccountCharacterView(Integer accountId) { accountCharsLock.lock(); try { SortedMap accChars = accountChars.remove(accountId); if (accChars != null) { accChars.clear(); } } finally { accountCharsLock.unlock(); } } public void loadAccountStorage(Integer accountId) { if (getAccountStorage(accountId) == null) { registerAccountStorage(accountId); } } private void registerAccountStorage(Integer accountId) { MapleStorage storage = MapleStorage.loadOrCreateFromDB(accountId, this.id); accountCharsLock.lock(); try { accountStorages.put(accountId, storage); } finally { accountCharsLock.unlock(); } } public void unregisterAccountStorage(Integer accountId) { accountCharsLock.lock(); try { accountStorages.remove(accountId); } finally { accountCharsLock.unlock(); } } public MapleStorage getAccountStorage(Integer accountId) { return accountStorages.get(accountId); } private static List>> getSortedAccountCharacterView(Map> map) { List>> list = new ArrayList<>(map.size()); for(Entry> e : map.entrySet()) { list.add(e); } Collections.sort(list, (o1, o2) -> o1.getKey() - o2.getKey()); return list; } public List loadAndGetAllCharactersView() { Server.getInstance().loadAllAccountsCharactersView(); return getAllCharactersView(); } public List getAllCharactersView() { // sorting by accountid, charid List chrList = new LinkedList<>(); Map> accChars; accountCharsLock.lock(); try { accChars = new HashMap<>(accountChars); } finally { accountCharsLock.unlock(); } for (Entry> e : getSortedAccountCharacterView(accChars)) { for (MapleCharacter chr : e.getValue().values()) { chrList.add(chr); } } return chrList; } public List getAccountCharactersView(Integer accountId) { List chrList; accountCharsLock.lock(); try { SortedMap accChars = accountChars.get(accountId); if(accChars != null) { chrList = new LinkedList<>(accChars.values()); } else { accountChars.put(accountId, new TreeMap()); chrList = null; } } finally { accountCharsLock.unlock(); } return chrList; } public PlayerStorage getPlayerStorage() { return players; } public MapleMatchCheckerCoordinator getMatchCheckerCoordinator() { return matchChecker; } public MaplePartySearchCoordinator getPartySearchCoordinator() { return partySearch; } public void addPlayer(MapleCharacter chr) { players.addPlayer(chr); } public void removePlayer(MapleCharacter chr) { Channel cserv = chr.getClient().getChannelServer(); if(cserv != null) { if(!cserv.removePlayer(chr)) { // oy the player is not where they should be, find this mf for(Channel ch : getChannels()) { if(ch.removePlayer(chr)) { break; } } } } players.removePlayer(chr.getId()); } public int getId() { return id; } public void addFamily(int id, MapleFamily f) { synchronized (families) { if (!families.containsKey(id)) { families.put(id, f); } } } public void removeFamily(int id) { synchronized (families) { families.remove(id); } } public MapleFamily getFamily(int id) { synchronized (families) { if (families.containsKey(id)) { return families.get(id); } return null; } } public Collection getFamilies() { synchronized(families) { return Collections.unmodifiableCollection((Collection) families.values()); } } public MapleGuild getGuild(MapleGuildCharacter mgc) { if(mgc == null) return null; int gid = mgc.getGuildId(); MapleGuild g = Server.getInstance().getGuild(gid, mgc.getWorld(), mgc.getCharacter()); if (gsStore.get(gid) == null) { gsStore.put(gid, new MapleGuildSummary(g)); } return g; } public boolean isWorldCapacityFull() { return getWorldCapacityStatus() == 2; } public int getWorldCapacityStatus() { int worldCap = getChannelsSize() * YamlConfig.config.server.CHANNEL_LOAD; int num = players.getSize(); int status; if (num >= worldCap) { status = 2; } else if (num >= worldCap * .8) { // More than 80 percent o___o status = 1; } else { status = 0; } return status; } public MapleGuildSummary getGuildSummary(int gid, int wid) { if (gsStore.containsKey(gid)) { return gsStore.get(gid); } else { MapleGuild g = Server.getInstance().getGuild(gid, wid, null); if (g != null) { gsStore.put(gid, new MapleGuildSummary(g)); } return gsStore.get(gid); } } public void updateGuildSummary(int gid, MapleGuildSummary mgs) { gsStore.put(gid, mgs); } public void reloadGuildSummary() { MapleGuild g; Server server = Server.getInstance(); for (int i : gsStore.keySet()) { g = server.getGuild(i, getId(), null); if (g != null) { gsStore.put(i, new MapleGuildSummary(g)); } else { gsStore.remove(i); } } } public void setGuildAndRank(List cids, int guildid, int rank, int exception) { for (int cid : cids) { if (cid != exception) { setGuildAndRank(cid, guildid, rank); } } } public void setOfflineGuildStatus(int guildid, int guildrank, int cid) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("UPDATE characters SET guildid = ?, guildrank = ? WHERE id = ?")) { ps.setInt(1, guildid); ps.setInt(2, guildrank); ps.setInt(3, cid); ps.executeUpdate(); } catch (SQLException se) { se.printStackTrace(); } } public void setGuildAndRank(int cid, int guildid, int rank) { MapleCharacter mc = getPlayerStorage().getCharacterById(cid); if (mc == null) { return; } boolean bDifferentGuild; if (guildid == -1 && rank == -1) { bDifferentGuild = true; } else { bDifferentGuild = guildid != mc.getGuildId(); mc.getMGC().setGuildId(guildid); mc.getMGC().setGuildRank(rank); if(bDifferentGuild) mc.getMGC().setAllianceRank(5); mc.saveGuildStatus(); } if (bDifferentGuild) { if (mc.isLoggedinWorld()) { MapleGuild guild = Server.getInstance().getGuild(guildid); if (guild != null) { mc.getMap().broadcastMessage(mc, MaplePacketCreator.guildNameChanged(cid, guild.getName())); mc.getMap().broadcastMessage(mc, MaplePacketCreator.guildMarkChanged(cid, guild)); } else { mc.getMap().broadcastMessage(mc, MaplePacketCreator.guildNameChanged(cid, "")); } } } } public void changeEmblem(int gid, List affectedPlayers, MapleGuildSummary mgs) { updateGuildSummary(gid, mgs); sendPacket(affectedPlayers, MaplePacketCreator.guildEmblemChange(gid, mgs.getLogoBG(), mgs.getLogoBGColor(), mgs.getLogo(), mgs.getLogoColor()), -1); setGuildAndRank(affectedPlayers, -1, -1, -1); //respawn player } public void sendPacket(List targetIds, final byte[] packet, int exception) { MapleCharacter chr; for (int i : targetIds) { if (i == exception) { continue; } chr = getPlayerStorage().getCharacterById(i); if (chr != null) { chr.getClient().announce(packet); } } } public boolean isGuildQueued(int guildId) { return queuedGuilds.contains(guildId); } public void putGuildQueued(int guildId) { queuedGuilds.add(guildId); } public void removeGuildQueued(int guildId) { queuedGuilds.remove(guildId); } public boolean isMarriageQueued(int marriageId) { return queuedMarriages.containsKey(marriageId); } public Pair getMarriageQueuedLocation(int marriageId) { Pair, Pair> qm = queuedMarriages.get(marriageId); return (qm != null) ? qm.getLeft() : null; } public Pair getMarriageQueuedCouple(int marriageId) { Pair, Pair> qm = queuedMarriages.get(marriageId); return (qm != null) ? qm.getRight() : null; } public void putMarriageQueued(int marriageId, boolean cathedral, boolean premium, int groomId, int brideId) { queuedMarriages.put(marriageId, new Pair<>(new Pair<>(cathedral, premium), new Pair<>(groomId, brideId))); marriageGuests.put(marriageId, new HashSet()); } public Pair> removeMarriageQueued(int marriageId) { Boolean type = queuedMarriages.remove(marriageId).getLeft().getRight(); Set guests = marriageGuests.remove(marriageId); return new Pair<>(type, guests); } public boolean addMarriageGuest(int marriageId, int playerId) { Set guests = marriageGuests.get(marriageId); if(guests != null) { if(guests.contains(playerId)) return false; guests.add(playerId); return true; } return false; } public Pair getWeddingCoupleForGuest(int guestId, Boolean cathedral) { for(Channel ch : getChannels()) { Pair p = ch.getWeddingCoupleForGuest(guestId, cathedral); if(p != null) { return p; } } List possibleWeddings = new LinkedList<>(); for(Entry> mg : new HashSet<>(marriageGuests.entrySet())) { if(mg.getValue().contains(guestId)) { Pair loc = getMarriageQueuedLocation(mg.getKey()); if(loc != null && cathedral.equals(loc.getLeft())) { possibleWeddings.add(mg.getKey()); } } } int pwSize = possibleWeddings.size(); if(pwSize == 0) { return null; } else if(pwSize > 1) { int selectedPw = -1; int selectedPos = Integer.MAX_VALUE; for(Integer pw : possibleWeddings) { for(Channel ch : getChannels()) { int pos = ch.getWeddingReservationStatus(pw, cathedral); if(pos != -1) { if(pos < selectedPos) { selectedPos = pos; selectedPw = pw; break; } } } } if(selectedPw == -1) return null; possibleWeddings.clear(); possibleWeddings.add(selectedPw); } return getMarriageQueuedCouple(possibleWeddings.get(0)); } public void debugMarriageStatus() { System.out.println("Queued marriages: " + queuedMarriages); System.out.println("Guest list: " + marriageGuests); } private void registerCharacterParty(Integer chrid, Integer partyid) { partyLock.lock(); try { partyChars.put(chrid, partyid); } finally { partyLock.unlock(); } } private void unregisterCharacterPartyInternal(Integer chrid) { partyChars.remove(chrid); } private void unregisterCharacterParty(Integer chrid) { partyLock.lock(); try { unregisterCharacterPartyInternal(chrid); } finally { partyLock.unlock(); } } public Integer getCharacterPartyid(Integer chrid) { partyLock.lock(); try { return partyChars.get(chrid); } finally { partyLock.unlock(); } } public MapleParty createParty(MaplePartyCharacter chrfor) { int partyid = runningPartyId.getAndIncrement(); MapleParty party = new MapleParty(partyid, chrfor); partyLock.lock(); try { parties.put(party.getId(), party); registerCharacterParty(chrfor.getId(), partyid); } finally { partyLock.unlock(); } party.addMember(chrfor); return party; } public MapleParty getParty(int partyid) { partyLock.lock(); try { return parties.get(partyid); } finally { partyLock.unlock(); } } private MapleParty disbandParty(int partyid) { partyLock.lock(); try { return parties.remove(partyid); } finally { partyLock.unlock(); } } private void updateCharacterParty(MapleParty party, PartyOperation operation, MaplePartyCharacter target, Collection partyMembers) { switch (operation) { case JOIN: registerCharacterParty(target.getId(), party.getId()); break; case LEAVE: case EXPEL: unregisterCharacterParty(target.getId()); break; case DISBAND: partyLock.lock(); try { for (MaplePartyCharacter partychar : partyMembers) { unregisterCharacterPartyInternal(partychar.getId()); } } finally { partyLock.unlock(); } break; default: break; } } private void updateParty(MapleParty party, PartyOperation operation, MaplePartyCharacter target) { Collection partyMembers = party.getMembers(); updateCharacterParty(party, operation, target, partyMembers); for (MaplePartyCharacter partychar : partyMembers) { MapleCharacter chr = getPlayerStorage().getCharacterById(partychar.getId()); if (chr != null) { if (operation == PartyOperation.DISBAND) { chr.setParty(null); chr.setMPC(null); } else { chr.setParty(party); chr.setMPC(partychar); } chr.announce(MaplePacketCreator.updateParty(chr.getClient().getChannel(), party, operation, target)); } } switch (operation) { case LEAVE: case EXPEL: MapleCharacter chr = getPlayerStorage().getCharacterById(target.getId()); if (chr != null) { chr.announce(MaplePacketCreator.updateParty(chr.getClient().getChannel(), party, operation, target)); chr.setParty(null); chr.setMPC(null); } default: break; } } public void updateParty(int partyid, PartyOperation operation, MaplePartyCharacter target) { MapleParty party = getParty(partyid); if (party == null) { throw new IllegalArgumentException("no party with the specified partyid exists"); } switch (operation) { case JOIN: party.addMember(target); break; case EXPEL: case LEAVE: party.removeMember(target); break; case DISBAND: disbandParty(partyid); break; case SILENT_UPDATE: case LOG_ONOFF: party.updateMember(target); break; case CHANGE_LEADER: MapleCharacter mc = party.getLeader().getPlayer(); if (mc != null) { EventInstanceManager eim = mc.getEventInstance(); if(eim != null && eim.isEventLeader(mc)) { eim.changedLeader(target); } else { int oldLeaderMapid = mc.getMapId(); if (MapleMiniDungeonInfo.isDungeonMap(oldLeaderMapid)) { if (oldLeaderMapid != target.getMapId()) { MapleMiniDungeon mmd = mc.getClient().getChannelServer().getMiniDungeon(oldLeaderMapid); if(mmd != null) { mmd.close(); } } } } party.setLeader(target); } break; default: System.out.println("Unhandled updateParty operation " + operation.name()); } updateParty(party, operation, target); } public void removeMapPartyMembers(int partyid) { MapleParty party = getParty(partyid); if(party == null) return; for(MaplePartyCharacter mpc : party.getMembers()) { MapleCharacter mc = mpc.getPlayer(); if(mc != null) { MapleMap map = mc.getMap(); if(map != null) { map.removeParty(partyid); } } } } public int find(String name) { int channel = -1; MapleCharacter chr = getPlayerStorage().getCharacterByName(name); if (chr != null) { channel = chr.getClient().getChannel(); } return channel; } public int find(int id) { int channel = -1; MapleCharacter chr = getPlayerStorage().getCharacterById(id); if (chr != null) { channel = chr.getClient().getChannel(); } return channel; } public void partyChat(MapleParty party, String chattext, String namefrom) { for (MaplePartyCharacter partychar : party.getMembers()) { if (!(partychar.getName().equals(namefrom))) { MapleCharacter chr = getPlayerStorage().getCharacterByName(partychar.getName()); if (chr != null) { chr.getClient().announce(MaplePacketCreator.multiChat(namefrom, chattext, 1)); } } } } public void buddyChat(int[] recipientCharacterIds, int cidFrom, String nameFrom, String chattext) { PlayerStorage playerStorage = getPlayerStorage(); for (int characterId : recipientCharacterIds) { MapleCharacter chr = playerStorage.getCharacterById(characterId); if (chr != null) { if (chr.getBuddylist().containsVisible(cidFrom)) { chr.getClient().announce(MaplePacketCreator.multiChat(nameFrom, chattext, 0)); } } } } public CharacterIdChannelPair[] multiBuddyFind(int charIdFrom, int[] characterIds) { List foundsChars = new ArrayList<>(characterIds.length); for (Channel ch : getChannels()) { for (int charid : ch.multiBuddyFind(charIdFrom, characterIds)) { foundsChars.add(new CharacterIdChannelPair(charid, ch.getId())); } } return foundsChars.toArray(new CharacterIdChannelPair[foundsChars.size()]); } public MapleMessenger getMessenger(int messengerid) { return messengers.get(messengerid); } public void leaveMessenger(int messengerid, MapleMessengerCharacter target) { MapleMessenger messenger = getMessenger(messengerid); if (messenger == null) { throw new IllegalArgumentException("No messenger with the specified messengerid exists"); } int position = messenger.getPositionByName(target.getName()); messenger.removeMember(target); removeMessengerPlayer(messenger, position); } public void messengerInvite(String sender, int messengerid, String target, int fromchannel) { if (isConnected(target)) { MapleCharacter targetChr = getPlayerStorage().getCharacterByName(target); if (targetChr != null) { MapleMessenger messenger = targetChr.getMessenger(); if (messenger == null) { MapleCharacter from = getChannel(fromchannel).getPlayerStorage().getCharacterByName(sender); if (from != null) { if (MapleInviteCoordinator.createInvite(InviteType.MESSENGER, from, messengerid, targetChr.getId())) { targetChr.getClient().announce(MaplePacketCreator.messengerInvite(sender, messengerid)); from.getClient().announce(MaplePacketCreator.messengerNote(target, 4, 1)); } else { from.announce(MaplePacketCreator.messengerChat(sender + " : " + target + " is already managing a Maple Messenger invitation")); } } } else { MapleCharacter from = getChannel(fromchannel).getPlayerStorage().getCharacterByName(sender); from.getClient().announce(MaplePacketCreator.messengerChat(sender + " : " + target + " is already using Maple Messenger")); } } } } public void addMessengerPlayer(MapleMessenger messenger, String namefrom, int fromchannel, int position) { for (MapleMessengerCharacter messengerchar : messenger.getMembers()) { MapleCharacter chr = getPlayerStorage().getCharacterByName(messengerchar.getName()); if(chr == null){ continue; } if (!messengerchar.getName().equals(namefrom)) { MapleCharacter from = getChannel(fromchannel).getPlayerStorage().getCharacterByName(namefrom); chr.getClient().announce(MaplePacketCreator.addMessengerPlayer(namefrom, from, position, (byte) (fromchannel - 1))); from.getClient().announce(MaplePacketCreator.addMessengerPlayer(chr.getName(), chr, messengerchar.getPosition(), (byte) (messengerchar.getChannel() - 1))); } else { chr.getClient().announce(MaplePacketCreator.joinMessenger(messengerchar.getPosition())); } } } public void removeMessengerPlayer(MapleMessenger messenger, int position) { for (MapleMessengerCharacter messengerchar : messenger.getMembers()) { MapleCharacter chr = getPlayerStorage().getCharacterByName(messengerchar.getName()); if (chr != null) { chr.getClient().announce(MaplePacketCreator.removeMessengerPlayer(position)); } } } public void messengerChat(MapleMessenger messenger, String chattext, String namefrom) { String from = ""; String to1 = ""; String to2 = ""; for (MapleMessengerCharacter messengerchar : messenger.getMembers()) { if (!(messengerchar.getName().equals(namefrom))) { MapleCharacter chr = getPlayerStorage().getCharacterByName(messengerchar.getName()); if (chr != null) { chr.getClient().announce(MaplePacketCreator.messengerChat(chattext)); if (to1.equals("")){ to1 = messengerchar.getName(); } else if (to2.equals("")){ to2 = messengerchar.getName(); } } } else { from = messengerchar.getName(); } } } public void declineChat(String sender, MapleCharacter player) { if (isConnected(sender)) { MapleCharacter senderChr = getPlayerStorage().getCharacterByName(sender); if (senderChr != null && senderChr.getMessenger() != null) { if (MapleInviteCoordinator.answerInvite(InviteType.MESSENGER, player.getId(), senderChr.getMessenger().getId(), false).result == InviteResult.DENIED) { senderChr.getClient().announce(MaplePacketCreator.messengerNote(player.getName(), 5, 0)); } } } } public void updateMessenger(int messengerid, String namefrom, int fromchannel) { MapleMessenger messenger = getMessenger(messengerid); int position = messenger.getPositionByName(namefrom); updateMessenger(messenger, namefrom, position, fromchannel); } public void updateMessenger(MapleMessenger messenger, String namefrom, int position, int fromchannel) { for (MapleMessengerCharacter messengerchar : messenger.getMembers()) { Channel ch = getChannel(fromchannel); if (!(messengerchar.getName().equals(namefrom))) { MapleCharacter chr = ch.getPlayerStorage().getCharacterByName(messengerchar.getName()); if (chr != null) { chr.getClient().announce(MaplePacketCreator.updateMessengerPlayer(namefrom, getChannel(fromchannel).getPlayerStorage().getCharacterByName(namefrom), position, (byte) (fromchannel - 1))); } } } } public void silentLeaveMessenger(int messengerid, MapleMessengerCharacter target) { MapleMessenger messenger = getMessenger(messengerid); if (messenger == null) { throw new IllegalArgumentException("No messenger with the specified messengerid exists"); } messenger.addMember(target, target.getPosition()); } public void joinMessenger(int messengerid, MapleMessengerCharacter target, String from, int fromchannel) { MapleMessenger messenger = getMessenger(messengerid); if (messenger == null) { throw new IllegalArgumentException("No messenger with the specified messengerid exists"); } messenger.addMember(target, target.getPosition()); addMessengerPlayer(messenger, from, fromchannel, target.getPosition()); } public void silentJoinMessenger(int messengerid, MapleMessengerCharacter target, int position) { MapleMessenger messenger = getMessenger(messengerid); if (messenger == null) { throw new IllegalArgumentException("No messenger with the specified messengerid exists"); } messenger.addMember(target, position); } public MapleMessenger createMessenger(MapleMessengerCharacter chrfor) { int messengerid = runningMessengerId.getAndIncrement(); MapleMessenger messenger = new MapleMessenger(messengerid, chrfor); messengers.put(messenger.getId(), messenger); return messenger; } public boolean isConnected(String charName) { return getPlayerStorage().getCharacterByName(charName) != null; } public void whisper(String sender, String target, int channel, String message) { if (isConnected(target)) { getPlayerStorage().getCharacterByName(target).getClient().announce(MaplePacketCreator.getWhisper(sender, channel, message)); } } public BuddyAddResult requestBuddyAdd(String addName, int channelFrom, int cidFrom, String nameFrom) { MapleCharacter addChar = getPlayerStorage().getCharacterByName(addName); if (addChar != null) { BuddyList buddylist = addChar.getBuddylist(); if (buddylist.isFull()) { return BuddyAddResult.BUDDYLIST_FULL; } if (!buddylist.contains(cidFrom)) { buddylist.addBuddyRequest(addChar.getClient(), cidFrom, nameFrom, channelFrom); } else if (buddylist.containsVisible(cidFrom)) { return BuddyAddResult.ALREADY_ON_LIST; } } return BuddyAddResult.OK; } public void buddyChanged(int cid, int cidFrom, String name, int channel, BuddyOperation operation) { MapleCharacter addChar = getPlayerStorage().getCharacterById(cid); if (addChar != null) { BuddyList buddylist = addChar.getBuddylist(); switch (operation) { case ADDED: if (buddylist.contains(cidFrom)) { buddylist.put(new BuddylistEntry(name, "Default Group", cidFrom, channel, true)); addChar.getClient().announce(MaplePacketCreator.updateBuddyChannel(cidFrom, (byte) (channel - 1))); } break; case DELETED: if (buddylist.contains(cidFrom)) { buddylist.put(new BuddylistEntry(name, "Default Group", cidFrom, (byte) -1, buddylist.get(cidFrom).isVisible())); addChar.getClient().announce(MaplePacketCreator.updateBuddyChannel(cidFrom, (byte) -1)); } break; } } } public void loggedOff(String name, int characterId, int channel, int[] buddies) { updateBuddies(characterId, channel, buddies, true); } public void loggedOn(String name, int characterId, int channel, int[] buddies) { updateBuddies(characterId, channel, buddies, false); } private void updateBuddies(int characterId, int channel, int[] buddies, boolean offline) { PlayerStorage playerStorage = getPlayerStorage(); for (int buddy : buddies) { MapleCharacter chr = playerStorage.getCharacterById(buddy); if (chr != null) { BuddylistEntry ble = chr.getBuddylist().get(characterId); if (ble != null && ble.isVisible()) { int mcChannel; if (offline) { ble.setChannel((byte) -1); mcChannel = -1; } else { ble.setChannel(channel); mcChannel = (byte) (channel - 1); } chr.getBuddylist().put(ble); chr.getClient().announce(MaplePacketCreator.updateBuddyChannel(ble.getCharacterId(), mcChannel)); } } } } private static Integer getPetKey(MapleCharacter chr, byte petSlot) { // assuming max 3 pets return (chr.getId() << 2) + petSlot; } public void addOwlItemSearch(Integer itemid) { suggestWLock.lock(); try { Integer cur = owlSearched.get(itemid); if(cur != null) { owlSearched.put(itemid, cur + 1); } else { owlSearched.put(itemid, 1); } } finally { suggestWLock.unlock(); } } public List> getOwlSearchedItems() { if(YamlConfig.config.server.USE_ENFORCE_ITEM_SUGGESTION) { return new ArrayList<>(0); } suggestRLock.lock(); try { List> searchCounts = new ArrayList<>(owlSearched.size()); for(Entry e : owlSearched.entrySet()) { searchCounts.add(new Pair<>(e.getKey(), e.getValue())); } return searchCounts; } finally { suggestRLock.unlock(); } } public void addCashItemBought(Integer snid) { suggestWLock.lock(); try { Map tabItemBought = cashItemBought.get(snid / 10000000); Integer cur = tabItemBought.get(snid); if (cur != null) { tabItemBought.put(snid, cur + 1); } else { tabItemBought.put(snid, 1); } } finally { suggestWLock.unlock(); } } private List>> getBoughtCashItems() { if (YamlConfig.config.server.USE_ENFORCE_ITEM_SUGGESTION) { List>> boughtCounts = new ArrayList<>(9); // thanks GabrielSin for pointing out an issue here for (int i = 0; i < 9; i++) { List> tabCounts = new ArrayList<>(0); boughtCounts.add(tabCounts); } return boughtCounts; } suggestRLock.lock(); try { List>> boughtCounts = new ArrayList<>(cashItemBought.size()); for (Map tab : cashItemBought) { List> tabItems = new LinkedList<>(); boughtCounts.add(tabItems); for (Entry e : tab.entrySet()) { tabItems.add(new Pair<>(e.getKey(), e.getValue())); } } return boughtCounts; } finally { suggestRLock.unlock(); } } private List getMostSellerOnTab(List> tabSellers) { List tabLeaderboards; // descending order Comparator> comparator = (p1, p2) -> p2.getRight().compareTo(p1.getRight()); PriorityQueue> queue = new PriorityQueue<>(Math.max(1, tabSellers.size()), comparator); for(Pair p : tabSellers) { queue.add(p); } tabLeaderboards = new LinkedList<>(); for(int i = 0; i < Math.min(tabSellers.size(), 5); i++) { tabLeaderboards.add(queue.remove().getLeft()); } return tabLeaderboards; } public List> getMostSellerCashItems() { List>> mostSellers = this.getBoughtCashItems(); List> cashLeaderboards = new ArrayList<>(9); List tabLeaderboards; List allLeaderboards = null; for(List> tabSellers : mostSellers) { if (tabSellers.size() < 5) { if (allLeaderboards == null) { List> allSellers = new LinkedList<>(); for (List> tabItems : mostSellers) { allSellers.addAll(tabItems); } allLeaderboards = getMostSellerOnTab(allSellers); } tabLeaderboards = new LinkedList<>(); if (allLeaderboards.size() < 5) { for(int i : GameConstants.CASH_DATA) { tabLeaderboards.add(i); } } else { tabLeaderboards.addAll(allLeaderboards); } } else { tabLeaderboards = getMostSellerOnTab(tabSellers); } cashLeaderboards.add(tabLeaderboards); } return cashLeaderboards; } public void registerPetHunger(MapleCharacter chr, byte petSlot) { if(chr.isGM() && YamlConfig.config.server.GM_PETS_NEVER_HUNGRY || YamlConfig.config.server.PETS_NEVER_HUNGRY) { return; } Integer key = getPetKey(chr, petSlot); activePetsLock.lock(); try { int initProc; if(Server.getInstance().getCurrentTime() - petUpdate > 55000) initProc = YamlConfig.config.server.PET_EXHAUST_COUNT - 2; else initProc = YamlConfig.config.server.PET_EXHAUST_COUNT - 1; activePets.put(key, initProc); } finally { activePetsLock.unlock(); } } public void unregisterPetHunger(MapleCharacter chr, byte petSlot) { Integer key = getPetKey(chr, petSlot); activePetsLock.lock(); try { activePets.remove(key); } finally { activePetsLock.unlock(); } } public void runPetSchedule() { Map deployedPets; activePetsLock.lock(); try { petUpdate = Server.getInstance().getCurrentTime(); deployedPets = new HashMap<>(activePets); // exception here found thanks to MedicOP } finally { activePetsLock.unlock(); } for(Map.Entry dp: deployedPets.entrySet()) { MapleCharacter chr = this.getPlayerStorage().getCharacterById(dp.getKey() / 4); if(chr == null || !chr.isLoggedinWorld()) continue; Integer dpVal = dp.getValue() + 1; if(dpVal == YamlConfig.config.server.PET_EXHAUST_COUNT) { chr.runFullnessSchedule(dp.getKey() % 4); dpVal = 0; } activePetsLock.lock(); try { activePets.put(dp.getKey(), dpVal); } finally { activePetsLock.unlock(); } } } public void registerMountHunger(MapleCharacter chr) { if(chr.isGM() && YamlConfig.config.server.GM_PETS_NEVER_HUNGRY || YamlConfig.config.server.PETS_NEVER_HUNGRY) { return; } Integer key = chr.getId(); activeMountsLock.lock(); try { int initProc; if(Server.getInstance().getCurrentTime() - mountUpdate > 45000) initProc = YamlConfig.config.server.MOUNT_EXHAUST_COUNT - 2; else initProc = YamlConfig.config.server.MOUNT_EXHAUST_COUNT - 1; activeMounts.put(key, initProc); } finally { activeMountsLock.unlock(); } } public void unregisterMountHunger(MapleCharacter chr) { Integer key = chr.getId(); activeMountsLock.lock(); try { activeMounts.remove(key); } finally { activeMountsLock.unlock(); } } public void runMountSchedule() { Map deployedMounts; activeMountsLock.lock(); try { mountUpdate = Server.getInstance().getCurrentTime(); deployedMounts = new HashMap<>(activeMounts); } finally { activeMountsLock.unlock(); } for(Map.Entry dp: deployedMounts.entrySet()) { MapleCharacter chr = this.getPlayerStorage().getCharacterById(dp.getKey()); if(chr == null || !chr.isLoggedinWorld()) continue; int dpVal = dp.getValue() + 1; if(dpVal == YamlConfig.config.server.MOUNT_EXHAUST_COUNT) { if (!chr.runTirednessSchedule()) { continue; } dpVal = 0; } activeMountsLock.lock(); try { activeMounts.put(dp.getKey(), dpVal); } finally { activeMountsLock.unlock(); } } } public void registerPlayerShop(MaplePlayerShop ps) { activePlayerShopsLock.lock(); try { activePlayerShops.put(ps.getOwner().getId(), ps); } finally { activePlayerShopsLock.unlock(); } } public void unregisterPlayerShop(MaplePlayerShop ps) { activePlayerShopsLock.lock(); try { activePlayerShops.remove(ps.getOwner().getId()); } finally { activePlayerShopsLock.unlock(); } } public List getActivePlayerShops() { List psList = new ArrayList<>(); activePlayerShopsLock.lock(); try { for(MaplePlayerShop mps : activePlayerShops.values()) { psList.add(mps); } return psList; } finally { activePlayerShopsLock.unlock(); } } public MaplePlayerShop getPlayerShop(int ownerid) { activePlayerShopsLock.lock(); try { return activePlayerShops.get(ownerid); } finally { activePlayerShopsLock.unlock(); } } public void registerHiredMerchant(MapleHiredMerchant hm) { activeMerchantsLock.lock(); try { int initProc; if(Server.getInstance().getCurrentTime() - merchantUpdate > 5 * 60 * 1000) initProc = 1; else initProc = 0; activeMerchants.put(hm.getOwnerId(), new Pair<>(hm, initProc)); } finally { activeMerchantsLock.unlock(); } } public void unregisterHiredMerchant(MapleHiredMerchant hm) { activeMerchantsLock.lock(); try { activeMerchants.remove(hm.getOwnerId()); } finally { activeMerchantsLock.unlock(); } } public void runHiredMerchantSchedule() { Map> deployedMerchants; activeMerchantsLock.lock(); try { merchantUpdate = Server.getInstance().getCurrentTime(); deployedMerchants = new LinkedHashMap<>(activeMerchants); for(Map.Entry> dm: deployedMerchants.entrySet()) { int timeOn = dm.getValue().getRight(); MapleHiredMerchant hm = dm.getValue().getLeft(); if(timeOn <= 144) { // 1440 minutes == 24hrs activeMerchants.put(hm.getOwnerId(), new Pair<>(dm.getValue().getLeft(), timeOn + 1)); } else { hm.forceClose(); this.getChannel(hm.getChannel()).removeHiredMerchant(hm.getOwnerId()); activeMerchants.remove(dm.getKey()); } } } finally { activeMerchantsLock.unlock(); } } public List getActiveMerchants() { List hmList = new ArrayList<>(); activeMerchantsLock.lock(); try { for(Pair hmp : activeMerchants.values()) { MapleHiredMerchant hm = hmp.getLeft(); if(hm.isOpen()) { hmList.add(hm); } } return hmList; } finally { activeMerchantsLock.unlock(); } } public MapleHiredMerchant getHiredMerchant(int ownerid) { activeMerchantsLock.lock(); try { if(activeMerchants.containsKey(ownerid)) { return activeMerchants.get(ownerid).getLeft(); } return null; } finally { activeMerchantsLock.unlock(); } } public void registerTimedMapObject(Runnable r, long duration) { timedMapObjectLock.lock(); try { long expirationTime = Server.getInstance().getCurrentTime() + duration; registeredTimedMapObjects.put(r, expirationTime); } finally { timedMapObjectLock.unlock(); } } public void runTimedMapObjectSchedule() { List toRemove = new LinkedList<>(); timedMapObjectLock.lock(); try { long timeNow = Server.getInstance().getCurrentTime(); for(Entry rtmo : registeredTimedMapObjects.entrySet()) { if(rtmo.getValue() <= timeNow) { toRemove.add(rtmo.getKey()); } } for(Runnable r : toRemove) { registeredTimedMapObjects.remove(r); } } finally { timedMapObjectLock.unlock(); } for(Runnable r : toRemove) { r.run(); } } public void resetDisabledServerMessages() { srvMessagesLock.lock(); try { disabledServerMessages.clear(); } finally { srvMessagesLock.unlock(); } } public boolean registerDisabledServerMessage(int chrid) { srvMessagesLock.lock(); try { boolean alreadyDisabled = disabledServerMessages.containsKey(chrid); disabledServerMessages.put(chrid, 0); return alreadyDisabled; } finally { srvMessagesLock.unlock(); } } public boolean unregisterDisabledServerMessage(int chrid) { srvMessagesLock.lock(); try { return disabledServerMessages.remove(chrid) != null; } finally { srvMessagesLock.unlock(); } } public void runDisabledServerMessagesSchedule() { List toRemove = new LinkedList<>(); srvMessagesLock.lock(); try { for(Entry dsm : disabledServerMessages.entrySet()) { int b = dsm.getValue(); if(b >= 4) { // ~35sec duration, 10sec update toRemove.add(dsm.getKey()); } else { disabledServerMessages.put(dsm.getKey(), ++b); } } for(Integer chrid : toRemove) { disabledServerMessages.remove(chrid); } } finally { srvMessagesLock.unlock(); } if(!toRemove.isEmpty()) { for(Integer chrid : toRemove) { MapleCharacter chr = players.getCharacterById(chrid); if(chr != null && chr.isLoggedinWorld()) { chr.announce(MaplePacketCreator.serverMessage(chr.getClient().getChannelServer().getServerMessage())); } } } } public void setPlayerNpcMapStep(int mapid, int step) { setPlayerNpcMapData(mapid, step, -1, false); } public void setPlayerNpcMapPodiumData(int mapid, int podium) { setPlayerNpcMapData(mapid, -1, podium, false); } public void setPlayerNpcMapData(int mapid, int step, int podium) { setPlayerNpcMapData(mapid, step, podium, true); } private static void executePlayerNpcMapDataUpdate(Connection con, boolean isPodium, Map pnpcData, int value, int worldid, int mapid) throws SQLException { final String query; if (pnpcData.containsKey(mapid)) { query = "UPDATE playernpcs_field SET " + (isPodium ? "podium" : "step") + " = ? WHERE world = ? AND map = ?"; } else { query = "INSERT INTO playernpcs_field (" + (isPodium ? "podium" : "step") + ", world, map) VALUES (?, ?, ?)"; } try (PreparedStatement ps = con.prepareStatement(query)) { ps.setInt(1, value); ps.setInt(2, worldid); ps.setInt(3, mapid); ps.executeUpdate(); } } private void setPlayerNpcMapData(int mapid, int step, int podium, boolean silent) { if (!silent) { try (Connection con = DatabaseConnection.getConnection()) { if (step != -1) { executePlayerNpcMapDataUpdate(con, false, pnpcStep, step, id, mapid); } if (podium != -1) { executePlayerNpcMapDataUpdate(con, true, pnpcPodium, podium, id, mapid); } } catch (SQLException e) { e.printStackTrace(); } } if (step != -1) { pnpcStep.put(mapid, (byte) step); } if (podium != -1) { pnpcPodium.put(mapid, (short) podium); } } public int getPlayerNpcMapStep(int mapid) { try { return pnpcStep.get(mapid); } catch (NullPointerException npe) { return 0; } } public int getPlayerNpcMapPodiumData(int mapid) { try { return pnpcPodium.get(mapid); } catch (NullPointerException npe) { return 1; } } public void resetPlayerNpcMapData() { pnpcStep.clear(); pnpcPodium.clear(); } public void setServerMessage(String msg) { for (Channel ch : getChannels()) { ch.setServerMessage(msg); } } public void broadcastPacket(final byte[] data) { for (MapleCharacter chr : players.getAllCharacters()) { chr.announce(data); } } public List> getAvailableItemBundles(int itemid) { List> hmsAvailable = new ArrayList<>(); for (MapleHiredMerchant hm : getActiveMerchants()) { List itemBundles = hm.sendAvailableBundles(itemid); for(MaplePlayerShopItem mpsi : itemBundles) { hmsAvailable.add(new Pair<>(mpsi, (AbstractMapleMapObject) hm)); } } for (MaplePlayerShop ps : getActivePlayerShops()) { List itemBundles = ps.sendAvailableBundles(itemid); for(MaplePlayerShopItem mpsi : itemBundles) { hmsAvailable.add(new Pair<>(mpsi, (AbstractMapleMapObject) ps)); } } Collections.sort(hmsAvailable, (p1, p2) -> p1.getLeft().getPrice() - p2.getLeft().getPrice()); hmsAvailable.subList(0, Math.min(hmsAvailable.size(), 200)); //truncates the list to have up to 200 elements return hmsAvailable; } private void pushRelationshipCouple(Pair> couple) { int mid = couple.getLeft(), hid = couple.getRight().getLeft(), wid = couple.getRight().getRight(); relationshipCouples.put(mid, couple.getRight()); relationships.put(hid, mid); relationships.put(wid, mid); } public Pair getRelationshipCouple(int relationshipId) { Pair rc = relationshipCouples.get(relationshipId); if(rc == null) { Pair> couple = getRelationshipCoupleFromDb(relationshipId, true); if(couple == null) return null; pushRelationshipCouple(couple); rc = couple.getRight(); } return rc; } public int getRelationshipId(int playerId) { Integer ret = relationships.get(playerId); if(ret == null) { Pair> couple = getRelationshipCoupleFromDb(playerId, false); if(couple == null) return -1; pushRelationshipCouple(couple); ret = couple.getLeft(); } return ret; } private static Pair> getRelationshipCoupleFromDb(int id, boolean usingMarriageId) { try (Connection con = DatabaseConnection.getConnection()) { Integer mid = null, hid = null, wid = null; PreparedStatement ps; if (usingMarriageId) { ps = con.prepareStatement("SELECT * FROM marriages WHERE marriageid = ?"); ps.setInt(1, id); } else { ps = con.prepareStatement("SELECT * FROM marriages WHERE husbandid = ? OR wifeid = ?"); ps.setInt(1, id); ps.setInt(2, id); } try (ResultSet rs = ps.executeQuery()) { if (rs.next()) { mid = rs.getInt("marriageid"); hid = rs.getInt("husbandid"); wid = rs.getInt("wifeid"); } } ps.close(); return (mid == null) ? null : new Pair<>(mid, new Pair<>(hid, wid)); } catch (SQLException se) { se.printStackTrace(); return null; } } public int createRelationship(int groomId, int brideId) { int ret = addRelationshipToDb(groomId, brideId); pushRelationshipCouple(new Pair<>(ret, new Pair<>(groomId, brideId))); return ret; } private static int addRelationshipToDb(int groomId, int brideId) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("INSERT INTO marriages (husbandid, wifeid) VALUES (?, ?)", Statement.RETURN_GENERATED_KEYS)) { ps.setInt(1, groomId); ps.setInt(2, brideId); ps.executeUpdate(); try (ResultSet rs = ps.getGeneratedKeys()) { rs.next(); int ret = rs.getInt(1); return ret; } } catch (SQLException se) { se.printStackTrace(); return -1; } } public void deleteRelationship(int playerId, int partnerId) { int relationshipId = relationships.get(playerId); deleteRelationshipFromDb(relationshipId); relationshipCouples.remove(relationshipId); relationships.remove(playerId); relationships.remove(partnerId); } private static void deleteRelationshipFromDb(int playerId) { try (Connection con = DatabaseConnection.getConnection(); PreparedStatement ps = con.prepareStatement("DELETE FROM marriages WHERE marriageid = ?")) { ps.setInt(1, playerId); ps.executeUpdate(); } catch (SQLException se) { se.printStackTrace(); } } public void dropMessage(int type, String message) { for (MapleCharacter player : getPlayerStorage().getAllCharacters()) { player.dropMessage(type, message); } } public boolean registerFisherPlayer(MapleCharacter chr, int baitLevel) { synchronized (fishingAttempters) { if (fishingAttempters.containsKey(chr)) { return false; } fishingAttempters.put(chr, baitLevel); return true; } } public int unregisterFisherPlayer(MapleCharacter chr) { Integer baitLevel = fishingAttempters.remove(chr); if (baitLevel != null) { return baitLevel; } else { return 0; } } public void runCheckFishingSchedule() { double[] fishingLikelihoods = Fishing.fetchFishingLikelihood(); double yearLikelihood = fishingLikelihoods[0], timeLikelihood = fishingLikelihoods[1]; if (!fishingAttempters.isEmpty()) { List fishingAttemptersList; synchronized (fishingAttempters) { fishingAttemptersList = new ArrayList<>(fishingAttempters.keySet()); } for (MapleCharacter chr : fishingAttemptersList) { int baitLevel = unregisterFisherPlayer(chr); Fishing.doFishing(chr, baitLevel, yearLikelihood, timeLikelihood); } } } public void runPartySearchUpdateSchedule() { partySearch.updatePartySearchStorage(); partySearch.runPartySearch(); } public BaseService getServiceAccess(WorldServices sv) { return services.getAccess(sv).getService(); } private void closeWorldServices() { services.shutdown(); } private void clearWorldData() { List pList; partyLock.lock(); try { pList = new ArrayList<>(parties.values()); } finally { partyLock.unlock(); } for(MapleParty p : pList) { p.disposeLocks(); } closeWorldServices(); disposeLocks(); } private void disposeLocks() { LockCollector.getInstance().registerDisposeAction(() -> emptyLocks()); } private void emptyLocks() { accountCharsLock = accountCharsLock.dispose(); partyLock = partyLock.dispose(); srvMessagesLock = srvMessagesLock.dispose(); activePetsLock = activePetsLock.dispose(); activeMountsLock = activeMountsLock.dispose(); activePlayerShopsLock = activePlayerShopsLock.dispose(); activeMerchantsLock = activeMerchantsLock.dispose(); timedMapObjectLock = timedMapObjectLock.dispose(); } public final void shutdown() { for (Channel ch : getChannels()) { ch.shutdown(); } if(petsSchedule != null) { petsSchedule.cancel(false); petsSchedule = null; } if(srvMessagesSchedule != null) { srvMessagesSchedule.cancel(false); srvMessagesSchedule = null; } if(mountsSchedule != null) { mountsSchedule.cancel(false); mountsSchedule = null; } if(merchantSchedule != null) { merchantSchedule.cancel(false); merchantSchedule = null; } if(timedMapObjectsSchedule != null) { timedMapObjectsSchedule.cancel(false); timedMapObjectsSchedule = null; } if(charactersSchedule != null) { charactersSchedule.cancel(false); charactersSchedule = null; } if(marriagesSchedule != null) { marriagesSchedule.cancel(false); marriagesSchedule = null; } if(mapOwnershipSchedule != null) { mapOwnershipSchedule.cancel(false); mapOwnershipSchedule = null; } if(fishingSchedule != null) { fishingSchedule.cancel(false); fishingSchedule = null; } if(partySearchSchedule != null) { partySearchSchedule.cancel(false); partySearchSchedule = null; } if(timeoutSchedule != null) { timeoutSchedule.cancel(false); timeoutSchedule = null; } players.disconnectAll(); players = null; clearWorldData(); System.out.println("Finished shutting down world " + id + "\r\n"); } }