/* * This file is part of fabric-loom, licensed under the MIT License (MIT). * * Copyright (c) 2016, 2017, 2018 FabricMC * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ package net.fabricmc.loom.util; import com.google.common.base.Strings; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.gson.JsonElement; import com.google.gson.JsonObject; import net.fabricmc.loom.LoomGradleExtension; import org.apache.commons.io.FileUtils; import org.apache.commons.io.IOUtils; import org.gradle.api.Project; import org.gradle.api.plugins.JavaPluginConvention; import org.gradle.api.tasks.SourceSet; import org.gradle.plugins.ide.eclipse.model.EclipseModel; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.Node; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.nio.charset.StandardCharsets; import java.util.*; import java.util.stream.Collectors; import java.util.stream.Stream; import java.util.stream.StreamSupport; import static net.fabricmc.loom.AbstractPlugin.isRootProject; public class RunConfig { public String configName; public String eclipseProjectName; public String ideaModuleName; public String mainClass; public String runDir; public String vmArgs; public String programArgs; public List tasksBeforeRun = new ArrayList<>(); public final Map envVariables = new HashMap<>(); public Element genRuns(Element doc) { Element root = this.addXml(doc, "component", ImmutableMap.of("name", "ProjectRunConfigurationManager")); root = addXml(root, "configuration", ImmutableMap.of("default", "false", "name", configName, "type", "Application", "factoryName", "Application")); this.addXml(root, "module", ImmutableMap.of("name", ideaModuleName)); this.addXml(root, "option", ImmutableMap.of("name", "MAIN_CLASS_NAME", "value", mainClass)); this.addXml(root, "option", ImmutableMap.of("name", "WORKING_DIRECTORY", "value", runDir)); if (!Strings.isNullOrEmpty(vmArgs)) { this.addXml(root, "option", ImmutableMap.of("name", "VM_PARAMETERS", "value", vmArgs)); } if (!Strings.isNullOrEmpty(programArgs)) { this.addXml(root, "option", ImmutableMap.of("name", "PROGRAM_PARAMETERS", "value", programArgs)); } if (!envVariables.isEmpty()) { Element envs = this.addXml(root, "envs", ImmutableMap.of()); for (Map.Entry envEntry : envVariables.entrySet()) { this.addXml(envs, "env", ImmutableMap.of("name", envEntry.getKey(), "value", envEntry.getValue())); } } if (!tasksBeforeRun.isEmpty()) { Element methodElement = this.addXml(root, "method", ImmutableMap.of("v", "2")); this.addXml(methodElement, "option", ImmutableMap.of("name", "Make", "enabled", "true")); for (String s : tasksBeforeRun) { String project = s.substring(0, s.lastIndexOf(':')); String task = s.substring(s.lastIndexOf(':') + 1); this.addXml(methodElement, "option", ImmutableMap.builder() .put("name", "Gradle.BeforeRunTask") .put("enabled", "true") .put("tasks", task) .put("externalProjectPath", project) .put("vmOptions", "") .put("scriptParameters", "") .build()); } } return root; } public Element addXml(Node parent, String name, Map values) { Document doc = parent.getOwnerDocument(); if (doc == null) { doc = (Document) parent; } Element e = doc.createElement(name); for (Map.Entry entry : values.entrySet()) { e.setAttribute(entry.getKey(), entry.getValue()); } parent.appendChild(e); return e; } private static String getIdeaModuleName(Project project) { String module = project.getName() + ".main"; while ((project = project.getParent()) != null) { module = project.getName() + "." + module; } return module; } private static void populate(Project project, LoomGradleExtension extension, RunConfig runConfig, String mode) { runConfig.configName += isRootProject(project) ? "" : " (" + project.getPath() + ")"; runConfig.eclipseProjectName = project.getExtensions().getByType(EclipseModel.class).getProject().getName(); runConfig.ideaModuleName = getIdeaModuleName(project); runConfig.runDir = "file://$PROJECT_DIR$/" + extension.runDir; runConfig.vmArgs = ""; if ("launchwrapper".equals(extension.getLoaderLaunchMethod())) { runConfig.mainClass = "net.minecraft.launchwrapper.Launch"; runConfig.programArgs = "--tweakClass " + ("client".equals(mode) ? Constants.LaunchWrapper.DEFAULT_FABRIC_CLIENT_TWEAKER : Constants.LaunchWrapper.DEFAULT_FABRIC_SERVER_TWEAKER); } else { runConfig.mainClass = "net.fabricmc.devlaunchinjector.Main"; runConfig.programArgs = ""; runConfig.vmArgs = "-Dfabric.dli.config=" + encodeEscaped(extension.getDevLauncherConfig().getAbsolutePath()) + " -Dfabric.dli.env=" + mode.toLowerCase(); } if (extension.isForge()) { SourceSet main = project.getConvention().getPlugin(JavaPluginConvention.class).getSourceSets().getByName("main"); String modClasses = Stream.concat( Stream.of(main.getOutput().getResourcesDir().getAbsolutePath()), StreamSupport.stream(main.getOutput().getClassesDirs().spliterator(), false) .map(File::getAbsolutePath) ).map(s -> "loom%%" + s) .collect(Collectors.joining(File.pathSeparator)); runConfig.envVariables.put("MOD_CLASSES", modClasses); } if (extension.getLoaderLaunchMethod().equals("launchwrapper")) { // if installer.json found... JsonObject installerJson = extension.getInstallerJson(); if (installerJson != null) { List sideKeys = ImmutableList.of(mode, "common"); // copy launchwrapper tweakers if (installerJson.has("launchwrapper")) { JsonObject launchwrapperJson = installerJson.getAsJsonObject("launchwrapper"); if (launchwrapperJson.has("tweakers")) { JsonObject tweakersJson = launchwrapperJson.getAsJsonObject("tweakers"); StringBuilder builder = new StringBuilder(); for (String s : sideKeys) { if (tweakersJson.has(s)) { for (JsonElement element : tweakersJson.getAsJsonArray(s)) { builder.append(" --tweakClass ").append(element.getAsString()); } } } runConfig.programArgs += builder.toString(); } } } } } public static RunConfig clientRunConfig(Project project) { LoomGradleExtension extension = project.getExtensions().getByType(LoomGradleExtension.class); RunConfig ideaClient = new RunConfig(); ideaClient.configName = "Minecraft Client"; populate(project, extension, ideaClient, "client"); ideaClient.vmArgs += getOSClientJVMArgs(); ideaClient.vmArgs += " -Dfabric.dli.main=" + getMainClass("client", extension); ideaClient.tasksBeforeRun = new ArrayList<>(extension.getTasksBeforeRun()); return ideaClient; } public static RunConfig serverRunConfig(Project project) { LoomGradleExtension extension = project.getExtensions().getByType(LoomGradleExtension.class); RunConfig ideaServer = new RunConfig(); ideaServer.configName = "Minecraft Server"; populate(project, extension, ideaServer, "server"); ideaServer.vmArgs += " -Dfabric.dli.main=" + getMainClass("server", extension); ideaServer.tasksBeforeRun = new ArrayList<>(extension.getTasksBeforeRun()); return ideaServer; } // This can be removed at somepoint, its not ideal but its the best solution I could thing of public static boolean needsUpgrade(File file) throws IOException { String contents = FileUtils.readFileToString(file, StandardCharsets.UTF_8); return !(contents.contains("net.fabricmc.devlaunchinjector.Main")); } public String fromDummy(String dummy) throws IOException { String dummyConfig; try (InputStream input = SetupIntelijRunConfigs.class.getClassLoader().getResourceAsStream(dummy)) { dummyConfig = IOUtils.toString(input, StandardCharsets.UTF_8); } dummyConfig = dummyConfig.replace("%NAME%", configName); dummyConfig = dummyConfig.replace("%MAIN_CLASS%", mainClass); dummyConfig = dummyConfig.replace("%ECLIPSE_PROJECT%", eclipseProjectName); dummyConfig = dummyConfig.replace("%IDEA_MODULE%", ideaModuleName); dummyConfig = dummyConfig.replace("%PROGRAM_ARGS%", programArgs.replaceAll("\"", """)); dummyConfig = dummyConfig.replace("%VM_ARGS%", vmArgs.replaceAll("\"", """)); String envs = ""; if (!envVariables.isEmpty()) { StringBuilder builder = new StringBuilder(""); for (Map.Entry env : envVariables.entrySet()) { builder.append(""); } builder.append(""); envs = builder.toString(); } dummyConfig = dummyConfig.replace("%ENVS%", envs); StringBuilder tasksToRun = new StringBuilder(); for (String s : tasksBeforeRun) { String project = s.substring(0, s.lastIndexOf(':')); String task = s.substring(s.lastIndexOf(':') + 1); tasksToRun.append("