GameServer.java 8.28 KB
Newer Older
Melledy's avatar
Melledy committed
1
2
package emu.grasscutter.server.game;

3
import emu.grasscutter.GameConstants;
Melledy's avatar
Melledy committed
4
import emu.grasscutter.Grasscutter;
Jaida Wu's avatar
Jaida Wu committed
5
import emu.grasscutter.command.CommandMap;
Melledy's avatar
Melledy committed
6
import emu.grasscutter.database.DatabaseHelper;
Melledy's avatar
Melledy committed
7
import emu.grasscutter.game.Account;
Akka's avatar
Akka committed
8
import emu.grasscutter.game.combine.CombineManger;
Kengxxiao's avatar
Kengxxiao committed
9
import emu.grasscutter.game.drop.DropManager;
Melledy's avatar
Melledy committed
10
import emu.grasscutter.game.dungeons.DungeonManager;
Kinesis's avatar
Kinesis committed
11
import emu.grasscutter.game.expedition.ExpeditionManager;
Melledy's avatar
Melledy committed
12
13
14
import emu.grasscutter.game.gacha.GachaManager;
import emu.grasscutter.game.managers.InventoryManager;
import emu.grasscutter.game.managers.MultiplayerManager;
Melledy's avatar
Melledy committed
15
16
import emu.grasscutter.game.managers.chat.ChatManager;
import emu.grasscutter.game.managers.chat.ChatManagerHandler;
Melledy's avatar
Melledy committed
17
import emu.grasscutter.game.player.Player;
18
import emu.grasscutter.game.quest.ServerQuestHandler;
Melledy's avatar
Melledy committed
19
import emu.grasscutter.game.shop.ShopManager;
20
import emu.grasscutter.game.tower.TowerScheduleManager;
Melledy's avatar
Melledy committed
21
import emu.grasscutter.game.world.World;
Melledy's avatar
Melledy committed
22
23
import emu.grasscutter.net.packet.PacketHandler;
import emu.grasscutter.net.proto.SocialDetailOuterClass.SocialDetail;
KingRainbow44's avatar
KingRainbow44 committed
24
import emu.grasscutter.server.event.types.ServerEvent;
KingRainbow44's avatar
KingRainbow44 committed
25
26
27
import emu.grasscutter.server.event.game.ServerTickEvent;
import emu.grasscutter.server.event.internal.ServerStartEvent;
import emu.grasscutter.server.event.internal.ServerStopEvent;
Yazawazi's avatar
Yazawazi committed
28
import emu.grasscutter.task.TaskMap;
29
30
import kcp.highway.ChannelConfig;
import kcp.highway.KcpServer;
Melledy's avatar
Melledy committed
31

Kengxxiao's avatar
Kengxxiao committed
32
33
34
35
36
import java.net.InetSocketAddress;
import java.time.OffsetDateTime;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;

37
import static emu.grasscutter.utils.Language.translate;
38
import static emu.grasscutter.Configuration.*;
39

40
public final class GameServer extends KcpServer {
Melledy's avatar
Melledy committed
41
42
	private final InetSocketAddress address;
	private final GameServerPacketHandler packetHandler;
43
44
	private final ServerQuestHandler questHandler;
	
45
	private final Map<Integer, Player> players;
46
	private final Set<World> worlds;
Melledy's avatar
Melledy committed
47
	
48
	private ChatManagerHandler chatManager;
Melledy's avatar
Melledy committed
49
50
51
52
53
	private final InventoryManager inventoryManager;
	private final GachaManager gachaManager;
	private final ShopManager shopManager;
	private final MultiplayerManager multiplayerManager;
	private final DungeonManager dungeonManager;
Kinesis's avatar
Kinesis committed
54
	private final ExpeditionManager expeditionManager;
Melledy's avatar
Melledy committed
55
	private final CommandMap commandMap;
Yazawazi's avatar
Yazawazi committed
56
	private final TaskMap taskMap;
Kengxxiao's avatar
Kengxxiao committed
57
	private final DropManager dropManager;
Akka's avatar
Akka committed
58
59

	private final CombineManger combineManger;
60
	private final TowerScheduleManager towerScheduleManager;
Akka's avatar
Akka committed
61

62
	private static InetSocketAddress getAdapterInetSocketAddress(){
63
		InetSocketAddress inetSocketAddress;
64
65
66
67
68
69
70
71
72
73
		if(GAME_INFO.bindAddress.equals("")){
			inetSocketAddress=new InetSocketAddress(GAME_INFO.bindPort);
		}else{
			inetSocketAddress=new InetSocketAddress(
					GAME_INFO.bindAddress,
					GAME_INFO.bindPort
			);
		}
		return inetSocketAddress;
	}
74
	public GameServer() {
75
		this(getAdapterInetSocketAddress());
76
	}
Melledy's avatar
Melledy committed
77
	public GameServer(InetSocketAddress address) {
78
79
80
81
82
83
84
85
86
87
		ChannelConfig channelConfig = new ChannelConfig();
		channelConfig.nodelay(true,40,2,true);
		channelConfig.setMtu(1400);
		channelConfig.setSndwnd(256);
		channelConfig.setRcvwnd(256);
		channelConfig.setTimeoutMillis(30*1000);//30s
		channelConfig.setUseConvChannel(true);
		channelConfig.setAckNoDelay(false);

		this.init(GameSessionManager.getListener(),channelConfig,address);
88

Melledy's avatar
Melledy committed
89
90
		this.address = address;
		this.packetHandler = new GameServerPacketHandler(PacketHandler.class);
91
		this.questHandler = new ServerQuestHandler();
Melledy's avatar
Melledy committed
92
		this.players = new ConcurrentHashMap<>();
93
		this.worlds = Collections.synchronizedSet(new HashSet<>());
Melledy's avatar
Melledy committed
94
95
96
97
98
99
100
		
		this.chatManager = new ChatManager(this);
		this.inventoryManager = new InventoryManager(this);
		this.gachaManager = new GachaManager(this);
		this.shopManager = new ShopManager(this);
		this.multiplayerManager = new MultiplayerManager(this);
		this.dungeonManager = new DungeonManager(this);
Melledy's avatar
Melledy committed
101
		this.commandMap = new CommandMap(true);
Yazawazi's avatar
Yazawazi committed
102
		this.taskMap = new TaskMap(true);
Kengxxiao's avatar
Kengxxiao committed
103
		this.dropManager = new DropManager(this);
Kinesis's avatar
Kinesis committed
104
		this.expeditionManager = new ExpeditionManager(this);
Akka's avatar
Akka committed
105
		this.combineManger = new CombineManger(this);
106
		this.towerScheduleManager = new TowerScheduleManager(this);
107

Melledy's avatar
Melledy committed
108
		// Hook into shutdown event.
Melledy's avatar
Melledy committed
109
110
111
112
113
114
115
		Runtime.getRuntime().addShutdownHook(new Thread(this::onServerShutdown));
	}
	
	public GameServerPacketHandler getPacketHandler() {
		return packetHandler;
	}

116
117
118
119
	public ServerQuestHandler getQuestHandler() {
		return questHandler;
	}

120
	public Map<Integer, Player> getPlayers() {
Melledy's avatar
Melledy committed
121
122
123
		return players;
	}

124
125
126
127
	public Set<World> getWorlds() {
		return worlds;
	}

128
	public ChatManagerHandler getChatManager() {
Melledy's avatar
Melledy committed
129
130
		return chatManager;
	}
131
	
132
133
134
	public void setChatManager(ChatManagerHandler chatManager) {
		this.chatManager = chatManager;
	}
Melledy's avatar
Melledy committed
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150

	public InventoryManager getInventoryManager() {
		return inventoryManager;
	}

	public GachaManager getGachaManager() {
		return gachaManager;
	}
	
	public ShopManager getShopManager() {
		return shopManager;
	}

	public MultiplayerManager getMultiplayerManager() {
		return multiplayerManager;
	}
Kengxxiao's avatar
Kengxxiao committed
151
152
153
154

	public DropManager getDropManager() {
		return dropManager;
	}
Melledy's avatar
Melledy committed
155
156
157
158
	
	public DungeonManager getDungeonManager() {
		return dungeonManager;
	}
Kinesis's avatar
Kinesis committed
159
160
161
162
163

	public ExpeditionManager getExpeditionManager() {
		return expeditionManager;
	}

Melledy's avatar
Melledy committed
164
165
166
	public CommandMap getCommandMap() {
		return this.commandMap;
	}
Yazawazi's avatar
Yazawazi committed
167

Akka's avatar
Akka committed
168
169
170
	public CombineManger getCombineManger(){
		return this.combineManger;
	}
Kinesis's avatar
Kinesis committed
171

172
173
174
175
	public TowerScheduleManager getTowerScheduleManager() {
		return towerScheduleManager;
	}

176

Yazawazi's avatar
Yazawazi committed
177
178
179
	public TaskMap getTaskMap() {
		return this.taskMap;
	}
Melledy's avatar
Melledy committed
180
	
181
	public void registerPlayer(Player player) {
182
		getPlayers().put(player.getUid(), player);
Melledy's avatar
Melledy committed
183
184
	}

185
	public Player getPlayerByUid(int id) {
186
		return this.getPlayerByUid(id, false);
Melledy's avatar
Melledy committed
187
188
	}
	
189
	public Player getPlayerByUid(int id, boolean allowOfflinePlayers) {
Melledy's avatar
Melledy committed
190
		// Console check
191
		if (id == GameConstants.SERVER_CONSOLE_UID) {
Melledy's avatar
Melledy committed
192
193
194
195
			return null;
		}
		
		// Get from online players
196
		Player player = this.getPlayers().get(id);
197
198
199
200
		
		if (!allowOfflinePlayers) {
			return player;
		}
Melledy's avatar
Melledy committed
201
202
203
		
		// Check database if character isnt here
		if (player == null) {
204
			player = DatabaseHelper.getPlayerByUid(id);
Melledy's avatar
Melledy committed
205
206
207
208
209
		}
		
		return player;
	}
	
210
211
212
213
214
	public Player getPlayerByAccountId(String accountId) {
		Optional<Player> playerOpt = getPlayers().values().stream().filter(player -> player.getAccount().getId().equals(accountId)).findFirst();
		return playerOpt.orElse(null);
	}
	
215
	public SocialDetail.Builder getSocialDetailByUid(int id) {
Melledy's avatar
Melledy committed
216
		// Get from online players
217
		Player player = this.getPlayerByUid(id, true);
Melledy's avatar
Melledy committed
218
219
220
221
222
223
224
225
	
		if (player == null) {
			return null;
		}
		
		return player.getSocialDetail();
	}
	
Melledy's avatar
Melledy committed
226
	public Account getAccountByName(String username) {
227
		Optional<Player> playerOpt = getPlayers().values().stream().filter(player -> player.getAccount().getUsername().equals(username)).findFirst();
228
		if (playerOpt.isPresent()) {
Melledy's avatar
Melledy committed
229
230
231
232
			return playerOpt.get().getAccount();
		}
		return DatabaseHelper.getAccountByName(username);
	}
233
234

	public synchronized void onTick(){
235
236
237
		Iterator<World> it = this.getWorlds().iterator();
		while (it.hasNext()) {
			World world = it.next();
238

239
240
241
			if (world.getPlayerCount() == 0) {
				it.remove();
			}
242

243
244
			world.onTick();
		}
245

Melledy's avatar
Melledy committed
246
247
248
		for (Player player : this.getPlayers().values()) {
			player.onTick();
		}
249

Melledy's avatar
Melledy committed
250
		ServerTickEvent event = new ServerTickEvent(); event.call();
Melledy's avatar
Melledy committed
251
	}
252
253
254
255
256
257
258
259
260
	
	public void registerWorld(World world) {
		this.getWorlds().add(world);
	}
	
	public void deregisterWorld(World world) {
		// TODO Auto-generated method stub
		
	}
Melledy's avatar
Melledy committed
261

262
	public void start() {
263
		// Schedule game loop.
Bwly999's avatar
Bwly999 committed
264
		Timer gameLoop = new Timer();
265
266
267
268
269
270
		gameLoop.scheduleAtFixedRate(new TimerTask() {
			@Override
			public void run() {
				try {
					onTick();
				} catch (Exception e) {
271
					Grasscutter.getLogger().error(translate("messages.game.game_update_error"), e);
272
273
				}
			}
Bwly999's avatar
Bwly999 committed
274
		}, new Date(), 1000L);
275
276
		Grasscutter.getLogger().info(translate("messages.status.free_software"));
		Grasscutter.getLogger().info(translate("messages.game.port_bind", Integer.toString(address.getPort())));
277
278
		ServerStartEvent event = new ServerStartEvent(ServerEvent.Type.GAME, OffsetDateTime.now());
		event.call();
Melledy's avatar
Melledy committed
279
	}
280

Melledy's avatar
Melledy committed
281
	public void onServerShutdown() {
KingRainbow44's avatar
KingRainbow44 committed
282
		ServerStopEvent event = new ServerStopEvent(ServerEvent.Type.GAME, OffsetDateTime.now()); event.call();
283

Melledy's avatar
Melledy committed
284
		// Kick and save all players
285
		List<Player> list = new ArrayList<>(this.getPlayers().size());
Melledy's avatar
Melledy committed
286
		list.addAll(this.getPlayers().values());
287

288
		for (Player player : list) {
Melledy's avatar
Melledy committed
289
290
291
292
			player.getSession().close();
		}
	}
}