2010-12-21 17:52:15 +01:00
package org.bukkit.craftbukkit ;
2013-11-04 14:07:38 +01:00
import java.awt.image.BufferedImage ;
2010-12-24 18:24:21 +01:00
import java.io.File ;
2011-07-17 18:19:41 +02:00
import java.io.FileInputStream ;
2012-01-30 21:51:53 +01:00
import java.io.FileNotFoundException ;
2011-08-13 03:13:35 +02:00
import java.io.IOException ;
2014-05-14 12:49:04 +02:00
import java.io.InputStreamReader ;
2011-01-07 03:42:53 +01:00
import java.util.ArrayList ;
2011-12-27 06:48:09 +01:00
import java.util.Arrays ;
2012-10-17 11:31:36 +02:00
import java.util.Collections ;
2011-09-03 02:59:28 +02:00
import java.util.HashSet ;
2011-07-24 05:16:14 +02:00
import java.util.Iterator ;
2011-02-09 21:25:14 +01:00
import java.util.LinkedHashMap ;
2012-06-15 05:43:46 +02:00
import java.util.LinkedHashSet ;
2010-12-22 16:22:23 +01:00
import java.util.List ;
2011-02-08 13:03:36 +01:00
import java.util.Map ;
2011-07-17 18:19:41 +02:00
import java.util.Set ;
2014-04-17 08:53:13 +02:00
import java.util.UUID ;
2010-12-24 18:24:21 +01:00
import java.util.logging.Level ;
import java.util.logging.Logger ;
2014-04-17 08:53:13 +02:00
import java.util.regex.Pattern ;
2012-01-30 21:51:53 +01:00
2013-11-04 14:07:38 +01:00
import javax.imageio.ImageIO ;
2014-11-25 22:32:16 +01:00
import net.minecraft.server.* ;
2012-01-30 21:51:53 +01:00
2014-02-05 04:52:50 +01:00
import org.bukkit.BanList ;
2012-01-30 21:51:53 +01:00
import org.bukkit.Bukkit ;
import org.bukkit.ChatColor ;
import org.bukkit.GameMode ;
import org.bukkit.OfflinePlayer ;
import org.bukkit.Server ;
2014-01-15 05:38:03 +01:00
import org.bukkit.UnsafeValues ;
2012-08-07 06:52:49 +02:00
import org.bukkit.Warning.WarningState ;
2012-01-30 21:51:53 +01:00
import org.bukkit.World ;
import org.bukkit.World.Environment ;
import org.bukkit.WorldCreator ;
import org.bukkit.command.Command ;
2012-10-09 20:44:04 +02:00
import org.bukkit.command.CommandException ;
2012-01-30 21:51:53 +01:00
import org.bukkit.command.CommandSender ;
import org.bukkit.command.ConsoleCommandSender ;
import org.bukkit.command.PluginCommand ;
import org.bukkit.command.SimpleCommandMap ;
2011-10-10 22:10:48 +02:00
import org.bukkit.configuration.ConfigurationSection ;
import org.bukkit.configuration.file.YamlConfiguration ;
import org.bukkit.configuration.serialization.ConfigurationSerialization ;
2012-01-19 09:32:05 +01:00
import org.bukkit.conversations.Conversable ;
2014-02-08 00:40:59 +01:00
import org.bukkit.craftbukkit.command.VanillaCommandWrapper ;
2014-06-24 23:11:25 +02:00
import org.bukkit.craftbukkit.entity.CraftPlayer ;
2012-03-01 06:19:11 +01:00
import org.bukkit.craftbukkit.help.SimpleHelpMap ;
2011-04-16 04:11:13 +02:00
import org.bukkit.craftbukkit.inventory.CraftFurnaceRecipe ;
2012-02-29 19:56:35 +01:00
import org.bukkit.craftbukkit.inventory.CraftInventoryCustom ;
2012-12-17 08:31:41 +01:00
import org.bukkit.craftbukkit.inventory.CraftItemFactory ;
2011-04-16 04:11:13 +02:00
import org.bukkit.craftbukkit.inventory.CraftRecipe ;
import org.bukkit.craftbukkit.inventory.CraftShapedRecipe ;
import org.bukkit.craftbukkit.inventory.CraftShapelessRecipe ;
2011-07-24 05:16:14 +02:00
import org.bukkit.craftbukkit.inventory.RecipeIterator ;
2011-08-13 04:47:47 +02:00
import org.bukkit.craftbukkit.map.CraftMapView ;
2011-12-08 06:33:59 +01:00
import org.bukkit.craftbukkit.metadata.EntityMetadataStore ;
import org.bukkit.craftbukkit.metadata.PlayerMetadataStore ;
import org.bukkit.craftbukkit.metadata.WorldMetadataStore ;
2012-01-09 08:51:32 +01:00
import org.bukkit.craftbukkit.potion.CraftPotionBrewer ;
2011-02-03 00:53:04 +01:00
import org.bukkit.craftbukkit.scheduler.CraftScheduler ;
2013-03-22 22:21:33 +01:00
import org.bukkit.craftbukkit.scoreboard.CraftScoreboardManager ;
2013-11-04 14:07:38 +01:00
import org.bukkit.craftbukkit.util.CraftIconCache ;
2014-01-15 05:38:03 +01:00
import org.bukkit.craftbukkit.util.CraftMagicNumbers ;
2011-12-04 18:57:00 +01:00
import org.bukkit.craftbukkit.util.DatFileFilter ;
2011-10-31 05:02:54 +01:00
import org.bukkit.craftbukkit.util.Versioning ;
2012-01-30 21:51:53 +01:00
import org.bukkit.entity.Player ;
2012-02-29 19:56:35 +01:00
import org.bukkit.event.inventory.InventoryType ;
2012-10-17 11:31:36 +02:00
import org.bukkit.event.player.PlayerChatTabCompleteEvent ;
2011-06-06 15:52:02 +02:00
import org.bukkit.event.world.WorldInitEvent ;
2012-01-30 21:51:53 +01:00
import org.bukkit.event.world.WorldLoadEvent ;
import org.bukkit.event.world.WorldSaveEvent ;
import org.bukkit.event.world.WorldUnloadEvent ;
import org.bukkit.generator.ChunkGenerator ;
2012-03-01 06:19:11 +01:00
import org.bukkit.help.HelpMap ;
2012-01-30 21:51:53 +01:00
import org.bukkit.inventory.FurnaceRecipe ;
2011-07-24 05:16:14 +02:00
import org.bukkit.inventory.ItemStack ;
2012-02-29 19:56:35 +01:00
import org.bukkit.inventory.Inventory ;
import org.bukkit.inventory.InventoryHolder ;
2012-01-30 21:51:53 +01:00
import org.bukkit.inventory.Recipe ;
import org.bukkit.inventory.ShapedRecipe ;
import org.bukkit.inventory.ShapelessRecipe ;
import org.bukkit.permissions.Permissible ;
2011-07-17 18:19:41 +02:00
import org.bukkit.permissions.Permission ;
2012-01-30 21:51:53 +01:00
import org.bukkit.plugin.Plugin ;
2011-06-22 18:46:12 +02:00
import org.bukkit.plugin.PluginLoadOrder ;
2012-01-30 21:51:53 +01:00
import org.bukkit.plugin.PluginManager ;
import org.bukkit.plugin.ServicesManager ;
import org.bukkit.plugin.SimplePluginManager ;
import org.bukkit.plugin.SimpleServicesManager ;
import org.bukkit.plugin.java.JavaPluginLoader ;
import org.bukkit.plugin.messaging.Messenger ;
2012-01-09 08:51:32 +01:00
import org.bukkit.potion.Potion ;
import org.bukkit.potion.PotionEffectType ;
2012-01-13 09:52:26 +01:00
import org.bukkit.plugin.messaging.StandardMessenger ;
2012-01-30 21:51:53 +01:00
import org.bukkit.scheduler.BukkitWorker ;
2012-10-17 11:31:36 +02:00
import org.bukkit.util.StringUtil ;
2012-01-30 21:51:53 +01:00
import org.bukkit.util.permissions.DefaultPermissions ;
2011-07-17 18:19:41 +02:00
import org.yaml.snakeyaml.Yaml ;
import org.yaml.snakeyaml.constructor.SafeConstructor ;
import org.yaml.snakeyaml.error.MarkedYAMLException ;
2012-02-29 19:56:35 +01:00
import org.apache.commons.lang.Validate ;
2014-02-05 04:52:50 +01:00
2012-01-30 21:51:53 +01:00
import com.avaje.ebean.config.DataSourceConfig ;
import com.avaje.ebean.config.ServerConfig ;
import com.avaje.ebean.config.dbplatform.SQLitePlatform ;
import com.avaje.ebeaninternal.server.lib.sql.TransactionIsolation ;
2014-11-25 22:32:16 +01:00
import com.google.common.base.Charsets ;
import com.google.common.base.Function ;
2012-02-29 17:27:38 +01:00
import com.google.common.collect.ImmutableList ;
2014-11-25 22:32:16 +01:00
import com.google.common.collect.Lists ;
2012-01-30 21:51:53 +01:00
import com.google.common.collect.MapMaker ;
2014-11-25 22:32:16 +01:00
import com.mojang.authlib.GameProfile ;
import io.netty.buffer.ByteBuf ;
import io.netty.buffer.ByteBufOutputStream ;
import io.netty.buffer.Unpooled ;
import io.netty.handler.codec.base64.Base64 ;
2012-01-30 21:51:53 +01:00
2012-03-14 21:59:14 +01:00
import jline.console.ConsoleReader ;
2012-01-30 21:51:53 +01:00
2011-01-20 04:53:43 +01:00
public final class CraftServer implements Server {
2014-06-24 23:11:25 +02:00
private static final Player [ ] EMPTY_PLAYER_ARRAY = new Player [ 0 ] ;
2012-01-28 00:23:16 +01:00
private final String serverName = " CraftBukkit " ;
2011-02-01 22:55:30 +01:00
private final String serverVersion ;
2011-10-31 05:02:54 +01:00
private final String bukkitVersion = Versioning . getBukkitVersion ( ) ;
2014-01-21 19:47:18 +01:00
private final Logger logger = Logger . getLogger ( " Minecraft " ) ;
2011-05-02 20:31:02 +02:00
private final ServicesManager servicesManager = new SimpleServicesManager ( ) ;
2012-07-29 09:33:13 +02:00
private final CraftScheduler scheduler = new CraftScheduler ( ) ;
2011-06-17 05:09:43 +02:00
private final SimpleCommandMap commandMap = new SimpleCommandMap ( this ) ;
2012-03-15 04:39:19 +01:00
private final SimpleHelpMap helpMap = new SimpleHelpMap ( this ) ;
2012-01-13 09:52:26 +01:00
private final StandardMessenger messenger = new StandardMessenger ( ) ;
2011-06-26 03:46:04 +02:00
private final PluginManager pluginManager = new SimplePluginManager ( this , commandMap ) ;
2010-12-22 16:22:23 +01:00
protected final MinecraftServer console ;
2012-12-20 05:03:52 +01:00
protected final DedicatedPlayerList playerList ;
2011-02-09 21:25:14 +01:00
private final Map < String , World > worlds = new LinkedHashMap < String , World > ( ) ;
2011-10-10 22:10:48 +02:00
private YamlConfiguration configuration ;
2014-02-08 00:40:59 +01:00
private YamlConfiguration commandsConfiguration ;
2011-07-17 18:19:41 +02:00
private final Yaml yaml = new Yaml ( new SafeConstructor ( ) ) ;
2014-04-17 08:53:13 +02:00
private final Map < UUID , OfflinePlayer > offlinePlayers = new MapMaker ( ) . softValues ( ) . makeMap ( ) ;
2011-12-08 06:33:59 +01:00
private final EntityMetadataStore entityMetadata = new EntityMetadataStore ( ) ;
private final PlayerMetadataStore playerMetadata = new PlayerMetadataStore ( ) ;
private final WorldMetadataStore worldMetadata = new WorldMetadataStore ( ) ;
2012-04-24 02:01:22 +02:00
private int monsterSpawn = - 1 ;
private int animalSpawn = - 1 ;
private int waterAnimalSpawn = - 1 ;
2012-10-31 17:18:41 +01:00
private int ambientSpawn = - 1 ;
[Bleeding] Implement periodic chunk garbage collector
This adds two settings to bukkit.yml, allowing activation and control of
two chunk garbage collection triggering conditions:
chunk-gc/period-in-ticks controls a periodic GC, run once every N ticks
(default is 600); chunk-gc/load-threshold causes the GC to run once
after every N calls to loadChunk() on a given world (this call is an API
call used by plugins, and is distinct from the path taken for routine
player movement-based loading). In both cases, setting to zero will
disable the given GC scheduling strategy.
In either case, the act of doing the GC is simply one of scanning the
loaded chunks, seeing which are NOT being used by one or more players
(due to view-distance) and which are not already queued for unload, and
queueing them for a normal unload. Ultimately, the unload is then
processed the same as if the chunk were unloaded due to leaving the
view-distance range of all players, so the impact on plugins should be
no different (and strategies such as handling the ChunkUnloadEvent in
order to prevent unload will still work).
The initial interval for the periodic GC is randomized on a per-world
basis, in order to avoid all world being GCed at the same time -
minimizing potential lag spikes.
2012-12-10 16:38:26 +01:00
public int chunkGCPeriod = - 1 ;
public int chunkGCLoadThresh = 0 ;
2012-07-29 09:33:13 +02:00
private File container ;
2012-08-07 06:52:49 +02:00
private WarningState warningState = WarningState . DEFAULT ;
2012-12-20 05:03:52 +01:00
private final BooleanWrapper online = new BooleanWrapper ( ) ;
2013-03-22 22:21:33 +01:00
public CraftScoreboardManager scoreboardManager ;
2013-09-23 23:43:21 +02:00
public boolean playerCommandState ;
private boolean printSaveWarning ;
2013-11-04 14:07:38 +01:00
private CraftIconCache icon ;
2014-02-08 00:40:59 +01:00
private boolean overrideAllCommandBlockCommands = false ;
2014-04-17 08:53:13 +02:00
private final Pattern validUserPattern = Pattern . compile ( " ^[a-zA-Z0-9_]{2,16}$ " ) ;
private final UUID invalidUserUUID = UUID . nameUUIDFromBytes ( " InvalidUsername " . getBytes ( Charsets . UTF_8 ) ) ;
2014-06-24 23:11:25 +02:00
private final List < CraftPlayer > playerView ;
2012-12-20 05:03:52 +01:00
private final class BooleanWrapper {
private boolean value = true ;
}
2010-12-22 16:22:23 +01:00
2011-10-10 22:10:48 +02:00
static {
ConfigurationSerialization . registerClass ( CraftOfflinePlayer . class ) ;
2012-12-17 08:31:41 +01:00
CraftItemFactory . instance ( ) ;
2011-10-10 22:10:48 +02:00
}
2012-12-20 05:03:52 +01:00
public CraftServer ( MinecraftServer console , PlayerList playerList ) {
2011-01-03 20:41:23 +01:00
this . console = console ;
2012-12-20 05:03:52 +01:00
this . playerList = ( DedicatedPlayerList ) playerList ;
2014-11-25 22:32:16 +01:00
this . playerView = Collections . unmodifiableList ( Lists . transform ( playerList . players , new Function < EntityPlayer , CraftPlayer > ( ) {
2014-06-24 23:11:25 +02:00
@Override
public CraftPlayer apply ( EntityPlayer player ) {
return player . getBukkitEntity ( ) ;
}
} ) ) ;
2011-02-01 22:55:30 +01:00
this . serverVersion = CraftServer . class . getPackage ( ) . getImplementationVersion ( ) ;
2012-12-20 05:03:52 +01:00
online . value = console . getPropertyManager ( ) . getBoolean ( " online-mode " , true ) ;
2010-12-24 18:24:21 +01:00
2011-03-30 15:55:42 +02:00
Bukkit . setServer ( this ) ;
2011-11-20 09:01:14 +01:00
2012-01-09 08:51:32 +01:00
// Register all the Enchantments and PotionTypes now so we can stop new registration immediately after
2011-11-27 01:39:25 +01:00
Enchantment . DAMAGE_ALL . getClass ( ) ;
org . bukkit . enchantments . Enchantment . stopAcceptingRegistrations ( ) ;
2012-01-09 08:51:32 +01:00
Potion . setPotionBrewer ( new CraftPotionBrewer ( ) ) ;
MobEffectList . BLINDNESS . getClass ( ) ;
PotionEffectType . stopAcceptingRegistrations ( ) ;
2011-11-27 01:39:25 +01:00
// Ugly hack :(
2011-11-05 21:14:26 +01:00
if ( ! Main . useConsole ) {
getLogger ( ) . info ( " Console input is disabled due to --noconsole command argument " ) ;
}
2011-03-30 15:55:42 +02:00
2011-10-10 22:10:48 +02:00
configuration = YamlConfiguration . loadConfiguration ( getConfigFile ( ) ) ;
configuration . options ( ) . copyDefaults ( true ) ;
2014-05-14 12:49:04 +02:00
configuration . setDefaults ( YamlConfiguration . loadConfiguration ( new InputStreamReader ( getClass ( ) . getClassLoader ( ) . getResourceAsStream ( " configurations/bukkit.yml " ) , Charsets . UTF_8 ) ) ) ;
2014-02-08 00:40:59 +01:00
ConfigurationSection legacyAlias = null ;
if ( ! configuration . isString ( " aliases " ) ) {
legacyAlias = configuration . getConfigurationSection ( " aliases " ) ;
configuration . set ( " aliases " , " now-in-commands.yml " ) ;
}
2011-10-10 22:10:48 +02:00
saveConfig ( ) ;
2014-02-08 00:40:59 +01:00
if ( getCommandsConfigFile ( ) . isFile ( ) ) {
legacyAlias = null ;
}
commandsConfiguration = YamlConfiguration . loadConfiguration ( getCommandsConfigFile ( ) ) ;
commandsConfiguration . options ( ) . copyDefaults ( true ) ;
2014-05-14 12:49:04 +02:00
commandsConfiguration . setDefaults ( YamlConfiguration . loadConfiguration ( new InputStreamReader ( getClass ( ) . getClassLoader ( ) . getResourceAsStream ( " configurations/commands.yml " ) , Charsets . UTF_8 ) ) ) ;
2014-02-08 00:40:59 +01:00
saveCommandsConfig ( ) ;
2014-02-08 22:02:44 +01:00
// Migrate aliases from old file and add previously implicit $1- to pass all arguments
2014-02-08 00:40:59 +01:00
if ( legacyAlias ! = null ) {
ConfigurationSection aliases = commandsConfiguration . createSection ( " aliases " ) ;
2014-02-08 22:02:44 +01:00
for ( String key : legacyAlias . getKeys ( false ) ) {
ArrayList < String > commands = new ArrayList < String > ( ) ;
if ( legacyAlias . isList ( key ) ) {
for ( String command : legacyAlias . getStringList ( key ) ) {
commands . add ( command + " $1- " ) ;
}
} else {
commands . add ( legacyAlias . getString ( key ) + " $1- " ) ;
}
aliases . set ( key , commands ) ;
2014-02-08 00:40:59 +01:00
}
}
2014-02-08 22:02:44 +01:00
2014-02-08 00:40:59 +01:00
saveCommandsConfig ( ) ;
overrideAllCommandBlockCommands = commandsConfiguration . getStringList ( " command-block-overrides " ) . contains ( " * " ) ;
2012-02-10 20:56:16 +01:00
( ( SimplePluginManager ) pluginManager ) . useTimings ( configuration . getBoolean ( " settings.plugin-profiling " ) ) ;
2012-04-24 02:01:22 +02:00
monsterSpawn = configuration . getInt ( " spawn-limits.monsters " ) ;
animalSpawn = configuration . getInt ( " spawn-limits.animals " ) ;
waterAnimalSpawn = configuration . getInt ( " spawn-limits.water-animals " ) ;
2012-10-31 17:18:41 +01:00
ambientSpawn = configuration . getInt ( " spawn-limits.ambient " ) ;
2012-08-19 00:36:39 +02:00
console . autosavePeriod = configuration . getInt ( " ticks-per.autosave " ) ;
2012-08-07 06:52:49 +02:00
warningState = WarningState . value ( configuration . getString ( " settings.deprecated-verbose " ) ) ;
[Bleeding] Implement periodic chunk garbage collector
This adds two settings to bukkit.yml, allowing activation and control of
two chunk garbage collection triggering conditions:
chunk-gc/period-in-ticks controls a periodic GC, run once every N ticks
(default is 600); chunk-gc/load-threshold causes the GC to run once
after every N calls to loadChunk() on a given world (this call is an API
call used by plugins, and is distinct from the path taken for routine
player movement-based loading). In both cases, setting to zero will
disable the given GC scheduling strategy.
In either case, the act of doing the GC is simply one of scanning the
loaded chunks, seeing which are NOT being used by one or more players
(due to view-distance) and which are not already queued for unload, and
queueing them for a normal unload. Ultimately, the unload is then
processed the same as if the chunk were unloaded due to leaving the
view-distance range of all players, so the impact on plugins should be
no different (and strategies such as handling the ChunkUnloadEvent in
order to prevent unload will still work).
The initial interval for the periodic GC is randomized on a per-world
basis, in order to avoid all world being GCed at the same time -
minimizing potential lag spikes.
2012-12-10 16:38:26 +01:00
chunkGCPeriod = configuration . getInt ( " chunk-gc.period-in-ticks " ) ;
chunkGCLoadThresh = configuration . getInt ( " chunk-gc.load-threshold " ) ;
2013-11-04 14:07:38 +01:00
loadIcon ( ) ;
2011-11-20 09:01:14 +01:00
2011-06-22 18:46:12 +02:00
loadPlugins ( ) ;
enablePlugins ( PluginLoadOrder . STARTUP ) ;
2011-03-31 03:35:08 +02:00
}
2011-11-20 09:01:14 +01:00
2014-02-08 00:40:59 +01:00
public boolean getCommandBlockOverride ( String command ) {
return overrideAllCommandBlockCommands | | commandsConfiguration . getStringList ( " command-block-overrides " ) . contains ( command ) ;
}
2011-10-10 22:10:48 +02:00
private File getConfigFile ( ) {
2012-01-14 23:02:10 +01:00
return ( File ) console . options . valueOf ( " bukkit-settings " ) ;
2011-10-10 22:10:48 +02:00
}
2011-11-20 09:01:14 +01:00
2014-02-08 00:40:59 +01:00
private File getCommandsConfigFile ( ) {
return ( File ) console . options . valueOf ( " commands-settings " ) ;
}
2011-10-10 22:10:48 +02:00
private void saveConfig ( ) {
try {
configuration . save ( getConfigFile ( ) ) ;
} catch ( IOException ex ) {
Logger . getLogger ( CraftServer . class . getName ( ) ) . log ( Level . SEVERE , " Could not save " + getConfigFile ( ) , ex ) ;
2011-06-17 05:09:43 +02:00
}
2011-01-06 19:33:59 +01:00
}
2011-01-17 21:50:37 +01:00
2014-02-08 00:40:59 +01:00
private void saveCommandsConfig ( ) {
try {
commandsConfiguration . save ( getCommandsConfigFile ( ) ) ;
} catch ( IOException ex ) {
Logger . getLogger ( CraftServer . class . getName ( ) ) . log ( Level . SEVERE , " Could not save " + getCommandsConfigFile ( ) , ex ) ;
}
}
2011-01-06 19:33:59 +01:00
public void loadPlugins ( ) {
2011-03-07 19:58:19 +01:00
pluginManager . registerInterface ( JavaPluginLoader . class ) ;
2011-06-12 01:12:43 +02:00
File pluginFolder = ( File ) console . options . valueOf ( " plugins " ) ;
2011-01-03 03:25:09 +01:00
2010-12-24 20:04:35 +01:00
if ( pluginFolder . exists ( ) ) {
2011-06-22 18:46:12 +02:00
Plugin [ ] plugins = pluginManager . loadPlugins ( pluginFolder ) ;
for ( Plugin plugin : plugins ) {
try {
2012-02-15 20:56:36 +01:00
String message = String . format ( " Loading %s " , plugin . getDescription ( ) . getFullName ( ) ) ;
plugin . getLogger ( ) . info ( message ) ;
2011-06-22 18:46:12 +02:00
plugin . onLoad ( ) ;
} catch ( Throwable ex ) {
Logger . getLogger ( CraftServer . class . getName ( ) ) . log ( Level . SEVERE , ex . getMessage ( ) + " initializing " + plugin . getDescription ( ) . getFullName ( ) + " (Is it up to date?) " , ex ) ;
2011-03-12 18:38:10 +01:00
}
2010-12-24 20:04:35 +01:00
}
} else {
pluginFolder . mkdir ( ) ;
2010-12-24 18:24:21 +01:00
}
2010-12-21 17:52:15 +01:00
}
2011-06-22 18:46:12 +02:00
public void enablePlugins ( PluginLoadOrder type ) {
2012-03-01 06:19:11 +01:00
if ( type = = PluginLoadOrder . STARTUP ) {
helpMap . clear ( ) ;
2012-03-15 04:39:19 +01:00
helpMap . initializeGeneralTopics ( ) ;
2012-03-01 06:19:11 +01:00
}
2011-06-22 18:46:12 +02:00
Plugin [ ] plugins = pluginManager . getPlugins ( ) ;
2011-06-24 01:57:46 +02:00
2011-06-22 18:46:12 +02:00
for ( Plugin plugin : plugins ) {
if ( ( ! plugin . isEnabled ( ) ) & & ( plugin . getDescription ( ) . getLoad ( ) = = type ) ) {
loadPlugin ( plugin ) ;
}
}
2011-06-24 01:57:46 +02:00
if ( type = = PluginLoadOrder . POSTWORLD ) {
2014-02-08 00:40:59 +01:00
commandMap . setFallbackCommands ( ) ;
setVanillaCommands ( ) ;
2011-06-24 01:57:46 +02:00
commandMap . registerServerAliases ( ) ;
2011-07-17 18:19:41 +02:00
loadCustomPermissions ( ) ;
2011-08-18 15:04:59 +02:00
DefaultPermissions . registerCorePermissions ( ) ;
2012-03-15 04:39:19 +01:00
helpMap . initializeCommands ( ) ;
2011-06-24 01:57:46 +02:00
}
2011-06-22 18:46:12 +02:00
}
2011-01-27 22:15:41 +01:00
public void disablePlugins ( ) {
pluginManager . disablePlugins ( ) ;
}
2014-02-08 00:40:59 +01:00
private void setVanillaCommands ( ) {
2014-11-25 22:32:16 +01:00
Map < String , ICommand > commands = new CommandDispatcher ( ) . getCommands ( ) ;
for ( ICommand cmd : commands . values ( ) ) {
commandMap . register ( " minecraft " , new VanillaCommandWrapper ( ( CommandAbstract ) cmd , LocaleI18n . get ( cmd . getUsage ( null ) ) ) ) ;
}
2014-02-08 00:40:59 +01:00
}
2011-01-18 01:13:45 +01:00
private void loadPlugin ( Plugin plugin ) {
2011-02-27 15:55:29 +01:00
try {
pluginManager . enablePlugin ( plugin ) ;
2011-07-17 18:19:41 +02:00
List < Permission > perms = plugin . getDescription ( ) . getPermissions ( ) ;
for ( Permission perm : perms ) {
try {
pluginManager . addPermission ( perm ) ;
} catch ( IllegalArgumentException ex ) {
getLogger ( ) . log ( Level . WARNING , " Plugin " + plugin . getDescription ( ) . getFullName ( ) + " tried to register permission ' " + perm . getName ( ) + " ' but it's already registered " , ex ) ;
}
}
2011-02-27 15:55:29 +01:00
} 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
}
2014-06-24 23:11:25 +02:00
@Override
2010-12-21 17:52:15 +01:00
public String getName ( ) {
2010-12-22 16:22:23 +01:00
return serverName ;
2010-12-21 17:52:15 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
2010-12-21 17:52:15 +01:00
public String getVersion ( ) {
2011-11-20 09:01:14 +01:00
return serverVersion + " (MC: " + console . getVersion ( ) + " ) " ;
2010-12-22 16:22:23 +01:00
}
2011-11-20 09:01:14 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-10-31 05:02:54 +01:00
public String getBukkitVersion ( ) {
return bukkitVersion ;
}
2010-12-22 16:22:23 +01:00
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-07-17 15:34:40 +02:00
@SuppressWarnings ( " unchecked " )
2014-06-24 23:11:25 +02:00
public Player [ ] _INVALID_getOnlinePlayers ( ) {
return getOnlinePlayers ( ) . toArray ( EMPTY_PLAYER_ARRAY ) ;
}
2010-12-22 16:22:23 +01:00
2014-06-24 23:11:25 +02:00
@Override
public List < CraftPlayer > getOnlinePlayers ( ) {
return this . playerView ;
2010-12-21 17:52:15 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-01-03 01:16:00 +01:00
public Player getPlayer ( final String name ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( name , " Name cannot be null " ) ;
2011-01-17 21:50:37 +01:00
Player found = null ;
String lowerName = name . toLowerCase ( ) ;
int delta = Integer . MAX_VALUE ;
2014-06-24 23:11:25 +02:00
for ( Player player : getOnlinePlayers ( ) ) {
2011-01-17 21:50:37 +01:00
if ( player . getName ( ) . toLowerCase ( ) . startsWith ( lowerName ) ) {
int curDelta = player . getName ( ) . length ( ) - lowerName . length ( ) ;
if ( curDelta < delta ) {
found = player ;
delta = curDelta ;
}
2011-03-11 22:25:35 +01:00
if ( curDelta = = 0 ) break ;
2011-01-03 01:16:00 +01:00
}
}
2011-01-17 21:50:37 +01:00
return found ;
2011-01-03 01:16:00 +01:00
}
2011-01-17 21:50:37 +01:00
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-09-03 16:57:54 +02:00
public Player getPlayerExact ( String name ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( name , " Name cannot be null " ) ;
2011-09-03 16:57:54 +02:00
String lname = name . toLowerCase ( ) ;
for ( Player player : getOnlinePlayers ( ) ) {
if ( player . getName ( ) . equalsIgnoreCase ( lname ) ) {
return player ;
}
}
return null ;
}
2014-04-17 08:53:13 +02:00
// TODO: In 1.8+ this should use the server's UUID->EntityPlayer map
2014-06-24 23:11:25 +02:00
@Override
2014-04-17 08:53:13 +02:00
public Player getPlayer ( UUID id ) {
2014-03-29 05:48:15 +01:00
for ( Player player : getOnlinePlayers ( ) ) {
if ( player . getUniqueId ( ) . equals ( id ) ) {
return player ;
}
}
return null ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-01-15 22:40:15 +01:00
public int broadcastMessage ( String message ) {
2011-09-02 23:24:28 +02:00
return broadcast ( message , BROADCAST_CHANNEL_USERS ) ;
2011-01-15 22:40:15 +01:00
}
2011-01-03 01:16:00 +01:00
2011-01-29 22:50:29 +01:00
public Player getPlayer ( final EntityPlayer entity ) {
2014-11-25 22:32:16 +01:00
return entity . getBukkitEntity ( ) ;
2010-12-26 03:20:29 +01:00
}
2011-01-17 21:50:37 +01:00
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-01-07 03:42:53 +01:00
public List < Player > matchPlayer ( String partialName ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( partialName , " PartialName cannot be null " ) ;
2011-01-07 03:42:53 +01:00
List < Player > matchedPlayers = new ArrayList < Player > ( ) ;
2011-01-17 21:50:37 +01:00
2011-01-06 06:10:28 +01:00
for ( Player iterPlayer : this . getOnlinePlayers ( ) ) {
String iterPlayerName = iterPlayer . getName ( ) ;
2011-01-07 03:42:53 +01:00
if ( partialName . equalsIgnoreCase ( iterPlayerName ) ) {
2011-01-06 06:10:28 +01:00
// Exact match
2011-01-10 09:30:34 +01:00
matchedPlayers . clear ( ) ;
matchedPlayers . add ( iterPlayer ) ;
break ;
}
2012-09-10 06:19:28 +02:00
if ( iterPlayerName . toLowerCase ( ) . contains ( partialName . toLowerCase ( ) ) ) {
2011-01-10 09:30:34 +01:00
// Partial match
2011-01-07 03:42:53 +01:00
matchedPlayers . add ( iterPlayer ) ;
2011-01-06 06:10:28 +01:00
}
}
2011-01-07 03:42:53 +01:00
return matchedPlayers ;
2011-01-06 06:10:28 +01:00
}
2010-12-26 03:20:29 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-02-04 07:12:33 +01:00
public int getMaxPlayers ( ) {
2012-12-20 05:03:52 +01:00
return playerList . getMaxPlayers ( ) ;
2011-02-04 07:12:33 +01:00
}
2011-02-06 13:30:50 +01:00
2011-03-12 19:23:57 +01:00
// NOTE: These are dependent on the corrisponding call in MinecraftServer
// so if that changes this will need to as well
2014-06-24 23:11:25 +02:00
@Override
2011-03-12 19:23:57 +01:00
public int getPort ( ) {
return this . getConfigInt ( " server-port " , 25565 ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-08-15 20:27:02 +02:00
public int getViewDistance ( ) {
return this . getConfigInt ( " view-distance " , 10 ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-03-12 19:23:57 +01:00
public String getIp ( ) {
return this . getConfigString ( " server-ip " , " " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-03-26 09:39:23 +01:00
public String getServerName ( ) {
2011-03-12 19:23:57 +01:00
return this . getConfigString ( " server-name " , " Unknown Server " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-03-26 09:39:23 +01:00
public String getServerId ( ) {
return this . getConfigString ( " server-id " , " unnamed " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-03-10 08:28:32 +01:00
public String getWorldType ( ) {
return this . getConfigString ( " level-type " , " DEFAULT " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-03-10 08:28:32 +01:00
public boolean getGenerateStructures ( ) {
return this . getConfigBoolean ( " generate-structures " , true ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-12-09 18:55:36 +01:00
public boolean getAllowEnd ( ) {
return this . configuration . getBoolean ( " settings.allow-end " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-08-08 02:25:46 +02:00
public boolean getAllowNether ( ) {
return this . getConfigBoolean ( " allow-nether " , true ) ;
}
2012-01-15 11:11:25 +01:00
public boolean getWarnOnOverload ( ) {
return this . configuration . getBoolean ( " settings.warn-on-overload " ) ;
}
2012-07-04 01:00:37 +02:00
2012-06-09 19:59:52 +02:00
public boolean getQueryPlugins ( ) {
return this . configuration . getBoolean ( " settings.query-plugins " ) ;
2012-07-04 01:00:37 +02:00
}
2012-01-15 11:11:25 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-08-08 02:25:46 +02:00
public boolean hasWhitelist ( ) {
return this . getConfigBoolean ( " white-list " , false ) ;
}
2011-03-12 19:23:57 +01:00
// NOTE: Temporary calls through to server.properies until its replaced
2011-03-26 09:39:23 +01:00
private String getConfigString ( String variable , String defaultValue ) {
2012-07-29 09:33:13 +02:00
return this . console . getPropertyManager ( ) . getString ( variable , defaultValue ) ;
2011-03-12 19:23:57 +01:00
}
2011-03-26 09:39:23 +01:00
private int getConfigInt ( String variable , int defaultValue ) {
2012-07-29 09:33:13 +02:00
return this . console . getPropertyManager ( ) . getInt ( variable , defaultValue ) ;
2011-03-12 19:23:57 +01:00
}
2011-03-26 09:39:23 +01:00
2011-08-08 02:25:46 +02:00
private boolean getConfigBoolean ( String variable , boolean defaultValue ) {
2012-07-29 09:33:13 +02:00
return this . console . getPropertyManager ( ) . getBoolean ( variable , defaultValue ) ;
2011-08-08 02:25:46 +02:00
}
2011-03-12 19:23:57 +01:00
// End Temporary calls
2014-06-24 23:11:25 +02:00
@Override
2011-05-05 22:57:22 +02:00
public String getUpdateFolder ( ) {
return this . configuration . getString ( " settings.update-folder " , " update " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-25 03:22:06 +02:00
public File getUpdateFolderFile ( ) {
return new File ( ( File ) console . options . valueOf ( " plugins " ) , this . configuration . getString ( " settings.update-folder " , " update " ) ) ;
}
2011-09-17 02:57:43 +02:00
public int getPingPacketLimit ( ) {
return this . configuration . getInt ( " settings.ping-packet-limit " , 100 ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-03-22 13:03:24 +01:00
public long getConnectionThrottle ( ) {
return this . configuration . getInt ( " settings.connection-throttle " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-02-10 16:23:52 +01:00
public int getTicksPerAnimalSpawns ( ) {
return this . configuration . getInt ( " ticks-per.animal-spawns " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-02-10 16:23:52 +01:00
public int getTicksPerMonsterSpawns ( ) {
return this . configuration . getInt ( " ticks-per.monster-spawns " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2010-12-24 18:24:21 +01:00
public PluginManager getPluginManager ( ) {
return pluginManager ;
}
2010-12-27 03:13:03 +01:00
2014-06-24 23:11:25 +02:00
@Override
2012-07-29 09:33:13 +02:00
public CraftScheduler getScheduler ( ) {
2011-02-03 00:53:04 +01:00
return scheduler ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-05-02 20:31:02 +02:00
public ServicesManager getServicesManager ( ) {
return servicesManager ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-02-07 02:59:06 +01:00
public List < World > getWorlds ( ) {
2011-02-08 13:03:36 +01:00
return new ArrayList < World > ( worlds . values ( ) ) ;
2010-12-27 03:13:03 +01:00
}
2010-12-30 05:30:12 +01:00
2012-12-20 05:03:52 +01:00
public DedicatedPlayerList getHandle ( ) {
return playerList ;
2010-12-30 05:30:12 +01:00
}
2011-01-16 16:35:37 +01:00
2012-07-29 09:33:13 +02:00
// NOTE: Should only be called from DedicatedServer.ah()
2012-01-19 09:32:05 +01:00
public boolean dispatchServerCommand ( CommandSender sender , ServerCommand serverCommand ) {
if ( sender instanceof Conversable ) {
Conversable conversable = ( Conversable ) sender ;
if ( conversable . isConversing ( ) ) {
conversable . acceptConversationInput ( serverCommand . command ) ;
return true ;
}
}
2012-09-07 05:11:17 +02:00
try {
2013-09-23 23:43:21 +02:00
this . playerCommandState = true ;
2012-09-07 05:11:17 +02:00
return dispatchCommand ( sender , serverCommand . command ) ;
} catch ( Exception ex ) {
getLogger ( ) . log ( Level . WARNING , " Unexpected exception while parsing console command \" " + serverCommand . command + '"' , ex ) ;
return false ;
2013-09-23 23:43:21 +02:00
} finally {
this . playerCommandState = false ;
2012-09-07 05:11:17 +02:00
}
2011-03-12 19:23:57 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-01-29 22:04:02 +01:00
public boolean dispatchCommand ( CommandSender sender , String commandLine ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( sender , " Sender cannot be null " ) ;
Validate . notNull ( commandLine , " CommandLine cannot be null " ) ;
2011-03-12 19:23:57 +01:00
if ( commandMap . dispatch ( sender , commandLine ) ) {
return true ;
}
2011-08-13 04:47:47 +02:00
2013-08-03 02:04:39 +02:00
if ( sender instanceof Player ) {
sender . sendMessage ( " Unknown command. Type \" /help \" for help. " ) ;
} else {
sender . sendMessage ( " Unknown command. Type \" help \" for help. " ) ;
}
2011-03-12 19:23:57 +01:00
2011-09-03 16:57:54 +02:00
return false ;
2011-01-16 16:35:37 +01:00
}
2011-01-20 04:53:43 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-01-20 04:53:43 +01:00
public void reload ( ) {
2011-10-10 22:10:48 +02:00
configuration = YamlConfiguration . loadConfiguration ( getConfigFile ( ) ) ;
2014-02-08 00:40:59 +01:00
commandsConfiguration = YamlConfiguration . loadConfiguration ( getCommandsConfigFile ( ) ) ;
2013-11-04 14:07:38 +01:00
PropertyManager config = new PropertyManager ( console . options ) ;
2011-01-20 04:53:43 +01:00
2012-07-29 09:33:13 +02:00
( ( DedicatedServer ) console ) . propertyManager = config ;
2011-01-29 22:04:02 +01:00
2012-07-29 09:33:13 +02:00
boolean animals = config . getBoolean ( " spawn-animals " , console . getSpawnAnimals ( ) ) ;
2014-11-25 22:32:16 +01:00
boolean monsters = config . getBoolean ( " spawn-monsters " , console . worlds . get ( 0 ) . getDifficulty ( ) ! = EnumDifficulty . PEACEFUL ) ;
EnumDifficulty difficulty = EnumDifficulty . getById ( config . getInt ( " difficulty " , console . worlds . get ( 0 ) . getDifficulty ( ) . ordinal ( ) ) ) ;
2011-01-20 04:53:43 +01:00
2012-12-20 05:03:52 +01:00
online . value = config . getBoolean ( " online-mode " , console . getOnlineMode ( ) ) ;
2012-07-29 09:33:13 +02:00
console . setSpawnAnimals ( config . getBoolean ( " spawn-animals " , console . getSpawnAnimals ( ) ) ) ;
2014-11-25 22:32:16 +01:00
console . setPVP ( config . getBoolean ( " pvp " , console . getPVP ( ) ) ) ;
2012-07-29 09:33:13 +02:00
console . setAllowFlight ( config . getBoolean ( " allow-flight " , console . getAllowFlight ( ) ) ) ;
console . setMotd ( config . getString ( " motd " , console . getMotd ( ) ) ) ;
2012-04-24 02:01:22 +02:00
monsterSpawn = configuration . getInt ( " spawn-limits.monsters " ) ;
animalSpawn = configuration . getInt ( " spawn-limits.animals " ) ;
waterAnimalSpawn = configuration . getInt ( " spawn-limits.water-animals " ) ;
2012-10-31 17:18:41 +01:00
ambientSpawn = configuration . getInt ( " spawn-limits.ambient " ) ;
2012-08-07 06:52:49 +02:00
warningState = WarningState . value ( configuration . getString ( " settings.deprecated-verbose " ) ) ;
2013-09-23 23:43:21 +02:00
printSaveWarning = false ;
2012-08-19 00:36:39 +02:00
console . autosavePeriod = configuration . getInt ( " ticks-per.autosave " ) ;
[Bleeding] Implement periodic chunk garbage collector
This adds two settings to bukkit.yml, allowing activation and control of
two chunk garbage collection triggering conditions:
chunk-gc/period-in-ticks controls a periodic GC, run once every N ticks
(default is 600); chunk-gc/load-threshold causes the GC to run once
after every N calls to loadChunk() on a given world (this call is an API
call used by plugins, and is distinct from the path taken for routine
player movement-based loading). In both cases, setting to zero will
disable the given GC scheduling strategy.
In either case, the act of doing the GC is simply one of scanning the
loaded chunks, seeing which are NOT being used by one or more players
(due to view-distance) and which are not already queued for unload, and
queueing them for a normal unload. Ultimately, the unload is then
processed the same as if the chunk were unloaded due to leaving the
view-distance range of all players, so the impact on plugins should be
no different (and strategies such as handling the ChunkUnloadEvent in
order to prevent unload will still work).
The initial interval for the periodic GC is randomized on a per-world
basis, in order to avoid all world being GCed at the same time -
minimizing potential lag spikes.
2012-12-10 16:38:26 +01:00
chunkGCPeriod = configuration . getInt ( " chunk-gc.period-in-ticks " ) ;
chunkGCLoadThresh = configuration . getInt ( " chunk-gc.load-threshold " ) ;
2013-11-04 14:07:38 +01:00
loadIcon ( ) ;
2011-01-20 04:53:43 +01:00
2014-04-17 10:31:49 +02:00
try {
playerList . getIPBans ( ) . load ( ) ;
} catch ( IOException ex ) {
logger . log ( Level . WARNING , " Failed to load banned-ips.json, " + ex . getMessage ( ) ) ;
}
try {
playerList . getProfileBans ( ) . load ( ) ;
} catch ( IOException ex ) {
logger . log ( Level . WARNING , " Failed to load banned-players.json, " + ex . getMessage ( ) ) ;
}
2013-01-23 13:24:40 +01:00
2011-02-05 19:15:04 +01:00
for ( WorldServer world : console . worlds ) {
2014-11-25 22:32:16 +01:00
world . worldData . setDifficulty ( difficulty ) ;
2011-04-20 19:05:14 +02:00
world . setSpawnFlags ( monsters , animals ) ;
2012-02-10 16:23:52 +01:00
if ( this . getTicksPerAnimalSpawns ( ) < 0 ) {
world . ticksPerAnimalSpawns = 400 ;
} else {
world . ticksPerAnimalSpawns = this . getTicksPerAnimalSpawns ( ) ;
}
if ( this . getTicksPerMonsterSpawns ( ) < 0 ) {
world . ticksPerMonsterSpawns = 1 ;
} else {
world . ticksPerMonsterSpawns = this . getTicksPerMonsterSpawns ( ) ;
}
2011-02-05 19:15:04 +01:00
}
2011-01-28 15:18:49 +01:00
pluginManager . clearPlugins ( ) ;
commandMap . clearCommands ( ) ;
2011-07-24 05:16:14 +02:00
resetRecipes ( ) ;
2014-02-08 00:40:59 +01:00
overrideAllCommandBlockCommands = commandsConfiguration . getStringList ( " command-block-overrides " ) . contains ( " * " ) ;
2011-04-28 21:10:39 +02:00
int pollCount = 0 ;
// Wait for at most 2.5 seconds for plugins to close their threads
2011-06-12 01:12:43 +02:00
while ( pollCount < 50 & & getScheduler ( ) . getActiveWorkers ( ) . size ( ) > 0 ) {
2011-04-28 21:10:39 +02:00
try {
Thread . sleep ( 50 ) ;
2011-06-12 01:12:43 +02:00
} catch ( InterruptedException e ) { }
2011-04-28 21:10:39 +02:00
pollCount + + ;
}
List < BukkitWorker > overdueWorkers = getScheduler ( ) . getActiveWorkers ( ) ;
2011-06-12 01:12:43 +02:00
for ( BukkitWorker worker : overdueWorkers ) {
2011-04-28 21:10:39 +02:00
Plugin plugin = worker . getOwner ( ) ;
String author = " <NoAuthorGiven> " ;
if ( plugin . getDescription ( ) . getAuthors ( ) . size ( ) > 0 ) {
author = plugin . getDescription ( ) . getAuthors ( ) . get ( 0 ) ;
}
getLogger ( ) . log ( Level . SEVERE , String . format (
" Nag author: '%s' of '%s' about the following: %s " ,
author ,
plugin . getDescription ( ) . getName ( ) ,
" This plugin is not properly shutting down its async tasks when it is being reloaded. This may cause conflicts with the newly loaded version of the plugin "
) ) ;
}
2011-01-28 15:18:49 +01:00
loadPlugins ( ) ;
2011-06-22 18:46:12 +02:00
enablePlugins ( PluginLoadOrder . STARTUP ) ;
enablePlugins ( PluginLoadOrder . POSTWORLD ) ;
2011-01-20 04:53:43 +01:00
}
2011-02-02 19:23:19 +01:00
2013-11-04 14:07:38 +01:00
private void loadIcon ( ) {
icon = new CraftIconCache ( null ) ;
try {
final File file = new File ( new File ( " . " ) , " server-icon.png " ) ;
if ( file . isFile ( ) ) {
icon = loadServerIcon0 ( file ) ;
}
} catch ( Exception ex ) {
getLogger ( ) . log ( Level . WARNING , " Couldn't load server icon " , ex ) ;
}
}
2012-01-30 21:51:53 +01:00
@SuppressWarnings ( { " unchecked " , " finally " } )
2011-07-17 18:19:41 +02:00
private void loadCustomPermissions ( ) {
File file = new File ( configuration . getString ( " settings.permissions-file " ) ) ;
FileInputStream stream ;
try {
stream = new FileInputStream ( file ) ;
} catch ( FileNotFoundException ex ) {
try {
file . createNewFile ( ) ;
} finally {
return ;
}
}
Map < String , Map < String , Object > > perms ;
try {
2012-01-14 23:02:10 +01:00
perms = ( Map < String , Map < String , Object > > ) yaml . load ( stream ) ;
2011-07-17 18:19:41 +02:00
} catch ( MarkedYAMLException ex ) {
getLogger ( ) . log ( Level . WARNING , " Server permissions file " + file + " is not valid YAML: " + ex . toString ( ) ) ;
return ;
} catch ( Throwable ex ) {
getLogger ( ) . log ( Level . WARNING , " Server permissions file " + file + " is not valid YAML. " , ex ) ;
return ;
2011-08-13 03:13:35 +02:00
} finally {
try {
stream . close ( ) ;
} catch ( IOException ex ) { }
2011-07-17 18:19:41 +02:00
}
if ( perms = = null ) {
getLogger ( ) . log ( Level . INFO , " Server permissions file " + file + " is empty, ignoring it " ) ;
return ;
}
2012-04-15 20:24:38 +02:00
List < Permission > permsList = Permission . loadPermissions ( perms , " Permission node '%s' in " + file + " is invalid " , Permission . DEFAULT_PERMISSION ) ;
2011-07-17 18:19:41 +02:00
2012-04-15 20:24:38 +02:00
for ( Permission perm : permsList ) {
2011-07-17 18:19:41 +02:00
try {
2012-04-15 20:24:38 +02:00
pluginManager . addPermission ( perm ) ;
} catch ( IllegalArgumentException ex ) {
getLogger ( ) . log ( Level . SEVERE , " Permission in " + file + " was already defined " , ex ) ;
2011-07-17 18:19:41 +02:00
}
}
}
2011-02-02 19:23:19 +01:00
@Override
public String toString ( ) {
2011-11-20 09:01:14 +01:00
return " CraftServer{ " + " serverName= " + serverName + " ,serverVersion= " + serverVersion + " ,minecraftVersion= " + console . getVersion ( ) + '}' ;
2011-02-02 19:23:19 +01:00
}
2011-02-05 21:51:22 +01:00
2011-02-06 21:50:57 +01:00
public World createWorld ( String name , World . Environment environment ) {
2011-09-25 02:05:10 +02:00
return WorldCreator . name ( name ) . environment ( environment ) . createWorld ( ) ;
2011-03-31 22:40:00 +02:00
}
public World createWorld ( String name , World . Environment environment , long seed ) {
2011-09-25 02:05:10 +02:00
return WorldCreator . name ( name ) . environment ( environment ) . seed ( seed ) . createWorld ( ) ;
2011-06-06 15:52:02 +02:00
}
public World createWorld ( String name , Environment environment , ChunkGenerator generator ) {
2011-09-25 02:05:10 +02:00
return WorldCreator . name ( name ) . environment ( environment ) . generator ( generator ) . createWorld ( ) ;
2011-06-06 15:52:02 +02:00
}
public World createWorld ( String name , Environment environment , long seed , ChunkGenerator generator ) {
2011-09-25 02:05:10 +02:00
return WorldCreator . name ( name ) . environment ( environment ) . seed ( seed ) . generator ( generator ) . createWorld ( ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-25 02:05:10 +02:00
public World createWorld ( WorldCreator creator ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( creator , " Creator may not be null " ) ;
2011-09-25 02:05:10 +02:00
String name = creator . name ( ) ;
ChunkGenerator generator = creator . generator ( ) ;
2012-01-14 18:50:31 +01:00
File folder = new File ( getWorldContainer ( ) , name ) ;
2011-02-08 13:03:36 +01:00
World world = getWorld ( name ) ;
2012-02-29 22:31:04 +01:00
WorldType type = WorldType . getType ( creator . type ( ) . getName ( ) ) ;
2012-01-29 18:40:45 +01:00
boolean generateStructures = creator . generateStructures ( ) ;
2011-02-08 13:03:36 +01:00
if ( world ! = null ) {
return world ;
}
2011-03-12 19:23:57 +01:00
2011-02-05 21:51:22 +01:00
if ( ( folder . exists ( ) ) & & ( ! folder . isDirectory ( ) ) ) {
throw new IllegalArgumentException ( " File exists with the name ' " + name + " ' and isn't a folder " ) ;
}
2011-07-08 20:01:03 +02:00
if ( generator = = null ) {
generator = getGenerator ( name ) ;
}
2012-03-04 02:36:16 +01:00
Convertable converter = new WorldLoaderServer ( getWorldContainer ( ) ) ;
2011-04-20 19:05:14 +02:00
if ( converter . isConvertable ( name ) ) {
2011-02-24 13:07:33 +01:00
getLogger ( ) . info ( " Converting world ' " + name + " ' " ) ;
2011-04-20 19:05:14 +02:00
converter . convert ( name , new ConvertProgressUpdater ( console ) ) ;
2011-02-24 13:07:33 +01:00
}
2013-01-23 02:36:03 +01:00
int dimension = CraftWorld . CUSTOM_DIMENSION_OFFSET + console . worlds . size ( ) ;
2012-01-19 17:09:24 +01:00
boolean used = false ;
do {
for ( WorldServer server : console . worlds ) {
used = server . dimension = = dimension ;
2012-01-09 08:51:32 +01:00
if ( used ) {
2012-01-19 17:09:24 +01:00
dimension + + ;
break ;
}
}
} while ( used ) ;
2011-11-20 09:01:14 +01:00
boolean hardcore = false ;
2012-01-29 18:40:45 +01:00
2014-11-25 22:32:16 +01:00
WorldData worlddata = new WorldData ( new WorldSettings ( creator . seed ( ) , EnumGamemode . getById ( getDefaultGameMode ( ) . getValue ( ) ) , generateStructures , hardcore , type ) , name ) ;
WorldServer internal = ( WorldServer ) new WorldServer ( console , new ServerNBTManager ( getWorldContainer ( ) , name , true ) , worlddata , dimension , console . methodProfiler , creator . environment ( ) , generator ) . b ( ) ;
2011-07-13 09:29:43 +02:00
2011-07-13 21:35:05 +02:00
if ( ! ( worlds . containsKey ( name . toLowerCase ( ) ) ) ) {
2011-07-13 09:29:43 +02:00
return null ;
}
2013-04-05 06:35:16 +02:00
internal . scoreboard = getScoreboardManager ( ) . getMainScoreboard ( ) . getHandle ( ) ;
2011-02-05 21:51:22 +01:00
2013-04-05 06:35:16 +02:00
internal . tracker = new EntityTracker ( internal ) ;
2013-03-25 05:22:32 +01:00
internal . addIWorldAccess ( new WorldManager ( console , internal ) ) ;
2014-11-25 22:32:16 +01:00
internal . worldData . setDifficulty ( EnumDifficulty . EASY ) ;
2011-04-20 19:05:14 +02:00
internal . setSpawnFlags ( true , true ) ;
2011-02-05 21:51:22 +01:00
console . worlds . add ( internal ) ;
2011-06-06 15:52:02 +02:00
if ( generator ! = null ) {
internal . getWorld ( ) . getPopulators ( ) . addAll ( generator . getDefaultPopulators ( internal . getWorld ( ) ) ) ;
}
pluginManager . callEvent ( new WorldInitEvent ( internal . getWorld ( ) ) ) ;
2012-01-14 23:02:10 +01:00
System . out . print ( " Preparing start region for level " + ( console . worlds . size ( ) - 1 ) + " (Seed: " + internal . getSeed ( ) + " ) " ) ;
2011-06-06 15:52:02 +02:00
2011-07-27 01:24:27 +02:00
if ( internal . getWorld ( ) . getKeepSpawnInMemory ( ) ) {
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 ;
}
2014-11-25 22:32:16 +01:00
BlockPosition chunkcoordinates = internal . getSpawn ( ) ;
internal . chunkProviderServer . getChunkAt ( chunkcoordinates . getX ( ) + j > > 4 , chunkcoordinates . getZ ( ) + k > > 4 ) ;
2011-02-05 21:51:22 +01:00
}
}
}
2011-03-27 05:50:52 +02:00
pluginManager . callEvent ( new WorldLoadEvent ( internal . getWorld ( ) ) ) ;
2011-03-24 22:43:21 +01:00
return internal . getWorld ( ) ;
2011-02-05 21:51:22 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-06-07 08:54:14 +02:00
public boolean unloadWorld ( String name , boolean save ) {
return unloadWorld ( getWorld ( name ) , save ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-06-07 08:54:14 +02:00
public boolean unloadWorld ( World world , boolean save ) {
if ( world = = null ) {
return false ;
}
WorldServer handle = ( ( CraftWorld ) world ) . getHandle ( ) ;
if ( ! ( console . worlds . contains ( handle ) ) ) {
return false ;
}
if ( ! ( handle . dimension > 1 ) ) {
return false ;
}
if ( handle . players . size ( ) > 0 ) {
return false ;
}
WorldUnloadEvent e = new WorldUnloadEvent ( handle . getWorld ( ) ) ;
2011-11-24 21:27:16 +01:00
pluginManager . callEvent ( e ) ;
2011-06-07 08:54:14 +02:00
if ( e . isCancelled ( ) ) {
return false ;
}
if ( save ) {
2012-07-29 09:33:13 +02:00
try {
2013-03-25 05:22:32 +01:00
handle . save ( true , null ) ;
2012-07-29 09:33:13 +02:00
handle . saveLevel ( ) ;
WorldSaveEvent event = new WorldSaveEvent ( handle . getWorld ( ) ) ;
getPluginManager ( ) . callEvent ( event ) ;
} catch ( ExceptionWorldConflict ex ) {
getLogger ( ) . log ( Level . SEVERE , null , ex ) ;
}
2011-06-07 08:54:14 +02:00
}
worlds . remove ( world . getName ( ) . toLowerCase ( ) ) ;
console . worlds . remove ( console . worlds . indexOf ( handle ) ) ;
return true ;
}
2011-02-05 21:51:22 +01:00
public MinecraftServer getServer ( ) {
return console ;
}
2011-02-08 13:03:36 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-02-08 13:03:36 +01:00
public World getWorld ( String name ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( name , " Name cannot be null " ) ;
2011-02-08 13:03:36 +01:00
return worlds . get ( name . toLowerCase ( ) ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2014-04-17 08:53:13 +02:00
public World getWorld ( UUID uid ) {
2011-07-07 20:47:42 +02:00
for ( World world : worlds . values ( ) ) {
2011-07-13 06:06:07 +02:00
if ( world . getUID ( ) . equals ( uid ) ) {
2011-07-05 05:48:27 +02:00
return world ;
}
}
return null ;
}
2011-06-17 04:06:45 +02:00
public void addWorld ( World world ) {
2011-07-13 21:35:05 +02:00
// Check if a World already exists with the UID.
if ( getWorld ( world . getUID ( ) ) ! = null ) {
System . out . println ( " World " + world . getName ( ) + " is a duplicate of another world and has been prevented from loading. Please delete the uid.dat file from " + world . getName ( ) + " 's world directory if you want to be able to load the duplicate world. " ) ;
return ;
}
2011-02-08 13:03:36 +01:00
worlds . put ( world . getName ( ) . toLowerCase ( ) , world ) ;
}
2011-02-20 02:53:06 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-02-20 02:53:06 +01:00
public Logger getLogger ( ) {
2013-11-04 14:07:38 +01:00
return logger ;
2011-02-20 02:53:06 +01:00
}
2011-02-25 17:12:38 +01:00
public ConsoleReader getReader ( ) {
return console . reader ;
}
2011-02-28 01:31:25 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-02-28 01:31:25 +01:00
public PluginCommand getPluginCommand ( String name ) {
Command command = commandMap . getCommand ( name ) ;
if ( command instanceof PluginCommand ) {
2011-06-12 01:12:43 +02:00
return ( PluginCommand ) command ;
2011-02-28 01:31:25 +01:00
} else {
return null ;
}
}
2011-02-26 03:29:42 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-02-26 03:29:42 +01:00
public void savePlayers ( ) {
2013-09-23 23:43:21 +02:00
checkSaveState ( ) ;
2012-12-20 05:03:52 +01:00
playerList . savePlayers ( ) ;
2011-02-26 03:29:42 +01:00
}
2011-03-12 19:23:57 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-03-31 03:35:08 +02:00
public void configureDbConfig ( ServerConfig config ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( config , " Config cannot be null " ) ;
2011-03-31 03:35:08 +02:00
DataSourceConfig ds = new DataSourceConfig ( ) ;
ds . setDriver ( configuration . getString ( " database.driver " ) ) ;
ds . setUrl ( configuration . getString ( " database.url " ) ) ;
ds . setUsername ( configuration . getString ( " database.username " ) ) ;
ds . setPassword ( configuration . getString ( " database.password " ) ) ;
ds . setIsolationLevel ( TransactionIsolation . getLevel ( configuration . getString ( " database.isolation " ) ) ) ;
if ( ds . getDriver ( ) . contains ( " sqlite " ) ) {
config . setDatabasePlatform ( new SQLitePlatform ( ) ) ;
config . getDatabasePlatform ( ) . getDbDdlSyntax ( ) . setIdentity ( " " ) ;
}
config . setDataSourceConfig ( ds ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-04-16 04:11:13 +02:00
public boolean addRecipe ( Recipe recipe ) {
CraftRecipe toAdd ;
2011-06-12 01:12:43 +02:00
if ( recipe instanceof CraftRecipe ) {
2011-04-16 04:11:13 +02:00
toAdd = ( CraftRecipe ) recipe ;
} else {
if ( recipe instanceof ShapedRecipe ) {
toAdd = CraftShapedRecipe . fromBukkitRecipe ( ( ShapedRecipe ) recipe ) ;
} else if ( recipe instanceof ShapelessRecipe ) {
toAdd = CraftShapelessRecipe . fromBukkitRecipe ( ( ShapelessRecipe ) recipe ) ;
} else if ( recipe instanceof FurnaceRecipe ) {
toAdd = CraftFurnaceRecipe . fromBukkitRecipe ( ( FurnaceRecipe ) recipe ) ;
} else {
return false ;
}
}
toAdd . addToCraftingManager ( ) ;
2011-07-24 05:16:14 +02:00
CraftingManager . getInstance ( ) . sort ( ) ;
2011-04-16 04:11:13 +02:00
return true ;
}
2011-06-17 05:09:43 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-07-24 05:16:14 +02:00
public List < Recipe > getRecipesFor ( ItemStack result ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( result , " Result cannot be null " ) ;
2011-07-24 05:16:14 +02:00
List < Recipe > results = new ArrayList < Recipe > ( ) ;
Iterator < Recipe > iter = recipeIterator ( ) ;
while ( iter . hasNext ( ) ) {
Recipe recipe = iter . next ( ) ;
ItemStack stack = recipe . getResult ( ) ;
if ( stack . getType ( ) ! = result . getType ( ) ) {
continue ;
}
if ( result . getDurability ( ) = = - 1 | | result . getDurability ( ) = = stack . getDurability ( ) ) {
results . add ( recipe ) ;
}
}
return results ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-07-24 05:16:14 +02:00
public Iterator < Recipe > recipeIterator ( ) {
return new RecipeIterator ( ) ;
}
2012-02-29 17:27:38 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-07-24 05:16:14 +02:00
public void clearRecipes ( ) {
2012-07-29 09:33:13 +02:00
CraftingManager . getInstance ( ) . recipes . clear ( ) ;
RecipesFurnace . getInstance ( ) . recipes . clear ( ) ;
2013-12-21 00:44:13 +01:00
RecipesFurnace . getInstance ( ) . customRecipes . clear ( ) ;
2011-07-24 05:16:14 +02:00
}
2012-02-29 17:27:38 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-07-24 05:16:14 +02:00
public void resetRecipes ( ) {
2012-07-29 09:33:13 +02:00
CraftingManager . getInstance ( ) . recipes = new CraftingManager ( ) . recipes ;
RecipesFurnace . getInstance ( ) . recipes = new RecipesFurnace ( ) . recipes ;
2013-12-21 00:44:13 +01:00
RecipesFurnace . getInstance ( ) . customRecipes . clear ( ) ;
2011-07-24 05:16:14 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-06-22 20:10:29 +02:00
public Map < String , String [ ] > getCommandAliases ( ) {
2014-02-08 00:40:59 +01:00
ConfigurationSection section = commandsConfiguration . getConfigurationSection ( " aliases " ) ;
2011-07-13 05:14:28 +02:00
Map < String , String [ ] > result = new LinkedHashMap < String , String [ ] > ( ) ;
2011-06-17 05:09:43 +02:00
2011-10-10 22:10:48 +02:00
if ( section ! = null ) {
for ( String key : section . getKeys ( false ) ) {
2013-03-25 05:22:32 +01:00
List < String > commands ;
2011-06-22 20:10:29 +02:00
2011-10-10 22:10:48 +02:00
if ( section . isList ( key ) ) {
2012-02-29 17:27:38 +01:00
commands = section . getStringList ( key ) ;
2011-06-22 20:10:29 +02:00
} else {
2013-03-25 05:22:32 +01:00
commands = ImmutableList . of ( section . getString ( key ) ) ;
2011-06-22 20:10:29 +02:00
}
2012-09-10 06:19:28 +02:00
result . put ( key , commands . toArray ( new String [ commands . size ( ) ] ) ) ;
2011-06-17 05:09:43 +02:00
}
}
return result ;
}
2011-06-17 05:49:58 +02:00
2012-10-28 06:30:09 +01:00
public void removeBukkitSpawnRadius ( ) {
configuration . set ( " settings.spawn-radius " , null ) ;
saveConfig ( ) ;
}
public int getBukkitSpawnRadius ( ) {
return configuration . getInt ( " settings.spawn-radius " , - 1 ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-12-04 02:32:44 +01:00
public String getShutdownMessage ( ) {
return configuration . getString ( " settings.shutdown-message " ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-06-17 05:49:58 +02:00
public int getSpawnRadius ( ) {
2012-10-28 06:30:09 +01:00
return ( ( DedicatedServer ) console ) . propertyManager . getInt ( " spawn-protection " , 16 ) ;
2011-06-17 05:49:58 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-06-17 05:49:58 +02:00
public void setSpawnRadius ( int value ) {
2011-10-10 22:10:48 +02:00
configuration . set ( " settings.spawn-radius " , value ) ;
saveConfig ( ) ;
2011-06-17 05:49:58 +02:00
}
2011-06-19 12:13:52 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-06-19 12:13:52 +02:00
public boolean getOnlineMode ( ) {
2012-12-20 05:03:52 +01:00
return online . value ;
2011-06-19 12:13:52 +02:00
}
2011-06-22 19:27:27 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-08-08 02:25:46 +02:00
public boolean getAllowFlight ( ) {
2012-07-29 09:33:13 +02:00
return console . getAllowFlight ( ) ;
2011-08-08 02:25:46 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2012-10-29 06:35:31 +01:00
public boolean isHardcore ( ) {
return console . isHardcore ( ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-01-29 12:36:43 +01:00
public boolean useExactLoginLocation ( ) {
return configuration . getBoolean ( " settings.use-exact-login-location " ) ;
}
2011-06-22 19:27:27 +02:00
public ChunkGenerator getGenerator ( String world ) {
2011-10-10 22:10:48 +02:00
ConfigurationSection section = configuration . getConfigurationSection ( " worlds " ) ;
2011-06-22 19:27:27 +02:00
ChunkGenerator result = null ;
2011-10-10 22:10:48 +02:00
if ( section ! = null ) {
section = section . getConfigurationSection ( world ) ;
2011-06-22 19:27:27 +02:00
2011-10-10 22:10:48 +02:00
if ( section ! = null ) {
String name = section . getString ( " generator " ) ;
2011-06-22 19:27:27 +02:00
2011-06-23 21:41:07 +02:00
if ( ( name ! = null ) & & ( ! name . equals ( " " ) ) ) {
2011-06-22 19:27:27 +02:00
String [ ] split = name . split ( " : " , 2 ) ;
String id = ( split . length > 1 ) ? split [ 1 ] : null ;
Plugin plugin = pluginManager . getPlugin ( split [ 0 ] ) ;
if ( plugin = = null ) {
getLogger ( ) . severe ( " Could not set generator for default world ' " + world + " ': Plugin ' " + split [ 0 ] + " ' does not exist " ) ;
} else if ( ! plugin . isEnabled ( ) ) {
2013-03-21 18:50:58 +01:00
getLogger ( ) . severe ( " Could not set generator for default world ' " + world + " ': Plugin ' " + plugin . getDescription ( ) . getFullName ( ) + " ' is not enabled yet (is it load:STARTUP?) " ) ;
2011-06-22 19:27:27 +02:00
} else {
2013-04-19 02:50:05 +02:00
try {
result = plugin . getDefaultWorldGenerator ( world , id ) ;
if ( result = = null ) {
getLogger ( ) . severe ( " Could not set generator for default world ' " + world + " ': Plugin ' " + plugin . getDescription ( ) . getFullName ( ) + " ' lacks a default world generator " ) ;
}
} catch ( Throwable t ) {
plugin . getLogger ( ) . log ( Level . SEVERE , " Could not set generator for default world ' " + world + " ': Plugin ' " + plugin . getDescription ( ) . getFullName ( ) , t ) ;
2013-03-21 18:50:58 +01:00
}
2011-06-22 19:27:27 +02:00
}
}
}
}
return result ;
}
2011-08-13 04:47:47 +02:00
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-08-13 04:47:47 +02:00
public CraftMapView getMap ( short id ) {
2013-11-04 14:07:38 +01:00
PersistentCollection collection = console . worlds . get ( 0 ) . worldMaps ;
2012-01-12 16:27:39 +01:00
WorldMap worldmap = ( WorldMap ) collection . get ( WorldMap . class , " map_ " + id ) ;
2011-08-13 04:47:47 +02:00
if ( worldmap = = null ) {
return null ;
}
return worldmap . mapView ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-08-13 04:47:47 +02:00
public CraftMapView createMap ( World world ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( world , " World cannot be null " ) ;
2013-11-04 14:07:38 +01:00
net . minecraft . server . ItemStack stack = new net . minecraft . server . ItemStack ( Items . MAP , 1 , - 1 ) ;
2014-11-25 22:32:16 +01:00
WorldMap worldmap = Items . FILLED_MAP . getSavedMap ( stack , ( ( CraftWorld ) world ) . getHandle ( ) ) ;
2011-08-13 04:47:47 +02:00
return worldmap . mapView ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-02 21:18:35 +02:00
public void shutdown ( ) {
2011-09-24 23:03:31 +02:00
console . safeShutdown ( ) ;
2011-09-02 21:18:35 +02:00
}
2011-09-02 23:24:28 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-09-02 23:24:28 +02:00
public int broadcast ( String message , String permission ) {
int count = 0 ;
Set < Permissible > permissibles = getPluginManager ( ) . getPermissionSubscriptions ( permission ) ;
for ( Permissible permissible : permissibles ) {
2012-01-15 11:32:00 +01:00
if ( permissible instanceof CommandSender & & permissible . hasPermission ( permission ) ) {
2012-01-14 23:02:10 +01:00
CommandSender user = ( CommandSender ) permissible ;
2011-09-02 23:24:28 +02:00
user . sendMessage ( message ) ;
count + + ;
}
}
return count ;
}
2011-09-03 01:41:48 +02:00
2014-06-24 23:11:25 +02:00
@Override
@Deprecated
2011-09-03 01:41:48 +02:00
public OfflinePlayer getOfflinePlayer ( String name ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( name , " Name cannot be null " ) ;
2014-04-17 08:53:13 +02:00
// If the name given cannot ever be a valid username give a dummy return, for scoreboard plugins
if ( ! validUserPattern . matcher ( name ) . matches ( ) ) {
return new CraftOfflinePlayer ( this , new GameProfile ( invalidUserUUID , name ) ) ;
}
2014-04-12 09:38:53 +02:00
OfflinePlayer result = getPlayerExact ( name ) ;
if ( result = = null ) {
// This is potentially blocking :(
2014-06-26 20:05:08 +02:00
GameProfile profile = MinecraftServer . getServer ( ) . getUserCache ( ) . getProfile ( name ) ;
2014-04-12 09:38:53 +02:00
if ( profile = = null ) {
// Make an OfflinePlayer using an offline mode UUID since the name has no profile
2014-04-17 08:53:13 +02:00
result = getOfflinePlayer ( new GameProfile ( UUID . nameUUIDFromBytes ( ( " OfflinePlayer: " + name ) . getBytes ( Charsets . UTF_8 ) ) , name ) ) ;
2014-04-12 09:38:53 +02:00
} else {
// Use the GameProfile even when we get a UUID so we ensure we still have a name
result = getOfflinePlayer ( profile ) ;
}
} else {
offlinePlayers . remove ( result . getUniqueId ( ) ) ;
2014-04-11 03:04:38 +02:00
}
2011-09-03 01:41:48 +02:00
2014-04-12 09:38:53 +02:00
return result ;
2014-04-11 03:04:38 +02:00
}
2011-09-25 05:21:35 +02:00
2014-06-24 23:11:25 +02:00
@Override
2014-04-17 08:53:13 +02:00
public OfflinePlayer getOfflinePlayer ( UUID id ) {
2014-04-11 03:04:38 +02:00
Validate . notNull ( id , " UUID cannot be null " ) ;
2012-08-09 18:27:40 +02:00
2014-04-11 03:04:38 +02:00
OfflinePlayer result = getPlayer ( id ) ;
if ( result = = null ) {
result = offlinePlayers . get ( id ) ;
if ( result = = null ) {
result = new CraftOfflinePlayer ( this , new GameProfile ( id , null ) ) ;
offlinePlayers . put ( id , result ) ;
2011-09-25 05:21:35 +02:00
}
} else {
2014-04-11 03:04:38 +02:00
offlinePlayers . remove ( id ) ;
2011-09-03 01:41:48 +02:00
}
return result ;
}
2011-09-03 02:59:28 +02:00
2014-04-11 03:04:38 +02:00
public OfflinePlayer getOfflinePlayer ( GameProfile profile ) {
OfflinePlayer player = new CraftOfflinePlayer ( this , profile ) ;
offlinePlayers . put ( profile . getId ( ) , player ) ;
return player ;
2014-03-29 05:48:15 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
2012-01-30 21:51:53 +01:00
@SuppressWarnings ( " unchecked " )
2011-09-03 02:59:28 +02:00
public Set < String > getIPBans ( ) {
2014-04-11 03:04:38 +02:00
return new HashSet < String > ( Arrays . asList ( playerList . getIPBans ( ) . getEntries ( ) ) ) ;
2011-09-03 02:59:28 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 02:59:28 +02:00
public void banIP ( String address ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( address , " Address cannot be null. " ) ;
2014-02-05 04:52:50 +01:00
this . getBanList ( org . bukkit . BanList . Type . IP ) . addBan ( address , null , null , null ) ;
2011-09-03 02:59:28 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 02:59:28 +02:00
public void unbanIP ( String address ) {
2014-02-05 04:52:50 +01:00
Validate . notNull ( address , " Address cannot be null. " ) ;
this . getBanList ( org . bukkit . BanList . Type . IP ) . pardon ( address ) ;
2011-09-03 02:59:28 +02:00
}
2011-09-03 03:13:07 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 03:13:07 +02:00
public Set < OfflinePlayer > getBannedPlayers ( ) {
Set < OfflinePlayer > result = new HashSet < OfflinePlayer > ( ) ;
2014-04-17 10:31:49 +02:00
for ( JsonListEntry entry : playerList . getProfileBans ( ) . getValues ( ) ) {
2014-06-26 20:05:08 +02:00
result . add ( getOfflinePlayer ( ( GameProfile ) entry . getKey ( ) ) ) ;
2014-11-25 22:32:16 +01:00
}
2011-09-03 03:13:07 +02:00
return result ;
}
2011-09-03 16:57:54 +02:00
2014-02-05 04:52:50 +01:00
@Override
2014-04-11 03:04:38 +02:00
public BanList getBanList ( BanList . Type type ) {
2014-02-05 04:52:50 +01:00
Validate . notNull ( type , " Type cannot be null " ) ;
switch ( type ) {
case IP :
2014-04-11 03:04:38 +02:00
return new CraftIpBanList ( playerList . getIPBans ( ) ) ;
2014-02-05 04:52:50 +01:00
case NAME :
2014-04-11 03:04:38 +02:00
default :
return new CraftProfileBanList ( playerList . getProfileBans ( ) ) ;
2014-02-05 04:52:50 +01:00
}
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 16:57:54 +02:00
public void setWhitelist ( boolean value ) {
2014-04-11 03:04:38 +02:00
playerList . setHasWhitelist ( value ) ;
2014-06-26 20:05:08 +02:00
console . getPropertyManager ( ) . setProperty ( " white-list " , value ) ;
2011-09-03 16:57:54 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 16:57:54 +02:00
public Set < OfflinePlayer > getWhitelistedPlayers ( ) {
2012-06-15 05:43:46 +02:00
Set < OfflinePlayer > result = new LinkedHashSet < OfflinePlayer > ( ) ;
2011-09-03 16:57:54 +02:00
2014-04-17 10:31:49 +02:00
for ( JsonListEntry entry : playerList . getWhitelist ( ) . getValues ( ) ) {
2014-06-26 20:05:08 +02:00
result . add ( getOfflinePlayer ( ( GameProfile ) entry . getKey ( ) ) ) ;
2011-09-03 16:57:54 +02:00
}
return result ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-25 04:28:00 +02:00
public Set < OfflinePlayer > getOperators ( ) {
Set < OfflinePlayer > result = new HashSet < OfflinePlayer > ( ) ;
2014-04-17 10:31:49 +02:00
for ( JsonListEntry entry : playerList . getOPs ( ) . getValues ( ) ) {
2014-06-26 20:05:08 +02:00
result . add ( getOfflinePlayer ( ( GameProfile ) entry . getKey ( ) ) ) ;
2011-09-25 04:28:00 +02:00
}
return result ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-03 16:57:54 +02:00
public void reloadWhitelist ( ) {
2012-12-20 05:03:52 +01:00
playerList . reloadWhitelist ( ) ;
2011-09-03 16:57:54 +02:00
}
2011-09-09 02:17:53 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-09-09 02:17:53 +02:00
public GameMode getDefaultGameMode ( ) {
2014-06-26 20:05:08 +02:00
return GameMode . getByValue ( console . worlds . get ( 0 ) . getWorldData ( ) . getGameType ( ) . getId ( ) ) ;
2011-09-09 02:17:53 +02:00
}
2014-06-24 23:11:25 +02:00
@Override
2011-09-09 02:17:53 +02:00
public void setDefaultGameMode ( GameMode mode ) {
2013-03-19 06:32:10 +01:00
Validate . notNull ( mode , " Mode cannot be null " ) ;
2011-09-15 03:22:20 +02:00
for ( World world : getWorlds ( ) ) {
2014-06-26 20:05:08 +02:00
( ( CraftWorld ) world ) . getHandle ( ) . worldData . setGameType ( EnumGamemode . getById ( mode . getValue ( ) ) ) ;
2011-09-15 03:22:20 +02:00
}
2011-09-09 02:17:53 +02:00
}
2011-09-25 00:14:13 +02:00
2014-06-24 23:11:25 +02:00
@Override
2011-09-25 00:14:13 +02:00
public ConsoleCommandSender getConsoleSender ( ) {
return console . console ;
}
2011-10-03 01:34:51 +02:00
2011-12-08 06:33:59 +01:00
public EntityMetadataStore getEntityMetadata ( ) {
return entityMetadata ;
}
public PlayerMetadataStore getPlayerMetadata ( ) {
return playerMetadata ;
}
public WorldMetadataStore getWorldMetadata ( ) {
return worldMetadata ;
}
2014-06-24 23:11:25 +02:00
@Override
2011-10-12 05:24:41 +02:00
public File getWorldContainer ( ) {
2012-07-29 09:33:13 +02:00
if ( this . getServer ( ) . universe ! = null ) {
return this . getServer ( ) . universe ;
}
if ( container = = null ) {
container = new File ( configuration . getString ( " settings.world-container " , " . " ) ) ;
}
return container ;
2011-10-12 05:24:41 +02:00
}
2011-12-04 18:57:00 +01:00
2014-06-24 23:11:25 +02:00
@Override
2011-12-04 18:57:00 +01:00
public OfflinePlayer [ ] getOfflinePlayers ( ) {
2011-12-27 06:48:09 +01:00
WorldNBTStorage storage = ( WorldNBTStorage ) console . worlds . get ( 0 ) . getDataManager ( ) ;
2011-12-04 18:57:00 +01:00
String [ ] files = storage . getPlayerDir ( ) . list ( new DatFileFilter ( ) ) ;
2011-12-27 06:48:09 +01:00
Set < OfflinePlayer > players = new HashSet < OfflinePlayer > ( ) ;
2011-12-04 18:57:00 +01:00
2012-09-10 06:19:28 +02:00
for ( String file : files ) {
2014-04-11 03:04:38 +02:00
try {
2014-04-17 08:53:13 +02:00
players . add ( getOfflinePlayer ( UUID . fromString ( file . substring ( 0 , file . length ( ) - 4 ) ) ) ) ;
2014-04-11 03:04:38 +02:00
} catch ( IllegalArgumentException ex ) {
// Who knows what is in this directory, just ignore invalid files
}
2011-12-04 18:57:00 +01:00
}
2014-04-11 03:04:38 +02:00
2014-06-24 23:11:25 +02:00
players . addAll ( getOnlinePlayers ( ) ) ;
2011-12-04 18:57:00 +01:00
2012-01-02 00:25:43 +01:00
return players . toArray ( new OfflinePlayer [ players . size ( ) ] ) ;
2011-12-04 18:57:00 +01:00
}
2012-01-13 09:52:26 +01:00
2014-06-24 23:11:25 +02:00
@Override
2012-01-13 09:52:26 +01:00
public Messenger getMessenger ( ) {
return messenger ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-01-13 09:52:26 +01:00
public void sendPluginMessage ( Plugin source , String channel , byte [ ] message ) {
StandardMessenger . validatePluginMessage ( getMessenger ( ) , source , channel , message ) ;
for ( Player player : getOnlinePlayers ( ) ) {
player . sendPluginMessage ( source , channel , message ) ;
}
}
2014-06-24 23:11:25 +02:00
@Override
2012-01-13 09:52:26 +01:00
public Set < String > getListeningPluginChannels ( ) {
Set < String > result = new HashSet < String > ( ) ;
for ( Player player : getOnlinePlayers ( ) ) {
result . addAll ( player . getListeningPluginChannels ( ) ) ;
}
return result ;
}
2012-02-16 23:31:40 +01:00
2014-06-24 23:11:25 +02:00
@Override
2012-02-29 19:56:35 +01:00
public Inventory createInventory ( InventoryHolder owner , InventoryType type ) {
// TODO: Create the appropriate type, rather than Custom?
return new CraftInventoryCustom ( owner , type ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2013-04-11 21:20:41 +02:00
public Inventory createInventory ( InventoryHolder owner , InventoryType type , String title ) {
return new CraftInventoryCustom ( owner , type , title ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-02-29 19:56:35 +01:00
public Inventory createInventory ( InventoryHolder owner , int size ) throws IllegalArgumentException {
Validate . isTrue ( size % 9 = = 0 , " Chests must have a size that is a multiple of 9! " ) ;
return new CraftInventoryCustom ( owner , size ) ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-02-29 19:56:35 +01:00
public Inventory createInventory ( InventoryHolder owner , int size , String title ) throws IllegalArgumentException {
Validate . isTrue ( size % 9 = = 0 , " Chests must have a size that is a multiple of 9! " ) ;
return new CraftInventoryCustom ( owner , size , title ) ;
}
2012-03-01 06:19:11 +01:00
2014-06-24 23:11:25 +02:00
@Override
2012-03-01 06:19:11 +01:00
public HelpMap getHelpMap ( ) {
return helpMap ;
}
public SimpleCommandMap getCommandMap ( ) {
return commandMap ;
}
2012-04-24 02:01:22 +02:00
2014-06-24 23:11:25 +02:00
@Override
2012-04-24 02:01:22 +02:00
public int getMonsterSpawnLimit ( ) {
return monsterSpawn ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-04-24 02:01:22 +02:00
public int getAnimalSpawnLimit ( ) {
return animalSpawn ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-04-24 02:01:22 +02:00
public int getWaterAnimalSpawnLimit ( ) {
return waterAnimalSpawn ;
}
2012-06-14 04:52:49 +02:00
2014-06-24 23:11:25 +02:00
@Override
2012-10-31 17:18:41 +01:00
public int getAmbientSpawnLimit ( ) {
return ambientSpawn ;
}
2014-06-24 23:11:25 +02:00
@Override
2012-06-14 04:52:49 +02:00
public boolean isPrimaryThread ( ) {
return Thread . currentThread ( ) . equals ( console . primaryThread ) ;
}
2012-07-04 01:00:37 +02:00
2014-06-24 23:11:25 +02:00
@Override
2012-07-04 01:00:37 +02:00
public String getMotd ( ) {
2012-07-29 09:33:13 +02:00
return console . getMotd ( ) ;
2012-07-04 01:00:37 +02:00
}
2012-08-07 06:52:49 +02:00
2014-06-24 23:11:25 +02:00
@Override
2012-08-07 06:52:49 +02:00
public WarningState getWarningState ( ) {
return warningState ;
}
2012-10-09 20:44:04 +02:00
public List < String > tabComplete ( net . minecraft . server . ICommandListener sender , String message ) {
if ( ! ( sender instanceof EntityPlayer ) ) {
return ImmutableList . of ( ) ;
}
Player player = ( ( EntityPlayer ) sender ) . getBukkitEntity ( ) ;
if ( message . startsWith ( " / " ) ) {
return tabCompleteCommand ( player , message ) ;
} else {
return tabCompleteChat ( player , message ) ;
}
}
public List < String > tabCompleteCommand ( Player player , String message ) {
List < String > completions = null ;
try {
completions = getCommandMap ( ) . tabComplete ( player , message . substring ( 1 ) ) ;
} catch ( CommandException ex ) {
player . sendMessage ( ChatColor . RED + " An internal error occurred while attempting to tab-complete this command " ) ;
getLogger ( ) . log ( Level . SEVERE , " Exception when " + player . getName ( ) + " attempted to tab complete " + message , ex ) ;
}
2012-10-17 11:31:36 +02:00
2012-10-09 20:44:04 +02:00
return completions = = null ? ImmutableList . < String > of ( ) : completions ;
}
public List < String > tabCompleteChat ( Player player , String message ) {
2012-10-17 11:31:36 +02:00
List < String > completions = new ArrayList < String > ( ) ;
PlayerChatTabCompleteEvent event = new PlayerChatTabCompleteEvent ( player , message , completions ) ;
String token = event . getLastToken ( ) ;
2014-06-24 23:11:25 +02:00
for ( Player p : getOnlinePlayers ( ) ) {
2012-10-17 11:31:36 +02:00
if ( player . canSee ( p ) & & StringUtil . startsWithIgnoreCase ( p . getName ( ) , token ) ) {
2012-10-09 20:44:04 +02:00
completions . add ( p . getName ( ) ) ;
}
}
2012-10-17 11:31:36 +02:00
pluginManager . callEvent ( event ) ;
Iterator < ? > it = completions . iterator ( ) ;
while ( it . hasNext ( ) ) {
Object current = it . next ( ) ;
if ( ! ( current instanceof String ) ) {
// Sanity
it . remove ( ) ;
}
}
Collections . sort ( completions , String . CASE_INSENSITIVE_ORDER ) ;
2012-10-09 20:44:04 +02:00
return completions ;
}
2012-12-17 08:31:41 +01:00
2014-06-24 23:11:25 +02:00
@Override
2012-12-17 08:31:41 +01:00
public CraftItemFactory getItemFactory ( ) {
return CraftItemFactory . instance ( ) ;
}
2013-03-22 22:21:33 +01:00
2014-06-24 23:11:25 +02:00
@Override
2013-03-22 22:21:33 +01:00
public CraftScoreboardManager getScoreboardManager ( ) {
return scoreboardManager ;
}
2013-09-23 23:43:21 +02:00
public void checkSaveState ( ) {
if ( this . playerCommandState | | this . printSaveWarning | | this . console . autosavePeriod < = 0 ) {
return ;
}
this . printSaveWarning = true ;
getLogger ( ) . log ( Level . WARNING , " A manual (plugin-induced) save has been detected while server is configured to auto-save. This may affect performance. " , warningState = = WarningState . ON ? new Throwable ( ) : null ) ;
}
2013-11-04 14:07:38 +01:00
@Override
public CraftIconCache getServerIcon ( ) {
return icon ;
}
@Override
public CraftIconCache loadServerIcon ( File file ) throws Exception {
Validate . notNull ( file , " File cannot be null " ) ;
if ( ! file . isFile ( ) ) {
throw new IllegalArgumentException ( file + " is not a file " ) ;
}
return loadServerIcon0 ( file ) ;
}
static CraftIconCache loadServerIcon0 ( File file ) throws Exception {
return loadServerIcon0 ( ImageIO . read ( file ) ) ;
}
@Override
public CraftIconCache loadServerIcon ( BufferedImage image ) throws Exception {
Validate . notNull ( image , " Image cannot be null " ) ;
return loadServerIcon0 ( image ) ;
}
static CraftIconCache loadServerIcon0 ( BufferedImage image ) throws Exception {
ByteBuf bytebuf = Unpooled . buffer ( ) ;
Validate . isTrue ( image . getWidth ( ) = = 64 , " Must be 64 pixels wide " ) ;
Validate . isTrue ( image . getHeight ( ) = = 64 , " Must be 64 pixels high " ) ;
ImageIO . write ( image , " PNG " , new ByteBufOutputStream ( bytebuf ) ) ;
ByteBuf bytebuf1 = Base64 . encode ( bytebuf ) ;
return new CraftIconCache ( " data:image/png;base64, " + bytebuf1 . toString ( Charsets . UTF_8 ) ) ;
}
2014-01-07 07:33:16 +01:00
2014-06-24 23:11:25 +02:00
@Override
2014-01-07 07:33:16 +01:00
public void setIdleTimeout ( int threshold ) {
2014-03-21 05:26:30 +01:00
console . setIdleTimeout ( threshold ) ;
2014-01-07 07:33:16 +01:00
}
2014-06-24 23:11:25 +02:00
@Override
2014-01-07 07:33:16 +01:00
public int getIdleTimeout ( ) {
2014-03-21 05:26:30 +01:00
return console . getIdleTimeout ( ) ;
2014-01-07 07:33:16 +01:00
}
2014-01-15 05:38:03 +01:00
@Deprecated
@Override
public UnsafeValues getUnsafe ( ) {
return CraftMagicNumbers . INSTANCE ;
}
2010-12-21 17:52:15 +01:00
}