2018-06-15 19:03:35 +00:00
|
|
|
/*
|
|
|
|
* Minecraft Forge
|
2019-02-10 22:57:03 +00:00
|
|
|
* Copyright (c) 2016-2019.
|
2018-06-15 19:03:35 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation version 2.1
|
|
|
|
* of the License.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
package net.minecraftforge.fml;
|
|
|
|
|
2019-03-11 22:32:37 +00:00
|
|
|
import com.google.common.collect.ImmutableList;
|
2019-01-04 16:44:57 +00:00
|
|
|
import cpw.mods.modlauncher.TransformingClassLoader;
|
2018-06-21 19:37:32 +00:00
|
|
|
import net.minecraftforge.api.distmarker.Dist;
|
2018-06-19 18:04:05 +00:00
|
|
|
import net.minecraftforge.client.event.ModelRegistryEvent;
|
|
|
|
import net.minecraftforge.common.MinecraftForge;
|
|
|
|
import net.minecraftforge.common.capabilities.CapabilityManager;
|
2019-01-29 03:42:37 +00:00
|
|
|
import net.minecraftforge.fml.config.ConfigTracker;
|
|
|
|
import net.minecraftforge.fml.config.ModConfig;
|
2018-06-15 19:03:35 +00:00
|
|
|
import net.minecraftforge.fml.loading.FMLLoader;
|
2019-02-16 08:46:21 +00:00
|
|
|
import net.minecraftforge.fml.loading.FMLPaths;
|
2018-06-15 19:03:35 +00:00
|
|
|
import net.minecraftforge.fml.loading.LoadingModList;
|
|
|
|
import net.minecraftforge.fml.loading.moddiscovery.ModFile;
|
|
|
|
import net.minecraftforge.fml.loading.moddiscovery.ModFileInfo;
|
2019-03-04 13:22:46 +00:00
|
|
|
import net.minecraftforge.fml.network.FMLNetworkConstants;
|
2019-03-17 23:01:23 +00:00
|
|
|
import net.minecraftforge.fml.network.NetworkRegistry;
|
2019-02-16 08:46:21 +00:00
|
|
|
import net.minecraftforge.forgespi.language.IModInfo;
|
2019-02-10 16:44:47 +00:00
|
|
|
import net.minecraftforge.forgespi.language.IModLanguageProvider;
|
2018-06-19 18:04:05 +00:00
|
|
|
import net.minecraftforge.registries.GameData;
|
2019-01-10 04:57:01 +00:00
|
|
|
import net.minecraftforge.registries.ObjectHolderRegistry;
|
2018-06-15 19:03:35 +00:00
|
|
|
import org.apache.logging.log4j.LogManager;
|
|
|
|
import org.apache.logging.log4j.Logger;
|
|
|
|
|
|
|
|
import java.util.Collection;
|
2019-03-11 22:32:37 +00:00
|
|
|
import java.util.Collections;
|
2018-06-15 19:03:35 +00:00
|
|
|
import java.util.List;
|
|
|
|
import java.util.Map;
|
2019-02-10 16:44:47 +00:00
|
|
|
import java.util.Optional;
|
2018-06-15 19:03:35 +00:00
|
|
|
import java.util.function.Function;
|
|
|
|
import java.util.stream.Collectors;
|
2018-06-21 19:37:32 +00:00
|
|
|
|
|
|
|
import static net.minecraftforge.fml.Logging.CORE;
|
2019-01-04 16:44:57 +00:00
|
|
|
import static net.minecraftforge.fml.Logging.LOADING;
|
2018-06-15 19:03:35 +00:00
|
|
|
|
2019-01-15 03:32:43 +00:00
|
|
|
/**
|
|
|
|
* Loads mods.
|
|
|
|
*
|
|
|
|
* Dispatch cycle is seen in {@link #loadMods()} and {@link #finishMods()}
|
|
|
|
*
|
|
|
|
* Overall sequence for loadMods is:
|
|
|
|
* <dl>
|
|
|
|
* <dt>CONSTRUCT</dt>
|
|
|
|
* <dd>Constructs the mod instance. Mods can typically setup basic environment such as Event listeners
|
|
|
|
* and Configuration specifications here.</dd>
|
|
|
|
* <dt>Automated dispatches</dt>
|
|
|
|
* <dd>Dispatches automated elements : {@link net.minecraftforge.fml.common.Mod.EventBusSubscriber},
|
|
|
|
* {@link net.minecraftforge.event.RegistryEvent}, {@link net.minecraftforge.common.capabilities.CapabilityInject}
|
|
|
|
* and others</dd>
|
|
|
|
* <dt>CONFIG_LOAD</dt>
|
|
|
|
* <dd>Dispatches ConfigLoadEvent to mods</dd>
|
|
|
|
* <dt>COMMON_SETUP</dt>
|
|
|
|
* <dd>Dispatches {@link net.minecraftforge.fml.event.lifecycle.FMLCommonSetupEvent} to mods</dd>
|
|
|
|
* <dt>SIDED_SETUP</dt>
|
|
|
|
* <dd>Dispatches {@link net.minecraftforge.fml.event.lifecycle.FMLClientSetupEvent} or
|
|
|
|
* {@link net.minecraftforge.fml.event.lifecycle.FMLDedicatedServerSetupEvent} to mods</dd>
|
|
|
|
* </dl>
|
|
|
|
*
|
|
|
|
* Overall sequence for finishMods is:
|
|
|
|
* <dl>
|
|
|
|
* <dt>ENQUEUE_IMC</dt>
|
|
|
|
* <dd>Dispatches {@link net.minecraftforge.fml.event.lifecycle.InterModEnqueueEvent} to mods,
|
|
|
|
* for enqueuing {@link InterModComms} messages for other mods to receive subsequently</dd>
|
|
|
|
* <dt>PROCESS_IMC</dt>
|
|
|
|
* <dd>Dispatches {@link net.minecraftforge.fml.event.lifecycle.InterModProcessEvent} to mods,
|
|
|
|
* for processing {@link InterModComms} messages received from other mods prior to this event</dd>
|
|
|
|
* <dt>COMPLETE</dt>
|
|
|
|
* <dd>Dispatches {@link net.minecraftforge.fml.event.lifecycle.FMLLoadCompleteEvent} to mods,
|
|
|
|
* and completes the mod loading sequence.</dd>
|
|
|
|
* </dl>
|
|
|
|
*/
|
2018-06-15 19:03:35 +00:00
|
|
|
public class ModLoader
|
|
|
|
{
|
2018-08-27 17:10:07 +00:00
|
|
|
private static final Logger LOGGER = LogManager.getLogger();
|
2018-06-15 19:03:35 +00:00
|
|
|
private static ModLoader INSTANCE;
|
2019-01-04 16:44:57 +00:00
|
|
|
private final TransformingClassLoader launchClassLoader;
|
2018-06-15 19:03:35 +00:00
|
|
|
private final LoadingModList loadingModList;
|
|
|
|
|
2018-09-29 01:07:46 +00:00
|
|
|
private final List<ModLoadingException> loadingExceptions;
|
2019-03-11 22:32:37 +00:00
|
|
|
private final List<ModLoadingWarning> loadingWarnings;
|
2018-06-15 19:03:35 +00:00
|
|
|
private ModLoader()
|
|
|
|
{
|
|
|
|
INSTANCE = this;
|
|
|
|
this.launchClassLoader = FMLLoader.getLaunchClassLoader();
|
|
|
|
this.loadingModList = FMLLoader.getLoadingModList();
|
2018-09-29 01:07:46 +00:00
|
|
|
this.loadingExceptions = FMLLoader.getLoadingModList().
|
2018-10-04 04:57:08 +00:00
|
|
|
getErrors().stream().flatMap(ModLoadingException::fromEarlyException).collect(Collectors.toList());
|
2019-03-11 22:32:37 +00:00
|
|
|
this.loadingWarnings = FMLLoader.getLoadingModList().
|
|
|
|
getBrokenFiles().stream().map(file -> new ModLoadingWarning(null, ModLoadingStage.VALIDATE, "fml.modloading.brokenfile", file.getFileName())).collect(Collectors.toList());
|
2019-03-04 13:22:46 +00:00
|
|
|
LOGGER.info(CORE, "Loading Network data for FML net version: {}", FMLNetworkConstants.NETVERSION);
|
2018-06-15 19:03:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static ModLoader get()
|
|
|
|
{
|
|
|
|
return INSTANCE == null ? INSTANCE = new ModLoader() : INSTANCE;
|
|
|
|
}
|
|
|
|
|
2018-09-29 01:07:46 +00:00
|
|
|
private static Runnable fireClientEvents()
|
2018-06-19 18:04:05 +00:00
|
|
|
{
|
|
|
|
return ()->MinecraftForge.EVENT_BUS.post(new ModelRegistryEvent());
|
|
|
|
}
|
|
|
|
|
2018-06-15 19:03:35 +00:00
|
|
|
public void loadMods() {
|
|
|
|
final ModList modList = ModList.of(loadingModList.getModFiles().stream().map(ModFileInfo::getFile).collect(Collectors.toList()), loadingModList.getMods());
|
2018-10-04 04:57:08 +00:00
|
|
|
if (!this.loadingExceptions.isEmpty()) {
|
|
|
|
throw new LoadingFailedException(loadingExceptions);
|
|
|
|
}
|
2019-02-16 08:46:21 +00:00
|
|
|
final List<ModContainer> modContainers = loadingModList.getModFiles().stream().
|
2018-06-15 19:03:35 +00:00
|
|
|
map(ModFileInfo::getFile).
|
2019-01-06 20:46:29 +00:00
|
|
|
map(mf -> buildMods(mf, launchClassLoader)).
|
2019-02-16 08:46:21 +00:00
|
|
|
flatMap(Collection::stream).
|
|
|
|
collect(Collectors.toList());
|
2018-09-29 01:07:46 +00:00
|
|
|
if (!loadingExceptions.isEmpty()) {
|
2019-01-06 20:46:29 +00:00
|
|
|
LOGGER.fatal(CORE, "Failed to initialize mod containers");
|
2018-09-29 01:07:46 +00:00
|
|
|
throw new LoadingFailedException(loadingExceptions);
|
|
|
|
}
|
2019-02-16 08:46:21 +00:00
|
|
|
modList.setLoadedMods(modContainers);
|
2018-09-29 01:07:46 +00:00
|
|
|
dispatchAndHandleError(LifecycleEventProvider.CONSTRUCT);
|
2019-01-20 04:26:25 +00:00
|
|
|
GameData.fireCreateRegistryEvents(LifecycleEventProvider.CREATE_REGISTRIES, this::dispatchAndHandleError);
|
2019-01-10 04:57:01 +00:00
|
|
|
ObjectHolderRegistry.findObjectHolders();
|
2018-06-19 18:04:05 +00:00
|
|
|
CapabilityManager.INSTANCE.injectCapabilities(modList.getAllScanData());
|
2019-01-20 04:26:25 +00:00
|
|
|
GameData.fireRegistryEvents(rl->true, LifecycleEventProvider.LOAD_REGISTRIES, this::dispatchAndHandleError);
|
2019-01-29 03:42:37 +00:00
|
|
|
DistExecutor.runWhenOn(Dist.CLIENT, ()->()-> ConfigTracker.INSTANCE.loadConfigs(ModConfig.Type.CLIENT, FMLPaths.CONFIGDIR.get()));
|
2019-02-16 02:45:28 +00:00
|
|
|
ConfigTracker.INSTANCE.loadConfigs(ModConfig.Type.COMMON, FMLPaths.CONFIGDIR.get());
|
2019-01-15 03:32:43 +00:00
|
|
|
dispatchAndHandleError(LifecycleEventProvider.SETUP);
|
2018-09-29 01:07:46 +00:00
|
|
|
DistExecutor.runWhenOn(Dist.CLIENT, ModLoader::fireClientEvents);
|
2019-01-15 03:32:43 +00:00
|
|
|
dispatchAndHandleError(LifecycleEventProvider.SIDED_SETUP);
|
2018-09-29 01:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void dispatchAndHandleError(LifecycleEventProvider event) {
|
|
|
|
if (!loadingExceptions.isEmpty()) {
|
|
|
|
LOGGER.error("Skipping lifecycle event {}, {} errors found.", event, loadingExceptions.size());
|
|
|
|
} else {
|
|
|
|
event.dispatch(this::accumulateErrors);
|
|
|
|
}
|
|
|
|
if (!loadingExceptions.isEmpty()) {
|
2019-01-06 20:46:29 +00:00
|
|
|
LOGGER.fatal("Failed to complete lifecycle event {}, {} errors found", event, loadingExceptions.size());
|
2018-09-29 01:07:46 +00:00
|
|
|
throw new LoadingFailedException(loadingExceptions);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private void accumulateErrors(List<ModLoadingException> errors) {
|
|
|
|
loadingExceptions.addAll(errors);
|
2018-06-15 19:03:35 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 20:46:29 +00:00
|
|
|
private List<ModContainer> buildMods(final ModFile modFile, final TransformingClassLoader modClassLoader)
|
2018-06-15 19:03:35 +00:00
|
|
|
{
|
|
|
|
final Map<String, IModInfo> modInfoMap = modFile.getModFileInfo().getMods().stream().collect(Collectors.toMap(IModInfo::getModId, Function.identity()));
|
|
|
|
|
2019-01-04 16:44:57 +00:00
|
|
|
LOGGER.debug(LOADING, "ModContainer is {}", ModContainer.class.getClassLoader());
|
2019-05-12 00:47:27 +00:00
|
|
|
final List<ModContainer> containers = modFile.getScanResult().getTargets().entrySet().stream().
|
|
|
|
map(e -> buildModContainerFromTOML(modFile, modClassLoader, modInfoMap, e)).collect(Collectors.toList());
|
|
|
|
if (containers.size() != modInfoMap.size()) {
|
|
|
|
LOGGER.fatal(LOADING,"File {} constructed {} mods: {}, but had {} mods specified: {}",
|
|
|
|
modFile.getFilePath(),
|
|
|
|
containers.size(), containers.stream().map(ModContainer::getModId).collect(Collectors.toList()),
|
|
|
|
modInfoMap.size(), modInfoMap.values().stream().map(IModInfo::getModId).collect(Collectors.toList()));
|
|
|
|
loadingExceptions.add(new ModLoadingException(null, ModLoadingStage.CONSTRUCT, "fml.modloading.missingclasses", null, modFile.getFilePath()));
|
|
|
|
}
|
|
|
|
return containers;
|
2019-02-10 16:44:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private ModContainer buildModContainerFromTOML(final ModFile modFile, final TransformingClassLoader modClassLoader, final Map<String, IModInfo> modInfoMap, final Map.Entry<String, ? extends IModLanguageProvider.IModLanguageLoader> idToProviderEntry) {
|
|
|
|
try {
|
|
|
|
final String modId = idToProviderEntry.getKey();
|
|
|
|
final IModLanguageProvider.IModLanguageLoader languageLoader = idToProviderEntry.getValue();
|
|
|
|
IModInfo info = Optional.ofNullable(modInfoMap.get(modId)).
|
|
|
|
// throw a missing metadata error if there is no matching modid in the modInfoMap from the mods.toml file
|
|
|
|
orElseThrow(()->new ModLoadingException(null, ModLoadingStage.CONSTRUCT, "fml.modloading.missingmetadata", null, modId));
|
|
|
|
return languageLoader.loadMod(info, modClassLoader, modFile.getScanResult());
|
|
|
|
} catch (ModLoadingException mle) {
|
|
|
|
// exceptions are caught and added to the error list for later handling. Null is returned here.
|
|
|
|
loadingExceptions.add(mle);
|
|
|
|
return null;
|
|
|
|
}
|
2018-06-15 19:03:35 +00:00
|
|
|
}
|
|
|
|
|
2018-06-19 18:04:05 +00:00
|
|
|
public void finishMods()
|
|
|
|
{
|
2019-01-15 03:32:43 +00:00
|
|
|
dispatchAndHandleError(LifecycleEventProvider.ENQUEUE_IMC);
|
|
|
|
dispatchAndHandleError(LifecycleEventProvider.PROCESS_IMC);
|
2018-09-29 01:07:46 +00:00
|
|
|
dispatchAndHandleError(LifecycleEventProvider.COMPLETE);
|
2018-06-21 19:37:32 +00:00
|
|
|
GameData.freezeData();
|
2019-03-17 23:01:23 +00:00
|
|
|
NetworkRegistry.lock();
|
2018-06-19 18:04:05 +00:00
|
|
|
}
|
2018-06-21 19:37:32 +00:00
|
|
|
|
2019-03-11 22:32:37 +00:00
|
|
|
public List<ModLoadingWarning> getWarnings()
|
|
|
|
{
|
|
|
|
return ImmutableList.copyOf(this.loadingWarnings);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void addWarning(ModLoadingWarning warning)
|
|
|
|
{
|
|
|
|
this.loadingWarnings.add(warning);
|
|
|
|
}
|
2018-06-15 19:03:35 +00:00
|
|
|
}
|