/* * Copyright (C) 2004-2015 L2J Server * * This file is part of L2J Server. * * L2J Server is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * L2J Server 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 * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.l2jserver.gameserver.instancemanager; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.Statement; import java.util.Arrays; import java.util.Collections; import java.util.Comparator; import java.util.List; import java.util.Map; import java.util.OptionalInt; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; import java.util.logging.Level; import java.util.logging.Logger; import java.util.stream.Collectors; import com.l2jserver.L2DatabaseFactory; import com.l2jserver.gameserver.ThreadPoolManager; import com.l2jserver.gameserver.model.clan.entry.PledgeApplicantInfo; import com.l2jserver.gameserver.model.clan.entry.PledgeRecruitInfo; import com.l2jserver.gameserver.model.clan.entry.PledgeWaitingInfo; import com.l2jserver.gameserver.util.Util; /** * @author Sdw */ public class ClanEntryManager { protected static final Logger _log = Logger.getLogger(ClanEntryManager.class.getName()); private static final Map _waitingList = new ConcurrentHashMap<>(); private static final Map _clanList = new ConcurrentHashMap<>(); private static final Map> _applicantList = new ConcurrentHashMap<>(); private static final Map> _clanLocked = new ConcurrentHashMap<>(); private static final Map> _playerLocked = new ConcurrentHashMap<>(); private static final String INSERT_APPLICANT = "INSERT INTO pledge_applicant VALUES (?, ?, ?, ?)"; private static final String DELETE_APPLICANT = "DELETE FROM pledge_applicant WHERE charId = ? AND clanId = ?"; private static final String INSERT_WAITING_LIST = "INSERT INTO pledge_waiting_list VALUES (?, ?)"; private static final String DELETE_WAITING_LIST = "DELETE FROM pledge_waiting_list WHERE char_id = ?"; private static final String INSERT_CLAN_RECRUIT = "INSERT INTO pledge_recruit VALUES (?, ?, ?, ?)"; private static final String UPDATE_CLAN_RECRUIT = "UPDATE pledge_recruit SET karma = ?, information = ?, detailed_information = ? WHERE clan_id = ?"; private static final String DELETE_CLAN_RECRUIT = "DELETE FROM pledge_recruit WHERE clan_id = ?"; //@formatter:off private static final List> PLAYER_COMPARATOR = Arrays.asList( null, Comparator.comparing(PledgeWaitingInfo::getPlayerName), Comparator.comparingInt(PledgeWaitingInfo::getKarma), Comparator.comparingInt(PledgeWaitingInfo::getPlayerLvl), Comparator.comparingInt(PledgeWaitingInfo::getPlayerClassId)); //@formatter:on //@formatter:off private static final List> CLAN_COMPARATOR = Arrays.asList( null, Comparator.comparing(PledgeRecruitInfo::getClanName), Comparator.comparing(PledgeRecruitInfo::getClanLeaderName), Comparator.comparingInt(PledgeRecruitInfo::getClanLevel), Comparator.comparingInt(PledgeRecruitInfo::getKarma)); //@formatter:on private static final long LOCK_TIME = TimeUnit.MINUTES.toMillis(5); protected ClanEntryManager() { load(); } private final void load() { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); Statement s = con.createStatement(); ResultSet rs = s.executeQuery("SELECT * FROM pledge_recruit")) { while (rs.next()) { _clanList.put(rs.getInt("clan_id"), new PledgeRecruitInfo(rs.getInt("clan_id"), rs.getInt("karma"), rs.getString("information"), rs.getString("detailed_information"))); } _log.info(getClass().getSimpleName() + ": Loaded: " + _clanList.size() + " clan entry"); } catch (Exception e) { _log.warning(getClass().getSimpleName() + ": Exception: ClanEntryManager.load(): " + e.getMessage()); } try (Connection con = L2DatabaseFactory.getInstance().getConnection(); Statement s = con.createStatement(); ResultSet rs = s.executeQuery("SELECT a.char_id, a.karma, b.base_class, b.level, b.char_name FROM pledge_waiting_list as a LEFT JOIN characters as b ON a.char_id = b.charId")) { while (rs.next()) { _waitingList.put(rs.getInt("char_id"), new PledgeWaitingInfo(rs.getInt("char_id"), rs.getInt("level"), rs.getInt("karma"), rs.getInt("base_class"), rs.getString("char_name"))); } _log.info(getClass().getSimpleName() + ": Loaded: " + _waitingList.size() + " player in waiting list"); } catch (Exception e) { _log.warning(getClass().getSimpleName() + ": Exception: ClanEntryManager.load(): " + e.getMessage()); } try (Connection con = L2DatabaseFactory.getInstance().getConnection(); Statement s = con.createStatement(); ResultSet rs = s.executeQuery("SELECT a.charId, a.clanId, a.karma, a.message, b.base_class, b.level, b.char_name FROM pledge_applicant as a LEFT JOIN characters as b ON a.charId = b.charId")) { while (rs.next()) { _applicantList.computeIfAbsent(rs.getInt("clanId"), k -> new ConcurrentHashMap<>()).put(rs.getInt("charId"), new PledgeApplicantInfo(rs.getInt("charId"), rs.getString("char_name"), rs.getInt("level"), rs.getInt("karma"), rs.getInt("clanId"), rs.getString("message"))); } _log.info(getClass().getSimpleName() + ": Loaded: " + _applicantList.size() + " player application"); } catch (Exception e) { _log.warning(getClass().getSimpleName() + ": Exception: ClanEntryManager.load(): " + e.getMessage()); } } public Map getWaitingList() { return _waitingList; } public Map getClanList() { return _clanList; } public Map> getApplicantList() { return _applicantList; } public Map getApplicantListForClan(int clanId) { return _applicantList.getOrDefault(clanId, Collections.emptyMap()); } public PledgeApplicantInfo getPlayerApplication(int clanId, int playerId) { return _applicantList.getOrDefault(clanId, Collections.emptyMap()).get(playerId); } public boolean removePlayerApplication(int clanId, int playerId) { final Map clanApplicantList = _applicantList.get(clanId); try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(DELETE_APPLICANT)) { statement.setInt(1, playerId); statement.setInt(2, clanId); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } return (clanApplicantList != null) && (clanApplicantList.remove(playerId) != null); } public boolean addPlayerApplicationToClan(int clanId, PledgeApplicantInfo info) { if (!_playerLocked.containsKey(info.getPlayerId())) { _applicantList.computeIfAbsent(clanId, k -> new ConcurrentHashMap<>()).put(info.getPlayerId(), info); try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(INSERT_APPLICANT)) { statement.setInt(1, info.getPlayerId()); statement.setInt(2, info.getRequestClanId()); statement.setInt(3, info.getKarma()); statement.setString(4, info.getMessage()); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } return true; } return false; } public OptionalInt getClanIdForPlayerApplication(int playerId) { return _applicantList.entrySet().stream().filter(e -> e.getValue().containsKey(playerId)).mapToInt(e -> e.getKey()).findFirst(); } public boolean addToWaitingList(int playerId, PledgeWaitingInfo info) { if (!_playerLocked.containsKey(playerId)) { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(INSERT_WAITING_LIST)) { statement.setInt(1, info.getPlayerId()); statement.setInt(2, info.getKarma()); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } return _waitingList.put(playerId, info) != null; } return false; } public boolean removeFromWaitingList(int playerId) { if (_waitingList.containsKey(playerId)) { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(DELETE_WAITING_LIST)) { statement.setInt(1, playerId); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } _waitingList.remove(playerId); lockPlayer(playerId); return true; } return false; } public boolean addToClanList(int clanId, PledgeRecruitInfo info) { if (!_clanList.containsKey(clanId) && !_clanLocked.containsKey(clanId)) { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(INSERT_CLAN_RECRUIT)) { statement.setInt(1, info.getClanId()); statement.setInt(2, info.getKarma()); statement.setString(3, info.getInformation()); statement.setString(4, info.getDetailedInformation()); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } return _clanList.put(clanId, info) != null; } return false; } public boolean updateClanList(int clanId, PledgeRecruitInfo info) { if (_clanList.containsKey(clanId) && !_clanLocked.containsKey(clanId)) { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(UPDATE_CLAN_RECRUIT)) { statement.setInt(1, info.getKarma()); statement.setString(2, info.getInformation()); statement.setString(3, info.getDetailedInformation()); statement.setInt(4, info.getClanId()); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } return _clanList.replace(clanId, info) != null; } return false; } public boolean removeFromClanList(int clanId) { if (_clanList.containsKey(clanId)) { try (Connection con = L2DatabaseFactory.getInstance().getConnection(); PreparedStatement statement = con.prepareStatement(DELETE_CLAN_RECRUIT)) { statement.setInt(1, clanId); statement.executeUpdate(); } catch (Exception e) { _log.log(Level.WARNING, e.getMessage(), e); } _clanList.remove(clanId); lockClan(clanId); return true; } return false; } public List getSortedWaitingList(int levelMin, int levelMax, int role, int sortBy, boolean descending) { sortBy = Util.constrain(sortBy, 1, PLAYER_COMPARATOR.size() - 1); // TODO: Handle Role //@formatter:off return _waitingList.values().stream() .filter(p -> ((p.getPlayerLvl() >= levelMin) && (p.getPlayerLvl() <= levelMax))) .sorted(descending ? PLAYER_COMPARATOR.get(sortBy).reversed() : PLAYER_COMPARATOR.get(sortBy)) .collect(Collectors.toList()); //@formatter:on } public List queryWaitingListByName(String name) { return _waitingList.values().stream().filter(p -> p.getPlayerName().toLowerCase().contains(name)).collect(Collectors.toList()); } public List getSortedClanListByName(String query, int type) { return type == 1 ? _clanList.values().stream().filter(p -> p.getClanName().toLowerCase().contains(query)).collect(Collectors.toList()) : _clanList.values().stream().filter(p -> p.getClanLeaderName().toLowerCase().contains(query)).collect(Collectors.toList()); } public PledgeRecruitInfo getClanById(int clanId) { return _clanList.get(clanId); } public boolean isClanRegistred(int clanId) { return _clanList.get(clanId) != null; } public boolean isPlayerRegistred(int playerId) { return _waitingList.get(playerId) != null; } public List getUnSortedClanList() { return _clanList.values().stream().collect(Collectors.toList()); } public List getSortedClanList(int clanLevel, int karma, int sortBy, boolean descending) { sortBy = Util.constrain(sortBy, 1, CLAN_COMPARATOR.size() - 1); //@formatter:off return _clanList.values().stream() .filter((p -> (((clanLevel < 0) && (karma >= 0) && (karma != p.getKarma())) || ((clanLevel >= 0) && (karma < 0) && (clanLevel != p.getClanLevel())) || ((clanLevel >= 0) && (karma >= 0) && ((clanLevel != p.getClanLevel()) || (karma != p.getKarma())))))) .sorted(descending ? CLAN_COMPARATOR.get(sortBy).reversed() : CLAN_COMPARATOR.get(sortBy)) .collect(Collectors.toList()); //@formatter:on } public long getPlayerLockTime(int playerId) { return _playerLocked.get(playerId) == null ? 0 : _playerLocked.get(playerId).getDelay(TimeUnit.MINUTES); } public long getClanLockTime(int playerId) { return _clanLocked.get(playerId) == null ? 0 : _playerLocked.get(playerId).getDelay(TimeUnit.MINUTES); } private static void lockPlayer(int playerId) { _playerLocked.put(playerId, ThreadPoolManager.getInstance().scheduleGeneral(() -> { _playerLocked.remove(playerId); }, LOCK_TIME)); } private static void lockClan(int clanId) { _clanLocked.put(clanId, ThreadPoolManager.getInstance().scheduleGeneral(() -> { _clanLocked.remove(clanId); }, LOCK_TIME)); } public static ClanEntryManager getInstance() { return SingletonHolder._instance; } private static class SingletonHolder { protected static final ClanEntryManager _instance = new ClanEntryManager(); } }