mirror of
https://github.com/HMCL-dev/HMCL.git
synced 2025-09-10 04:16:02 -04:00
fetch Forge remote version list only from BMCLAPI
This commit is contained in:
parent
88f6fd1965
commit
a3dc7798e7
@ -123,7 +123,7 @@ public final class VersionsPage extends StackPane implements WizardPage, Refresh
|
|||||||
@Override
|
@Override
|
||||||
public void refresh() {
|
public void refresh() {
|
||||||
getChildren().setAll(spinner);
|
getChildren().setAll(spinner);
|
||||||
executor = versionList.refreshAsync(downloadProvider).finalized((variables, isDependentsSucceeded) -> {
|
executor = versionList.refreshAsync(gameVersion, downloadProvider).finalized((variables, isDependentsSucceeded) -> {
|
||||||
if (isDependentsSucceeded) {
|
if (isDependentsSucceeded) {
|
||||||
List<VersionsPageItem> items = loadVersions();
|
List<VersionsPageItem> items = loadVersions();
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.jackhuang.hmcl.download;
|
package org.jackhuang.hmcl.download;
|
||||||
|
|
||||||
import org.jackhuang.hmcl.download.forge.ForgeVersionList;
|
import org.jackhuang.hmcl.download.forge.ForgeBMCLVersionList;
|
||||||
import org.jackhuang.hmcl.download.game.GameVersionList;
|
import org.jackhuang.hmcl.download.game.GameVersionList;
|
||||||
import org.jackhuang.hmcl.download.liteloader.LiteLoaderBMCLVersionList;
|
import org.jackhuang.hmcl.download.liteloader.LiteLoaderBMCLVersionList;
|
||||||
import org.jackhuang.hmcl.download.optifine.OptiFineBMCLVersionList;
|
import org.jackhuang.hmcl.download.optifine.OptiFineBMCLVersionList;
|
||||||
@ -44,7 +44,7 @@ public class BMCLAPIDownloadProvider implements DownloadProvider {
|
|||||||
case "game":
|
case "game":
|
||||||
return GameVersionList.INSTANCE;
|
return GameVersionList.INSTANCE;
|
||||||
case "forge":
|
case "forge":
|
||||||
return ForgeVersionList.INSTANCE;
|
return ForgeBMCLVersionList.INSTANCE;
|
||||||
case "liteloader":
|
case "liteloader":
|
||||||
return LiteLoaderBMCLVersionList.INSTANCE;
|
return LiteLoaderBMCLVersionList.INSTANCE;
|
||||||
case "optifine":
|
case "optifine":
|
||||||
|
@ -90,7 +90,7 @@ public class DefaultDependencyManager extends AbstractDependencyManager {
|
|||||||
@Override
|
@Override
|
||||||
public Task installLibraryAsync(String gameVersion, Version version, String libraryId, String libraryVersion) {
|
public Task installLibraryAsync(String gameVersion, Version version, String libraryId, String libraryVersion) {
|
||||||
VersionList<?> versionList = getVersionList(libraryId);
|
VersionList<?> versionList = getVersionList(libraryId);
|
||||||
return versionList.loadAsync(getDownloadProvider())
|
return versionList.loadAsync(gameVersion, getDownloadProvider())
|
||||||
.then(variables -> installLibraryAsync(version, versionList.getVersion(gameVersion, libraryVersion)
|
.then(variables -> installLibraryAsync(version, versionList.getVersion(gameVersion, libraryVersion)
|
||||||
.orElseThrow(() -> new IllegalStateException("Remote library " + libraryId + " has no version " + libraryVersion))));
|
.orElseThrow(() -> new IllegalStateException("Remote library " + libraryId + " has no version " + libraryVersion))));
|
||||||
}
|
}
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.jackhuang.hmcl.download;
|
package org.jackhuang.hmcl.download;
|
||||||
|
|
||||||
|
import org.jackhuang.hmcl.download.forge.ForgeBMCLVersionList;
|
||||||
import org.jackhuang.hmcl.download.forge.ForgeVersionList;
|
import org.jackhuang.hmcl.download.forge.ForgeVersionList;
|
||||||
import org.jackhuang.hmcl.download.game.GameVersionList;
|
import org.jackhuang.hmcl.download.game.GameVersionList;
|
||||||
import org.jackhuang.hmcl.download.liteloader.LiteLoaderVersionList;
|
import org.jackhuang.hmcl.download.liteloader.LiteLoaderVersionList;
|
||||||
@ -44,7 +45,7 @@ public class MojangDownloadProvider implements DownloadProvider {
|
|||||||
case "game":
|
case "game":
|
||||||
return GameVersionList.INSTANCE;
|
return GameVersionList.INSTANCE;
|
||||||
case "forge":
|
case "forge":
|
||||||
return ForgeVersionList.INSTANCE;
|
return ForgeBMCLVersionList.INSTANCE;
|
||||||
case "liteloader":
|
case "liteloader":
|
||||||
return LiteLoaderVersionList.INSTANCE;
|
return LiteLoaderVersionList.INSTANCE;
|
||||||
case "optifine":
|
case "optifine":
|
||||||
@ -56,6 +57,7 @@ public class MojangDownloadProvider implements DownloadProvider {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String injectURL(String baseURL) {
|
public String injectURL(String baseURL) {
|
||||||
return baseURL;
|
return baseURL
|
||||||
|
.replaceFirst("https?://files\\.minecraftforge\\.net/maven", "https://bmclapi2.bangbang93.com/maven");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -46,6 +46,14 @@ public abstract class VersionList<T extends RemoteVersion> {
|
|||||||
return !versions.isEmpty();
|
return !versions.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* True if the version list that contains the remote versions which depends on the specific game version has been loaded.
|
||||||
|
* @param gameVersion the remote version depends on
|
||||||
|
*/
|
||||||
|
public boolean isLoaded(String gameVersion) {
|
||||||
|
return !versions.get(gameVersion).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
public abstract boolean hasType();
|
public abstract boolean hasType();
|
||||||
|
|
||||||
protected final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
|
protected final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
|
||||||
@ -56,6 +64,15 @@ public abstract class VersionList<T extends RemoteVersion> {
|
|||||||
*/
|
*/
|
||||||
public abstract Task refreshAsync(DownloadProvider downloadProvider);
|
public abstract Task refreshAsync(DownloadProvider downloadProvider);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param gameVersion the remote version depends on
|
||||||
|
* @param downloadProvider DownloadProvider
|
||||||
|
* @return the task to reload the remote version list.
|
||||||
|
*/
|
||||||
|
public Task refreshAsync(String gameVersion, DownloadProvider downloadProvider) {
|
||||||
|
return refreshAsync(downloadProvider);
|
||||||
|
}
|
||||||
|
|
||||||
public Task loadAsync(DownloadProvider downloadProvider) {
|
public Task loadAsync(DownloadProvider downloadProvider) {
|
||||||
return Task.ofThen(variables -> {
|
return Task.ofThen(variables -> {
|
||||||
lock.readLock().lock();
|
lock.readLock().lock();
|
||||||
@ -70,6 +87,20 @@ public abstract class VersionList<T extends RemoteVersion> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Task loadAsync(String gameVersion, DownloadProvider downloadProvider) {
|
||||||
|
return Task.ofThen(variables -> {
|
||||||
|
lock.readLock().lock();
|
||||||
|
boolean loaded;
|
||||||
|
|
||||||
|
try {
|
||||||
|
loaded = isLoaded(gameVersion);
|
||||||
|
} finally {
|
||||||
|
lock.readLock().unlock();
|
||||||
|
}
|
||||||
|
return loaded ? null : refreshAsync(gameVersion, downloadProvider);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
protected Collection<T> getVersionsImpl(String gameVersion) {
|
protected Collection<T> getVersionsImpl(String gameVersion) {
|
||||||
lock.readLock().lock();
|
lock.readLock().lock();
|
||||||
try {
|
try {
|
||||||
|
@ -0,0 +1,185 @@
|
|||||||
|
/*
|
||||||
|
* Hello Minecraft! Launcher
|
||||||
|
* Copyright (C) 2019 huangyuhui <huanghongxun2008@126.com> and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program 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 General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package org.jackhuang.hmcl.download.forge;
|
||||||
|
|
||||||
|
import com.google.gson.JsonParseException;
|
||||||
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
import org.jackhuang.hmcl.download.DownloadProvider;
|
||||||
|
import org.jackhuang.hmcl.download.VersionList;
|
||||||
|
import org.jackhuang.hmcl.task.GetTask;
|
||||||
|
import org.jackhuang.hmcl.task.Task;
|
||||||
|
import org.jackhuang.hmcl.util.Immutable;
|
||||||
|
import org.jackhuang.hmcl.util.StringUtils;
|
||||||
|
import org.jackhuang.hmcl.util.gson.JsonUtils;
|
||||||
|
import org.jackhuang.hmcl.util.gson.Validation;
|
||||||
|
import org.jackhuang.hmcl.util.io.NetworkUtils;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public final class ForgeBMCLVersionList extends VersionList<ForgeRemoteVersion> {
|
||||||
|
|
||||||
|
public static final ForgeBMCLVersionList INSTANCE = new ForgeBMCLVersionList();
|
||||||
|
|
||||||
|
private ForgeBMCLVersionList() {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasType() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Task loadAsync(DownloadProvider downloadProvider) {
|
||||||
|
throw new UnsupportedOperationException("ForgeBMCLVersionList does not support loading the entire Forge remote version list.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Task refreshAsync(DownloadProvider downloadProvider) {
|
||||||
|
throw new UnsupportedOperationException("ForgeBMCLVersionList does not support loading the entire Forge remote version list.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Task refreshAsync(String gameVersion, DownloadProvider downloadProvider) {
|
||||||
|
final GetTask task = new GetTask(NetworkUtils.toURL("https://bmclapi2.bangbang93.com/forge/minecraft/" + gameVersion));
|
||||||
|
return new Task() {
|
||||||
|
@Override
|
||||||
|
public Collection<? extends Task> getDependents() {
|
||||||
|
return Collections.singleton(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute() {
|
||||||
|
lock.writeLock().lock();
|
||||||
|
|
||||||
|
try {
|
||||||
|
List<ForgeVersion> forgeVersions = JsonUtils.GSON.fromJson(task.getResult(), new TypeToken<List<ForgeVersion>>() {
|
||||||
|
}.getType());
|
||||||
|
versions.clear(gameVersion);
|
||||||
|
if (forgeVersions == null) return;
|
||||||
|
for (ForgeVersion version : forgeVersions) {
|
||||||
|
if (version == null)
|
||||||
|
continue;
|
||||||
|
String jar = null;
|
||||||
|
for (ForgeVersion.File file : version.getFiles())
|
||||||
|
if ("installer".equals(file.getCategory())) {
|
||||||
|
String classifier = gameVersion + "-" + version.getVersion()
|
||||||
|
+ (StringUtils.isNotBlank(version.getBranch()) ? "-" + version.getBranch() : "");
|
||||||
|
String fileName = "forge-" + classifier + "-" + file.getCategory() + "." + file.getFormat();
|
||||||
|
jar = "https://bmclapi2.bangbang93.com/maven/net/minecraftforge/forge/" + classifier + "/" + fileName;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (jar == null)
|
||||||
|
continue;
|
||||||
|
versions.put(gameVersion, new ForgeRemoteVersion(
|
||||||
|
version.getGameVersion(), version.getVersion(), jar
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
lock.writeLock().unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
public static final class ForgeVersion implements Validation {
|
||||||
|
|
||||||
|
private final String branch;
|
||||||
|
private final String mcversion;
|
||||||
|
private final String version;
|
||||||
|
private final int build;
|
||||||
|
private final List<File> files;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* No-arg constructor for Gson.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public ForgeVersion() {
|
||||||
|
this(null, null, null, 0, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ForgeVersion(String branch, String mcversion, String version, int build, List<File> files) {
|
||||||
|
this.branch = branch;
|
||||||
|
this.mcversion = mcversion;
|
||||||
|
this.version = version;
|
||||||
|
this.build = build;
|
||||||
|
this.files = files;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getBranch() {
|
||||||
|
return branch;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getGameVersion() {
|
||||||
|
return mcversion;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getBuild() {
|
||||||
|
return build;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<File> getFiles() {
|
||||||
|
return files;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void validate() throws JsonParseException {
|
||||||
|
if (files == null)
|
||||||
|
throw new JsonParseException("ForgeVersion files cannot be null");
|
||||||
|
if (version == null)
|
||||||
|
throw new JsonParseException("ForgeVersion version cannot be null");
|
||||||
|
if (mcversion == null)
|
||||||
|
throw new JsonParseException("ForgeVersion mcversion cannot be null");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
public static final class File {
|
||||||
|
private final String format;
|
||||||
|
private final String category;
|
||||||
|
private final String hash;
|
||||||
|
|
||||||
|
public File() {
|
||||||
|
this("", "", "");
|
||||||
|
}
|
||||||
|
|
||||||
|
public File(String format, String category, String hash) {
|
||||||
|
this.format = format;
|
||||||
|
this.category = category;
|
||||||
|
this.hash = hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFormat() {
|
||||||
|
return format;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCategory() {
|
||||||
|
return category;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getHash() {
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -18,24 +18,17 @@
|
|||||||
package org.jackhuang.hmcl.download.forge;
|
package org.jackhuang.hmcl.download.forge;
|
||||||
|
|
||||||
import org.jackhuang.hmcl.download.DefaultDependencyManager;
|
import org.jackhuang.hmcl.download.DefaultDependencyManager;
|
||||||
import org.jackhuang.hmcl.game.Library;
|
|
||||||
import org.jackhuang.hmcl.game.SimpleVersionProvider;
|
|
||||||
import org.jackhuang.hmcl.game.Version;
|
import org.jackhuang.hmcl.game.Version;
|
||||||
import org.jackhuang.hmcl.task.FileDownloadTask;
|
import org.jackhuang.hmcl.task.FileDownloadTask;
|
||||||
import org.jackhuang.hmcl.task.Task;
|
import org.jackhuang.hmcl.task.Task;
|
||||||
import org.jackhuang.hmcl.task.TaskResult;
|
import org.jackhuang.hmcl.task.TaskResult;
|
||||||
import org.jackhuang.hmcl.util.gson.JsonUtils;
|
|
||||||
import org.jackhuang.hmcl.util.io.FileUtils;
|
|
||||||
import org.jackhuang.hmcl.util.io.IOUtils;
|
|
||||||
import org.jackhuang.hmcl.util.io.NetworkUtils;
|
import org.jackhuang.hmcl.util.io.NetworkUtils;
|
||||||
|
import org.jackhuang.hmcl.util.versioning.VersionNumber;
|
||||||
|
|
||||||
import java.io.*;
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.zip.ZipEntry;
|
|
||||||
import java.util.zip.ZipFile;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
@ -45,31 +38,48 @@ public final class ForgeInstallTask extends TaskResult<Version> {
|
|||||||
|
|
||||||
private final DefaultDependencyManager dependencyManager;
|
private final DefaultDependencyManager dependencyManager;
|
||||||
private final Version version;
|
private final Version version;
|
||||||
private final File installer = new File("forge-installer.jar").getAbsoluteFile();
|
private Path installer;
|
||||||
private final ForgeRemoteVersion remote;
|
private final ForgeRemoteVersion remote;
|
||||||
private final List<Task> dependents = new LinkedList<>();
|
private Task dependent;
|
||||||
private final List<Task> dependencies = new LinkedList<>();
|
private TaskResult<Version> dependency;
|
||||||
|
|
||||||
private Task downloadFileTask() {
|
|
||||||
return new FileDownloadTask(NetworkUtils.toURL(remote.getUrl()), installer);
|
|
||||||
}
|
|
||||||
|
|
||||||
public ForgeInstallTask(DefaultDependencyManager dependencyManager, Version version, ForgeRemoteVersion remoteVersion) {
|
public ForgeInstallTask(DefaultDependencyManager dependencyManager, Version version, ForgeRemoteVersion remoteVersion) {
|
||||||
this.dependencyManager = dependencyManager;
|
this.dependencyManager = dependencyManager;
|
||||||
this.version = version;
|
this.version = version;
|
||||||
this.remote = remoteVersion;
|
this.remote = remoteVersion;
|
||||||
|
}
|
||||||
|
|
||||||
dependents.add(downloadFileTask());
|
@Override
|
||||||
|
public boolean doPreExecute() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void preExecute() throws Exception {
|
||||||
|
installer = Files.createTempFile("forge-installer", ".jar");
|
||||||
|
|
||||||
|
dependent = new FileDownloadTask(NetworkUtils.toURL(remote.getUrl()), installer.toFile());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean doPostExecute() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void postExecute() throws Exception {
|
||||||
|
Files.deleteIfExists(installer);
|
||||||
|
setResult(dependency.getResult());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Collection<Task> getDependents() {
|
public Collection<Task> getDependents() {
|
||||||
return dependents;
|
return Collections.singleton(dependent);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<Task> getDependencies() {
|
public Collection<Task> getDependencies() {
|
||||||
return dependencies;
|
return Collections.singleton(dependency);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -83,38 +93,10 @@ public final class ForgeInstallTask extends TaskResult<Version> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void execute() throws Exception {
|
public void execute() {
|
||||||
try (ZipFile zipFile = new ZipFile(installer)) {
|
if (VersionNumber.VERSION_COMPARATOR.compare("1.13", remote.getGameVersion()) <= 0)
|
||||||
InputStream stream = zipFile.getInputStream(zipFile.getEntry("install_profile.json"));
|
dependency = new ForgeNewInstallTask(dependencyManager, version, installer);
|
||||||
if (stream == null)
|
else
|
||||||
throw new IOException("Malformed forge installer file, install_profile.json does not exist.");
|
dependency = new ForgeOldInstallTask(dependencyManager, version, installer);
|
||||||
String json = IOUtils.readFullyAsString(stream);
|
|
||||||
ForgeInstallProfile installProfile = JsonUtils.fromNonNullJson(json, ForgeInstallProfile.class);
|
|
||||||
|
|
||||||
// unpack the universal jar in the installer file.
|
|
||||||
Library forgeLibrary = Library.fromName(installProfile.getInstall().getPath());
|
|
||||||
File forgeFile = dependencyManager.getGameRepository().getLibraryFile(version, forgeLibrary);
|
|
||||||
if (!FileUtils.makeFile(forgeFile))
|
|
||||||
throw new IOException("Cannot make directory " + forgeFile.getParent());
|
|
||||||
|
|
||||||
ZipEntry forgeEntry = zipFile.getEntry(installProfile.getInstall().getFilePath());
|
|
||||||
try (InputStream is = zipFile.getInputStream(forgeEntry); OutputStream os = new FileOutputStream(forgeFile)) {
|
|
||||||
IOUtils.copyTo(is, os);
|
|
||||||
}
|
|
||||||
|
|
||||||
// resolve the version
|
|
||||||
SimpleVersionProvider provider = new SimpleVersionProvider();
|
|
||||||
provider.addVersion(version);
|
|
||||||
|
|
||||||
setResult(installProfile.getVersionInfo()
|
|
||||||
.setInheritsFrom(version.getId())
|
|
||||||
.resolve(provider).setJar(null)
|
|
||||||
.setId(version.getId()).setLogging(Collections.emptyMap()));
|
|
||||||
|
|
||||||
dependencies.add(dependencyManager.checkLibraryCompletionAsync(installProfile.getVersionInfo()));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!installer.delete())
|
|
||||||
throw new IOException("Unable to delete installer file" + installer);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,94 @@
|
|||||||
|
/*
|
||||||
|
* Hello Minecraft! Launcher
|
||||||
|
* Copyright (C) 2019 huangyuhui <huanghongxun2008@126.com> and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program 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 General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package org.jackhuang.hmcl.download.forge;
|
||||||
|
|
||||||
|
import org.jackhuang.hmcl.download.DefaultDependencyManager;
|
||||||
|
import org.jackhuang.hmcl.game.*;
|
||||||
|
import org.jackhuang.hmcl.task.Task;
|
||||||
|
import org.jackhuang.hmcl.task.TaskResult;
|
||||||
|
import org.jackhuang.hmcl.util.gson.JsonUtils;
|
||||||
|
import org.jackhuang.hmcl.util.io.FileUtils;
|
||||||
|
import org.jackhuang.hmcl.util.io.IOUtils;
|
||||||
|
|
||||||
|
import java.io.*;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.zip.ZipEntry;
|
||||||
|
import java.util.zip.ZipFile;
|
||||||
|
|
||||||
|
public class ForgeOldInstallTask extends TaskResult<Version> {
|
||||||
|
|
||||||
|
private final DefaultDependencyManager dependencyManager;
|
||||||
|
private final Version version;
|
||||||
|
private final Path installer;
|
||||||
|
private final List<Task> dependencies = new LinkedList<>();
|
||||||
|
|
||||||
|
public ForgeOldInstallTask(DefaultDependencyManager dependencyManager, Version version, Path installer) {
|
||||||
|
this.dependencyManager = dependencyManager;
|
||||||
|
this.version = version;
|
||||||
|
this.installer = installer;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Task> getDependencies() {
|
||||||
|
return dependencies;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getId() {
|
||||||
|
return "version";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean doPreExecute() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute() throws Exception {
|
||||||
|
try (ZipFile zipFile = new ZipFile(installer.toFile())) {
|
||||||
|
InputStream stream = zipFile.getInputStream(zipFile.getEntry("install_profile.json"));
|
||||||
|
if (stream == null)
|
||||||
|
throw new IOException("Malformed forge installer file, install_profile.json does not exist.");
|
||||||
|
String json = IOUtils.readFullyAsString(stream);
|
||||||
|
ForgeInstallProfile installProfile = JsonUtils.fromNonNullJson(json, ForgeInstallProfile.class);
|
||||||
|
|
||||||
|
// unpack the universal jar in the installer file.
|
||||||
|
Library forgeLibrary = Library.fromName(installProfile.getInstall().getPath());
|
||||||
|
File forgeFile = dependencyManager.getGameRepository().getLibraryFile(version, forgeLibrary);
|
||||||
|
if (!FileUtils.makeFile(forgeFile))
|
||||||
|
throw new IOException("Cannot make directory " + forgeFile.getParent());
|
||||||
|
|
||||||
|
ZipEntry forgeEntry = zipFile.getEntry(installProfile.getInstall().getFilePath());
|
||||||
|
try (InputStream is = zipFile.getInputStream(forgeEntry); OutputStream os = new FileOutputStream(forgeFile)) {
|
||||||
|
IOUtils.copyTo(is, os);
|
||||||
|
}
|
||||||
|
|
||||||
|
// resolve the version
|
||||||
|
SimpleVersionProvider provider = new SimpleVersionProvider();
|
||||||
|
provider.addVersion(version);
|
||||||
|
|
||||||
|
setResult(installProfile.getVersionInfo()
|
||||||
|
.setInheritsFrom(version.getId())
|
||||||
|
.resolve(provider).setJar(null)
|
||||||
|
.setId(version.getId()).setLogging(Collections.emptyMap()));
|
||||||
|
|
||||||
|
dependencies.add(dependencyManager.checkLibraryCompletionAsync(installProfile.getVersionInfo()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -47,12 +47,11 @@ public final class ForgeVersionList extends VersionList<ForgeRemoteVersion> {
|
|||||||
@Override
|
@Override
|
||||||
public Task refreshAsync(DownloadProvider downloadProvider) {
|
public Task refreshAsync(DownloadProvider downloadProvider) {
|
||||||
final GetTask task = new GetTask(NetworkUtils.toURL(downloadProvider.injectURL(FORGE_LIST)));
|
final GetTask task = new GetTask(NetworkUtils.toURL(downloadProvider.injectURL(FORGE_LIST)));
|
||||||
final List<Task> dependents = Collections.singletonList(task);
|
|
||||||
return new Task() {
|
return new Task() {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Collection<Task> getDependents() {
|
public Collection<Task> getDependents() {
|
||||||
return dependents;
|
return Collections.singleton(task);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -88,4 +88,11 @@ public final class SimpleMultimap<K, V> {
|
|||||||
public void clear() {
|
public void clear() {
|
||||||
map.clear();
|
map.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void clear(K key) {
|
||||||
|
if (map.containsKey(key))
|
||||||
|
map.get(key).clear();
|
||||||
|
else
|
||||||
|
map.put(key, valuer.get());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user