2010-12-21 16:52:15 +00:00
package org.bukkit.craftbukkit ;
2011-01-29 21:04:02 +00:00
import org.bukkit.command.* ;
2011-01-15 21:21:05 +00:00
import org.bukkit.entity.Player ;
2011-03-26 23:20:35 +01:00
import org.bukkit.event.world.WorldLoadEvent ;
2010-12-24 17:24:21 +00:00
import java.io.File ;
2011-01-07 10:42:53 +08:00
import java.util.ArrayList ;
2011-02-09 20:25:14 +00:00
import java.util.LinkedHashMap ;
2010-12-22 15:22:23 +00:00
import java.util.List ;
2011-02-08 12:03:36 +00:00
import java.util.Map ;
2010-12-24 17:24:21 +00:00
import java.util.logging.Level ;
import java.util.logging.Logger ;
2011-02-25 16:12:38 +00:00
import jline.ConsoleReader ;
2011-02-23 02:37:56 +00:00
import net.minecraft.server.ChunkCoordinates ;
2011-02-24 12:07:33 +00:00
import net.minecraft.server.ConvertProgressUpdater ;
import net.minecraft.server.Convertable ;
2011-01-29 22:50:29 +01:00
import net.minecraft.server.EntityPlayer ;
2010-12-27 02:13:03 +00:00
import net.minecraft.server.MinecraftServer ;
2011-01-20 03:53:43 +00:00
import net.minecraft.server.PropertyManager ;
2010-12-27 02:13:03 +00:00
import net.minecraft.server.ServerConfigurationManager ;
2011-02-23 02:37:56 +00:00
import net.minecraft.server.ServerNBTManager ;
2011-02-24 12:07:33 +00:00
import net.minecraft.server.WorldLoaderServer ;
2011-02-07 01:59:06 +00:00
import net.minecraft.server.WorldManager ;
2011-02-05 18:15:04 +00:00
import net.minecraft.server.WorldServer ;
2011-03-12 18:23:57 +00:00
import net.minecraft.server.ServerCommand ;
import net.minecraft.server.ICommandListener ;
2010-12-22 15:22:23 +00:00
import org.bukkit.* ;
2011-01-03 02:25:09 +00:00
import org.bukkit.plugin.Plugin ;
2010-12-24 17:24:21 +00:00
import org.bukkit.plugin.PluginManager ;
2010-12-25 15:42:17 +00:00
import org.bukkit.plugin.SimplePluginManager ;
2010-12-24 17:24:21 +00:00
import org.bukkit.plugin.java.JavaPluginLoader ;
2011-02-02 23:53:04 +00:00
import org.bukkit.scheduler.BukkitScheduler ;
import org.bukkit.craftbukkit.scheduler.CraftScheduler ;
2010-12-21 16:52:15 +00:00
2011-01-20 03:53:43 +00:00
public final class CraftServer implements Server {
2010-12-24 17:24:21 +00:00
private final String serverName = " Craftbukkit " ;
2011-02-01 21:55:30 +00:00
private final String serverVersion ;
2011-02-23 11:32:19 +00:00
private final String protocolVersion = " 1.3 " ;
2010-12-25 15:42:17 +00:00
private final PluginManager pluginManager = new SimplePluginManager ( this ) ;
2011-02-02 23:53:04 +00:00
private final BukkitScheduler scheduler = new CraftScheduler ( this ) ;
2011-01-20 03:53:43 +00:00
private final CommandMap commandMap = new SimpleCommandMap ( this ) ;
2010-12-22 15:22:23 +00:00
protected final MinecraftServer console ;
2010-12-26 02:20:29 +00:00
protected final ServerConfigurationManager server ;
2011-02-09 20:25:14 +00:00
private final Map < String , World > worlds = new LinkedHashMap < String , World > ( ) ;
2010-12-22 15:22:23 +00:00
2011-01-03 19:41:23 +00:00
public CraftServer ( MinecraftServer console , ServerConfigurationManager server ) {
this . console = console ;
this . server = server ;
2011-02-01 21:55:30 +00:00
this . serverVersion = CraftServer . class . getPackage ( ) . getImplementationVersion ( ) ;
2010-12-24 17:24:21 +00:00
2011-03-05 12:33:11 +01:00
pluginManager . registerInterface ( JavaPluginLoader . class ) ;
2011-02-05 03:10:03 -05:00
Logger . getLogger ( " Minecraft " ) . log ( Level . INFO , " This server is running " + getName ( ) + " version " + getVersion ( ) ) ;
2011-01-07 02:33:59 +08:00
}
2011-01-17 15:50:37 -05:00
2011-01-07 02:33:59 +08:00
public void loadPlugins ( ) {
2011-01-18 01:13:45 +01:00
File pluginFolder = ( File ) console . options . valueOf ( " plugins " ) ;
2011-01-03 02:25:09 +00:00
2010-12-24 19:04:35 +00:00
if ( pluginFolder . exists ( ) ) {
try {
2011-01-11 23:11:10 +01:00
Plugin [ ] plugins = pluginManager . loadPlugins ( pluginFolder ) ;
2011-03-12 17:38:10 +00:00
for ( Plugin plugin : plugins ) {
try {
plugin . onLoad ( ) ;
} catch ( AbstractMethodError ame ) {
Logger . getLogger ( " Minecraft " ) . warning ( " Plugin: " + plugin . getDescription ( ) . getName ( ) + " does not support the onLoad() method " ) ;
}
}
2011-01-03 02:25:09 +00:00
for ( Plugin plugin : plugins ) {
2011-01-18 01:13:45 +01:00
loadPlugin ( plugin ) ;
2011-01-03 02:25:09 +00:00
}
2010-12-24 19:04:35 +00:00
} catch ( Throwable ex ) {
2011-01-05 22:56:31 +00:00
Logger . getLogger ( CraftServer . class . getName ( ) ) . log ( Level . SEVERE , ex . getMessage ( ) + " (Is it up to date?) " , ex ) ;
2010-12-24 19:04:35 +00:00
}
} else {
pluginFolder . mkdir ( ) ;
2010-12-24 17:24:21 +00:00
}
2010-12-21 16:52:15 +00:00
}
2011-01-27 16:15:41 -05:00
public void disablePlugins ( ) {
pluginManager . disablePlugins ( ) ;
}
2011-01-18 01:13:45 +01:00
private void loadPlugin ( Plugin plugin ) {
List < Command > pluginCommands = PluginCommandYamlParser . parse ( plugin ) ;
2011-02-27 14:55:29 +00:00
2011-01-18 01:13:45 +01:00
if ( ! pluginCommands . isEmpty ( ) ) {
commandMap . registerAll ( plugin . getDescription ( ) . getName ( ) , pluginCommands ) ;
}
2011-02-27 14:55:29 +00:00
try {
pluginManager . enablePlugin ( plugin ) ;
} catch ( Throwable ex ) {
Logger . getLogger ( CraftServer . class . getName ( ) ) . log ( Level . SEVERE , ex . getMessage ( ) + " loading " + plugin . getDescription ( ) . getFullName ( ) + " (Is it up to date?) " , ex ) ;
}
2011-01-18 01:13:45 +01:00
}
2010-12-21 16:52:15 +00:00
public String getName ( ) {
2010-12-22 15:22:23 +00:00
return serverName ;
2010-12-21 16:52:15 +00:00
}
public String getVersion ( ) {
2011-02-01 21:47:28 +00:00
return serverVersion + " (MC: " + protocolVersion + " ) " ;
2010-12-22 15:22:23 +00:00
}
public Player [ ] getOnlinePlayers ( ) {
2011-01-29 22:50:29 +01:00
List < EntityPlayer > online = server . b ;
2010-12-22 15:22:23 +00:00
Player [ ] players = new Player [ online . size ( ) ] ;
for ( int i = 0 ; i < players . length ; i + + ) {
2010-12-30 05:35:30 +00:00
players [ i ] = online . get ( i ) . a . getPlayer ( ) ;
2010-12-22 15:22:23 +00:00
}
return players ;
2010-12-21 16:52:15 +00:00
}
2011-01-03 00:16:00 +00:00
public Player getPlayer ( final String name ) {
Player [ ] players = getOnlinePlayers ( ) ;
2011-01-17 15:50:37 -05:00
Player found = null ;
String lowerName = name . toLowerCase ( ) ;
int delta = Integer . MAX_VALUE ;
2011-01-03 00:16:00 +00:00
for ( Player player : players ) {
2011-01-17 15:50:37 -05:00
if ( player . getName ( ) . toLowerCase ( ) . startsWith ( lowerName ) ) {
int curDelta = player . getName ( ) . length ( ) - lowerName . length ( ) ;
if ( curDelta < delta ) {
found = player ;
delta = curDelta ;
}
2011-03-11 16:25:35 -05:00
if ( curDelta = = 0 ) break ;
2011-01-03 00:16:00 +00:00
}
}
2011-01-17 15:50:37 -05:00
return found ;
2011-01-03 00:16:00 +00:00
}
2011-01-17 15:50:37 -05:00
2011-01-15 13:40:15 -08:00
public int broadcastMessage ( String message ) {
Player [ ] players = getOnlinePlayers ( ) ;
for ( Player player : players ) {
player . sendMessage ( message ) ;
}
return players . length ;
}
2011-01-03 00:16:00 +00:00
2011-01-29 22:50:29 +01:00
public Player getPlayer ( final EntityPlayer entity ) {
2010-12-30 05:35:30 +00:00
return entity . a . getPlayer ( ) ;
2010-12-26 02:20:29 +00:00
}
2011-01-17 15:50:37 -05:00
2011-01-07 10:42:53 +08:00
public List < Player > matchPlayer ( String partialName ) {
List < Player > matchedPlayers = new ArrayList < Player > ( ) ;
2011-01-17 15:50:37 -05:00
2011-01-06 13:10:28 +08:00
for ( Player iterPlayer : this . getOnlinePlayers ( ) ) {
String iterPlayerName = iterPlayer . getName ( ) ;
2011-01-07 10:42:53 +08:00
if ( partialName . equalsIgnoreCase ( iterPlayerName ) ) {
2011-01-06 13:10:28 +08:00
// Exact match
2011-01-10 09:30:34 +01:00
matchedPlayers . clear ( ) ;
matchedPlayers . add ( iterPlayer ) ;
break ;
}
2011-01-07 10:42:53 +08:00
if ( iterPlayerName . toLowerCase ( ) . indexOf ( partialName . toLowerCase ( ) ) ! = - 1 ) {
2011-01-10 09:30:34 +01:00
// Partial match
2011-01-07 10:42:53 +08:00
matchedPlayers . add ( iterPlayer ) ;
2011-01-06 13:10:28 +08:00
}
}
2011-01-07 10:42:53 +08:00
return matchedPlayers ;
2011-01-06 13:10:28 +08:00
}
2010-12-26 02:20:29 +00:00
2011-02-04 14:12:33 +08:00
public int getMaxPlayers ( ) {
2011-02-12 15:57:31 +01:00
return server . e ;
2011-02-04 14:12:33 +08:00
}
2011-02-06 13:30:50 +01:00
2011-03-12 18:23:57 +00:00
// NOTE: These are dependent on the corrisponding call in MinecraftServer
// so if that changes this will need to as well
public int getPort ( ) {
return this . getConfigInt ( " server-port " , 25565 ) ;
}
public String getIp ( ) {
return this . getConfigString ( " server-ip " , " " ) ;
}
2011-03-26 01:39:23 -07:00
public String getServerName ( ) {
2011-03-12 18:23:57 +00:00
return this . getConfigString ( " server-name " , " Unknown Server " ) ;
}
2011-03-26 01:39:23 -07:00
public String getServerId ( ) {
return this . getConfigString ( " server-id " , " unnamed " ) ;
}
2011-03-12 18:23:57 +00:00
// NOTE: Temporary calls through to server.properies until its replaced
2011-03-26 01:39:23 -07:00
private String getConfigString ( String variable , String defaultValue ) {
2011-03-12 18:23:57 +00:00
return this . console . d . a ( variable , defaultValue ) ;
}
2011-03-26 01:39:23 -07:00
private int getConfigInt ( String variable , int defaultValue ) {
2011-03-12 18:23:57 +00:00
return this . console . d . a ( variable , defaultValue ) ;
}
2011-03-26 01:39:23 -07:00
2011-03-12 18:23:57 +00:00
// End Temporary calls
2010-12-24 17:24:21 +00:00
public PluginManager getPluginManager ( ) {
return pluginManager ;
}
2010-12-27 02:13:03 +00:00
2011-02-02 23:53:04 +00:00
public BukkitScheduler getScheduler ( ) {
return scheduler ;
}
2011-02-07 01:59:06 +00:00
public List < World > getWorlds ( ) {
2011-02-08 12:03:36 +00:00
return new ArrayList < World > ( worlds . values ( ) ) ;
2010-12-27 02:13:03 +00:00
}
2010-12-30 04:30:12 +00:00
public ServerConfigurationManager getHandle ( ) {
return server ;
}
2011-01-16 16:35:37 +01:00
2011-03-12 18:23:57 +00:00
// NOTE: Should only be called from MinecraftServer.b()
public boolean dispatchCommand ( CommandSender sender , ServerCommand serverCommand ) {
if ( commandMap . dispatch ( sender , serverCommand . a ) ) {
return true ;
}
return console . o . a ( serverCommand ) ;
}
2011-01-29 21:04:02 +00:00
public boolean dispatchCommand ( CommandSender sender , String commandLine ) {
2011-03-12 18:23:57 +00:00
// CraftBukkit native commands
if ( commandMap . dispatch ( sender , commandLine ) ) {
return true ;
}
if ( ! sender . isOp ( ) ) {
return false ;
}
// See if the server can process this command
return console . o . a ( new ServerCommand ( commandLine , new CommandListener ( sender ) ) ) ;
2011-01-16 16:35:37 +01:00
}
2011-01-20 03:53:43 +00:00
public void reload ( ) {
PropertyManager config = new PropertyManager ( console . options ) ;
console . d = config ;
2011-01-29 21:04:02 +00:00
2011-01-20 03:53:43 +00:00
boolean animals = config . a ( " spawn-monsters " , console . m ) ;
2011-02-23 02:37:56 +00:00
boolean monsters = config . a ( " spawn-monsters " , console . worlds . get ( 0 ) . j > 0 ) ;
2011-01-20 03:53:43 +00:00
console . l = config . a ( " online-mode " , console . l ) ;
console . m = config . a ( " spawn-animals " , console . m ) ;
console . n = config . a ( " pvp " , console . n ) ;
2011-02-05 18:15:04 +00:00
for ( WorldServer world : console . worlds ) {
2011-02-23 02:37:56 +00:00
world . j = monsters ? 1 : 0 ;
2011-02-05 18:15:04 +00:00
world . a ( monsters , animals ) ;
}
2011-01-28 14:18:49 +00:00
pluginManager . clearPlugins ( ) ;
commandMap . clearCommands ( ) ;
loadPlugins ( ) ;
2011-01-20 03:53:43 +00:00
}
2011-02-02 18:23:19 +00:00
@Override
public String toString ( ) {
2011-03-12 18:23:57 +00:00
return " CraftServer{ " + " serverName= " + serverName + " ,serverVersion= " + serverVersion + " ,protocolVersion= " + protocolVersion + '}' ;
2011-02-02 18:23:19 +00:00
}
2011-02-05 20:51:22 +00:00
2011-02-06 20:50:57 +00:00
public World createWorld ( String name , World . Environment environment ) {
2011-02-05 20:51:22 +00:00
File folder = new File ( name ) ;
2011-02-08 12:03:36 +00:00
World world = getWorld ( name ) ;
if ( world ! = null ) {
return world ;
}
2011-03-12 18:23:57 +00:00
2011-02-05 20:51:22 +00:00
if ( ( folder . exists ( ) ) & & ( ! folder . isDirectory ( ) ) ) {
throw new IllegalArgumentException ( " File exists with the name ' " + name + " ' and isn't a folder " ) ;
}
2011-02-24 12:07:33 +00:00
Convertable converter = new WorldLoaderServer ( folder ) ;
if ( converter . a ( name ) ) {
getLogger ( ) . info ( " Converting world ' " + name + " ' " ) ;
converter . a ( name , new ConvertProgressUpdater ( console ) ) ;
}
2011-02-23 02:37:56 +00:00
WorldServer internal = new WorldServer ( console , new ServerNBTManager ( new File ( " . " ) , name , true ) , name , environment = = World . Environment . NETHER ? - 1 : 0 ) ;
2011-02-05 20:51:22 +00:00
internal . a ( new WorldManager ( console , internal ) ) ;
2011-02-23 02:37:56 +00:00
internal . j = 1 ;
2011-02-05 20:51:22 +00:00
internal . a ( true , true ) ;
console . f . a ( internal ) ;
console . worlds . add ( internal ) ;
short short1 = 196 ;
long i = System . currentTimeMillis ( ) ;
for ( int j = - short1 ; j < = short1 ; j + = 16 ) {
for ( int k = - short1 ; k < = short1 ; k + = 16 ) {
long l = System . currentTimeMillis ( ) ;
if ( l < i ) {
i = l ;
}
if ( l > i + 1000L ) {
int i1 = ( short1 * 2 + 1 ) * ( short1 * 2 + 1 ) ;
int j1 = ( j + short1 ) * ( short1 * 2 + 1 ) + k + 1 ;
System . out . println ( " Preparing spawn area for " + name + " , " + ( j1 * 100 / i1 ) + " % " ) ;
i = l ;
}
2011-02-23 02:37:56 +00:00
ChunkCoordinates chunkcoordinates = internal . l ( ) ;
internal . u . d ( chunkcoordinates . a + j > > 4 , chunkcoordinates . c + k > > 4 ) ;
2011-02-05 20:51:22 +00:00
2011-02-23 02:37:56 +00:00
while ( internal . e ( ) ) {
2011-02-05 20:51:22 +00:00
;
}
}
}
2011-02-25 21:29:42 -05:00
2011-03-24 22:43:21 +01:00
return internal . getWorld ( ) ;
2011-02-05 20:51:22 +00:00
}
public MinecraftServer getServer ( ) {
return console ;
}
2011-02-08 12:03:36 +00:00
public World getWorld ( String name ) {
return worlds . get ( name . toLowerCase ( ) ) ;
}
protected void addWorld ( World world ) {
worlds . put ( world . getName ( ) . toLowerCase ( ) , world ) ;
2011-02-08 12:50:36 +00:00
2011-03-26 23:20:35 +01:00
pluginManager . callEvent ( new WorldLoadEvent ( world ) ) ;
2011-02-08 12:03:36 +00:00
}
2011-02-20 01:53:06 +00:00
public Logger getLogger ( ) {
return MinecraftServer . a ;
}
2011-02-25 16:12:38 +00:00
public ConsoleReader getReader ( ) {
return console . reader ;
}
2011-02-28 00:31:25 +00:00
public PluginCommand getPluginCommand ( String name ) {
Command command = commandMap . getCommand ( name ) ;
if ( command instanceof PluginCommand ) {
return ( PluginCommand ) command ;
} else {
return null ;
}
}
2011-02-25 21:29:42 -05:00
public void savePlayers ( ) {
server . d ( ) ;
}
2011-03-12 18:23:57 +00:00
// Inner class to capture the output of default server commands
class CommandListener implements ICommandListener {
private final CommandSender commandSender ;
private final String prefix ;
CommandListener ( CommandSender commandSender ) {
this . commandSender = commandSender ;
String [ ] parts = commandSender . getClass ( ) . getName ( ) . split ( " \\ . " ) ;
this . prefix = parts [ parts . length - 1 ] ;
}
public void b ( String msg ) {
this . commandSender . sendMessage ( msg ) ;
}
public String c ( ) {
return this . prefix ;
}
}
2010-12-21 16:52:15 +00:00
}