mirror of
				https://github.com/packwiz/packwiz-installer.git
				synced 2025-10-20 17:14:32 +02:00 
			
		
		
		
	Compare commits
	
		
			46 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 46771ce870 | ||
|  | b143f67acd | ||
|  | 03b0f1b09b | ||
|  | 6c6a0100fd | ||
|  | 6d47c0d61f | ||
|  | 226e754547 | ||
|  | 2c02703101 | ||
|  | 81a60cc759 | ||
|  | 92afa93fd7 | ||
|  | 0858c90079 | ||
|  | 1d4c94f5b6 | ||
|  | 74ddca5d54 | ||
|  | 0df48d19a9 | ||
|  | f5b22f37a4 | ||
|  | f52cd19ad4 | ||
|  | 60887a4312 | ||
|  | a368268038 | ||
|  | 8beded7b41 | ||
|  | 91060dcd54 | ||
|  | e06ee21f3b | ||
|  | b3370739a5 | ||
|  | ecc6f0440a | ||
|  | 92b44352b3 | ||
|  | 1d5a787b02 | ||
|  | b5983800e8 | ||
|  | 4b3c279e71 | ||
|  | b413371306 | ||
|  | 1d2ec61232 | ||
|  | a0da889a02 | ||
|  | 432bb4e25f | ||
|  | c89d3b1e47 | ||
|  | e8538c22bc | ||
|  | a15489f5e4 | ||
|  | 9d3587c72e | ||
|  | bead683b7c | ||
|  | 0770029dc6 | ||
|  | ecaab219c2 | ||
|  | b45a2983e7 | ||
|  | c0c318772b | ||
|  | 580408b92a | ||
|  | dbdd1fb9f3 | ||
|  | 79a983bc2f | ||
|  | 0cba5ba17b | ||
|  | ce60cdc385 | ||
|  | b314fc8e0b | ||
|  | ca4a13589d | 
							
								
								
									
										30
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,9 +1,9 @@ | |||||||
|  |  | ||||||
| # Created by https://www.gitignore.io/api/java,gradle,intellij | # Created by https://www.toptal.com/developers/gitignore/api/java,gradle,intellij+all | ||||||
| # Edit at https://www.gitignore.io/?templates=java,gradle,intellij | # Edit at https://www.toptal.com/developers/gitignore?templates=java,gradle,intellij+all | ||||||
|  |  | ||||||
| ### Intellij ### | ### Intellij+all ### | ||||||
| # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and WebStorm | # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider | ||||||
| # Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839 | # Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839 | ||||||
|  |  | ||||||
| # User-specific stuff | # User-specific stuff | ||||||
| @@ -33,6 +33,9 @@ | |||||||
| # When using Gradle or Maven with auto-import, you should exclude module files, | # When using Gradle or Maven with auto-import, you should exclude module files, | ||||||
| # since they will be recreated, and may cause churn.  Uncomment if using | # since they will be recreated, and may cause churn.  Uncomment if using | ||||||
| # auto-import. | # auto-import. | ||||||
|  | # .idea/artifacts | ||||||
|  | # .idea/compiler.xml | ||||||
|  | # .idea/jarRepositories.xml | ||||||
| # .idea/modules.xml | # .idea/modules.xml | ||||||
| # .idea/*.iml | # .idea/*.iml | ||||||
| # .idea/modules | # .idea/modules | ||||||
| @@ -72,13 +75,18 @@ fabric.properties | |||||||
| # Android studio 3.1+ serialized cache file | # Android studio 3.1+ serialized cache file | ||||||
| .idea/caches/build_file_checksums.ser | .idea/caches/build_file_checksums.ser | ||||||
|  |  | ||||||
| ### Intellij Patch ### | ### Intellij+all Patch ### | ||||||
| # Comment Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-215987721 | # Ignores the whole .idea folder and all .iml files | ||||||
|  | # See https://github.com/joeblau/gitignore.io/issues/186 and https://github.com/joeblau/gitignore.io/issues/360 | ||||||
|  |  | ||||||
| # *.iml | .idea/ | ||||||
| # modules.xml |  | ||||||
| # .idea/misc.xml | # Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-249601023 | ||||||
| # *.ipr |  | ||||||
|  | *.iml | ||||||
|  | modules.xml | ||||||
|  | .idea/misc.xml | ||||||
|  | *.ipr | ||||||
|  |  | ||||||
| # Sonarlint plugin | # Sonarlint plugin | ||||||
| .idea/sonarlint | .idea/sonarlint | ||||||
| @@ -127,4 +135,4 @@ gradle-app.setting | |||||||
| ### Gradle Patch ### | ### Gradle Patch ### | ||||||
| **/build/ | **/build/ | ||||||
|  |  | ||||||
| # End of https://www.gitignore.io/api/java,gradle,intellij | # End of https://www.toptal.com/developers/gitignore/api/java,gradle,intellij+all | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										36
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,36 +0,0 @@ | |||||||
| <component name="InspectionProjectProfileManager"> |  | ||||||
|   <profile version="1.0"> |  | ||||||
|     <option name="myName" value="Project Default" /> |  | ||||||
|     <inspection_tool class="JavaDoc" enabled="true" level="WARNING" enabled_by_default="true"> |  | ||||||
|       <option name="TOP_LEVEL_CLASS_OPTIONS"> |  | ||||||
|         <value> |  | ||||||
|           <option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" /> |  | ||||||
|           <option name="REQUIRED_TAGS" value="" /> |  | ||||||
|         </value> |  | ||||||
|       </option> |  | ||||||
|       <option name="INNER_CLASS_OPTIONS"> |  | ||||||
|         <value> |  | ||||||
|           <option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" /> |  | ||||||
|           <option name="REQUIRED_TAGS" value="" /> |  | ||||||
|         </value> |  | ||||||
|       </option> |  | ||||||
|       <option name="METHOD_OPTIONS"> |  | ||||||
|         <value> |  | ||||||
|           <option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" /> |  | ||||||
|           <option name="REQUIRED_TAGS" value="@return@param@throws or @exception" /> |  | ||||||
|         </value> |  | ||||||
|       </option> |  | ||||||
|       <option name="FIELD_OPTIONS"> |  | ||||||
|         <value> |  | ||||||
|           <option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" /> |  | ||||||
|           <option name="REQUIRED_TAGS" value="" /> |  | ||||||
|         </value> |  | ||||||
|       </option> |  | ||||||
|       <option name="IGNORE_DEPRECATED" value="false" /> |  | ||||||
|       <option name="IGNORE_JAVADOC_PERIOD" value="true" /> |  | ||||||
|       <option name="IGNORE_DUPLICATED_THROWS" value="false" /> |  | ||||||
|       <option name="IGNORE_POINT_TO_ITSELF" value="false" /> |  | ||||||
|       <option name="myAdditionalJavadocTags" value="wbp.parser.entryPoint" /> |  | ||||||
|     </inspection_tool> |  | ||||||
|   </profile> |  | ||||||
| </component> |  | ||||||
							
								
								
									
										10
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										10
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,10 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8"?> |  | ||||||
| <project version="4"> |  | ||||||
|   <component name="EntryPointsManager"> |  | ||||||
|     <list size="1"> |  | ||||||
|       <item index="0" class="java.lang.String" itemvalue="com.google.gson.annotations.SerializedName" /> |  | ||||||
|     </list> |  | ||||||
|   </component> |  | ||||||
|   <component name="ExternalStorageConfigurationManager" enabled="true" /> |  | ||||||
|   <component name="ProjectRootManager" version="2" languageLevel="JDK_1_8" default="false" project-jdk-name="11" project-jdk-type="JavaSDK" /> |  | ||||||
| </project> |  | ||||||
							
								
								
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,6 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8"?> |  | ||||||
| <project version="4"> |  | ||||||
|   <component name="VcsDirectoryMappings"> |  | ||||||
|     <mapping directory="" vcs="Git" /> |  | ||||||
|   </component> |  | ||||||
| </project> |  | ||||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| MIT License | MIT License | ||||||
|  |  | ||||||
| Copyright (c) 2019 | Copyright (c) 2021 comp500 | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
| of this software and associated documentation files (the "Software"), to deal | of this software and associated documentation files (the "Software"), to deal | ||||||
|   | |||||||
| @@ -1,2 +1,2 @@ | |||||||
| # packwiz-installer | # packwiz-installer | ||||||
| An installer for launching packwiz modpacks with MultiMC. | An installer for launching packwiz modpacks with MultiMC. You'll need [the bootstrapper](https://github.com/comp500/packwiz-installer-bootstrap/releases) to actually use this. | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										50
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -1,50 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'java' |  | ||||||
|     id 'application' |  | ||||||
|     id 'com.github.johnrengelman.shadow' version '5.0.0' |  | ||||||
|     id 'com.palantir.git-version' version '0.11.0' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| sourceCompatibility = 1.8 |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
| 	implementation 'commons-cli:commons-cli:1.4' |  | ||||||
| 	implementation 'com.moandjiezana.toml:toml4j:0.7.2' |  | ||||||
|     // TODO: Implement tests |  | ||||||
|     //testImplementation 'junit:junit:4.12' |  | ||||||
|     implementation 'com.google.code.gson:gson:2.8.1' |  | ||||||
|     implementation 'com.squareup.okio:okio:2.2.2' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| repositories { |  | ||||||
|     jcenter() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| mainClassName = 'link.infra.packwiz.installer.RequiresBootstrap' |  | ||||||
| version gitVersion() |  | ||||||
|  |  | ||||||
| jar { |  | ||||||
|     manifest { |  | ||||||
|         attributes( |  | ||||||
|             'Main-Class': 'link.infra.packwiz.installer.RequiresBootstrap', |  | ||||||
|             'Implementation-Version': project.version |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Commons CLI and Minimal JSON are already included in packwiz-installer-bootstrap |  | ||||||
| shadowJar { |  | ||||||
|    dependencies { |  | ||||||
|        exclude(dependency('commons-cli:commons-cli:1.4')) |  | ||||||
|        exclude(dependency('com.eclipsesource.minimal-json:minimal-json:0.9.5')) |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Used for vscode launch.json |  | ||||||
| task copyJar(type: Copy) { |  | ||||||
|     from shadowJar |  | ||||||
|     rename "packwiz-installer-(.*)\\.jar", "packwiz-installer.jar" |  | ||||||
|     into "build/libs/" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| build.dependsOn copyJar |  | ||||||
							
								
								
									
										137
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,137 @@ | |||||||
|  | buildscript { | ||||||
|  | 	repositories { | ||||||
|  | 		jcenter() | ||||||
|  | 	} | ||||||
|  | 	dependencies { | ||||||
|  | 		classpath("com.guardsquare:proguard-gradle:7.0.0") { | ||||||
|  | 			exclude("com.android.tools.build") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | plugins { | ||||||
|  | 	java | ||||||
|  | 	application | ||||||
|  | 	id("com.github.johnrengelman.shadow") version "6.1.0" | ||||||
|  | 	id("com.palantir.git-version") version "0.12.3" | ||||||
|  | 	id("com.github.breadmoirai.github-release") version "2.2.12" | ||||||
|  | 	kotlin("jvm") version "1.4.21" | ||||||
|  | 	id("com.github.jk1.dependency-license-report") version "1.16" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | java { | ||||||
|  | 	sourceCompatibility = JavaVersion.VERSION_1_8 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val shrinkClasspath: Configuration by configurations.creating | ||||||
|  |  | ||||||
|  | dependencies { | ||||||
|  | 	implementation("commons-cli:commons-cli:1.4") | ||||||
|  | 	shrinkClasspath("commons-cli:commons-cli:1.4") | ||||||
|  | 	implementation("com.moandjiezana.toml:toml4j:0.7.2") | ||||||
|  | 	implementation("com.google.code.gson:gson:2.8.1") | ||||||
|  | 	implementation("com.squareup.okio:okio:2.9.0") | ||||||
|  | 	implementation(kotlin("stdlib-jdk8")) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | repositories { | ||||||
|  | 	jcenter() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | application { | ||||||
|  | 	mainClassName = "link.infra.packwiz.installer.RequiresBootstrap" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val gitVersion: groovy.lang.Closure<*> by extra | ||||||
|  | version = gitVersion() | ||||||
|  |  | ||||||
|  | tasks.jar { | ||||||
|  | 	manifest { | ||||||
|  | 		attributes["Main-Class"] = "link.infra.packwiz.installer.RequiresBootstrap" | ||||||
|  | 		attributes["Implementation-Version"] = project.version | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | licenseReport { | ||||||
|  | 	renderers = arrayOf<com.github.jk1.license.render.ReportRenderer>( | ||||||
|  | 		com.github.jk1.license.render.InventoryMarkdownReportRenderer("licenses.md", "packwiz-installer") | ||||||
|  | 	) | ||||||
|  | 	filters = arrayOf<com.github.jk1.license.filter.DependencyFilter>(com.github.jk1.license.filter.LicenseBundleNormalizer()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // TODO: build relocated jar for minecraft launcher lib, non-relocated jar for packwiz-installer | ||||||
|  | //tasks.register<com.github.jengelman.gradle.plugins.shadow.tasks.ConfigureShadowRelocation>("relocateShadowJar") { | ||||||
|  | //	target = tasks.shadowJar.get() | ||||||
|  | //	prefix = "link.infra.packwiz.deps" | ||||||
|  | //} | ||||||
|  |  | ||||||
|  | // Commons CLI and Minimal JSON are already included in packwiz-installer-bootstrap | ||||||
|  | tasks.shadowJar { | ||||||
|  | 	dependencies { | ||||||
|  | 		exclude(dependency("commons-cli:commons-cli:1.4")) | ||||||
|  | 		exclude(dependency("com.eclipsesource.minimal-json:minimal-json:0.9.5")) | ||||||
|  | 		// TODO: exclude meta inf files | ||||||
|  | 	} | ||||||
|  | 	exclude("**/*.kotlin_metadata") | ||||||
|  | 	exclude("**/*.kotlin_builtins") | ||||||
|  | 	exclude("META-INF/maven/**/*") | ||||||
|  | 	exclude("META-INF/proguard/**/*") | ||||||
|  | 	//dependsOn(tasks.named("relocateShadowJar")) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks.register<proguard.gradle.ProGuardTask>("shrinkJar") { | ||||||
|  | 	injars(tasks.shadowJar) | ||||||
|  | 	libraryjars(files(shrinkClasspath.files)) | ||||||
|  | 	outjars("build/libs/" + tasks.shadowJar.get().outputs.files.first().name.removeSuffix(".jar") + "-shrink.jar") | ||||||
|  | 	if (System.getProperty("java.version").startsWith("1.")) { | ||||||
|  | 		libraryjars("${System.getProperty("java.home")}/lib/rt.jar") | ||||||
|  | 		libraryjars("${System.getProperty("java.home")}/lib/jce.jar") | ||||||
|  | 	} else { | ||||||
|  | 		throw RuntimeException("Compiling with Java 9+ not supported!") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	keep("class link.infra.packwiz.installer.** { *; }") | ||||||
|  | 	dontoptimize() | ||||||
|  | 	dontobfuscate() | ||||||
|  | 	dontwarn("org.codehaus.mojo.animal_sniffer.*") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Used for vscode launch.json | ||||||
|  | tasks.register<Copy>("copyJar") { | ||||||
|  | 	from(tasks.named("shrinkJar")) | ||||||
|  | 	rename("packwiz-installer-(.*)\\.jar", "packwiz-installer.jar") | ||||||
|  | 	into("build/libs/") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks.build { | ||||||
|  | 	dependsOn("copyJar") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | if (project.hasProperty("github.token")) { | ||||||
|  | 	githubRelease { | ||||||
|  | 		owner("comp500") | ||||||
|  | 		repo("packwiz-installer") | ||||||
|  | 		tagName("${project.version}") | ||||||
|  | 		releaseName("Release ${project.version}") | ||||||
|  | 		draft(true) | ||||||
|  | 		token(findProperty("github.token") as String? ?: "") | ||||||
|  | 		releaseAssets(tasks.jar.get().destinationDirectory.file("packwiz-installer.jar").get()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tasks.githubRelease { | ||||||
|  | 		dependsOn(tasks.build) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks.compileKotlin { | ||||||
|  | 	kotlinOptions { | ||||||
|  | 		jvmTarget = "1.8" | ||||||
|  | 		freeCompilerArgs = listOf("-Xjvm-default=enable") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | tasks.compileTestKotlin { | ||||||
|  | 	kotlinOptions { | ||||||
|  | 		jvmTarget = "1.8" | ||||||
|  | 		freeCompilerArgs = listOf("-Xjvm-default=enable") | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-bin.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-6.7.1-bin.zip | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
|   | |||||||
| @@ -1,244 +0,0 @@ | |||||||
| package link.infra.packwiz.installer; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.IndexFile; |  | ||||||
| import link.infra.packwiz.installer.metadata.ManifestFile; |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.GeneralHashingSource; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.Hash; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.HashUtils; |  | ||||||
| import link.infra.packwiz.installer.ui.IExceptionDetails; |  | ||||||
| import link.infra.packwiz.installer.ui.IOptionDetails; |  | ||||||
| import okio.Buffer; |  | ||||||
| import okio.Okio; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.nio.file.Files; |  | ||||||
| import java.nio.file.Path; |  | ||||||
| import java.nio.file.Paths; |  | ||||||
| import java.nio.file.StandardCopyOption; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| class DownloadTask implements IOptionDetails, IExceptionDetails { |  | ||||||
| 	final IndexFile.File metadata; |  | ||||||
| 	ManifestFile.File cachedFile = null; |  | ||||||
| 	private Exception failure = null; |  | ||||||
| 	private boolean alreadyUpToDate = false; |  | ||||||
| 	private boolean metadataRequired = true; |  | ||||||
| 	private boolean invalidated = false; |  | ||||||
| 	// If file is new or isOptional changed to true, the option needs to be presented again |  | ||||||
| 	private boolean newOptional = true; |  | ||||||
| 	private final UpdateManager.Options.Side downloadSide; |  | ||||||
|  |  | ||||||
| 	private DownloadTask(IndexFile.File metadata, String defaultFormat, UpdateManager.Options.Side downloadSide) { |  | ||||||
| 		this.metadata = metadata; |  | ||||||
| 		if (metadata.hashFormat == null || metadata.hashFormat.length() == 0) { |  | ||||||
| 			metadata.hashFormat = defaultFormat; |  | ||||||
| 		} |  | ||||||
| 		this.downloadSide = downloadSide; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void invalidate() { |  | ||||||
| 		invalidated = true; |  | ||||||
| 		alreadyUpToDate = false; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void updateFromCache(ManifestFile.File cachedFile) { |  | ||||||
| 		if (failure != null) return; |  | ||||||
| 		if (cachedFile == null) { |  | ||||||
| 			this.cachedFile = new ManifestFile.File(); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		this.cachedFile = cachedFile; |  | ||||||
|  |  | ||||||
| 		if (!invalidated) { |  | ||||||
| 			Hash currHash; |  | ||||||
| 			try { |  | ||||||
| 				currHash = HashUtils.getHash(metadata.hashFormat, metadata.hash); |  | ||||||
| 			} catch (Exception e) { |  | ||||||
| 				failure = e; |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			if (currHash != null && currHash.equals(cachedFile.hash)) { |  | ||||||
| 				// Already up to date |  | ||||||
| 				alreadyUpToDate = true; |  | ||||||
| 				metadataRequired = false; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if (cachedFile.isOptional) { |  | ||||||
| 			// Because option selection dialog might set this task to true/false, metadata is always needed to download |  | ||||||
| 			// the file, and to show the description and name |  | ||||||
| 			metadataRequired = true; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void downloadMetadata(IndexFile parentIndexFile, SpaceSafeURI indexUri) { |  | ||||||
| 		if (failure != null) return; |  | ||||||
| 		if (metadataRequired) { |  | ||||||
| 			try { |  | ||||||
| 				metadata.downloadMeta(parentIndexFile, indexUri); |  | ||||||
| 			} catch (Exception e) { |  | ||||||
| 				failure = e; |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			if (metadata.linkedFile != null) { |  | ||||||
| 				if (metadata.linkedFile.option != null) { |  | ||||||
| 					if (metadata.linkedFile.option.optional) { |  | ||||||
| 						if (cachedFile.isOptional) { |  | ||||||
| 							// isOptional didn't change |  | ||||||
| 							newOptional = false; |  | ||||||
| 						} else { |  | ||||||
| 							// isOptional false -> true, set option to it's default value |  | ||||||
| 							// TODO: preserve previous option value, somehow?? |  | ||||||
| 							cachedFile.optionValue = this.metadata.linkedFile.option.defaultValue; |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				cachedFile.isOptional = isOptional(); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void download(String packFolder, SpaceSafeURI indexUri) { |  | ||||||
| 		if (failure != null) return; |  | ||||||
|  |  | ||||||
| 		// Ensure it is removed |  | ||||||
| 		if (!cachedFile.optionValue || !correctSide()) { |  | ||||||
| 			if (cachedFile.cachedLocation == null) return; |  | ||||||
| 			try { |  | ||||||
| 				Files.deleteIfExists(Paths.get(packFolder, cachedFile.cachedLocation)); |  | ||||||
| 			} catch (IOException e) { |  | ||||||
| 				// TODO: how much of a problem is this? use log4j/other log library to show warning? |  | ||||||
| 				e.printStackTrace(); |  | ||||||
| 			} |  | ||||||
| 			cachedFile.cachedLocation = null; |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (alreadyUpToDate) return; |  | ||||||
|  |  | ||||||
| 		Path destPath = Paths.get(packFolder, metadata.getDestURI().toString()); |  | ||||||
|  |  | ||||||
| 		// Don't update files marked with preserve if they already exist on disk |  | ||||||
| 		if (metadata.preserve) { |  | ||||||
| 			if (Files.exists(destPath)) { |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		try { |  | ||||||
| 			Hash hash; |  | ||||||
| 			String fileHashFormat; |  | ||||||
| 			if (metadata.linkedFile != null) { |  | ||||||
| 				hash = metadata.linkedFile.getHash(); |  | ||||||
| 				fileHashFormat = metadata.linkedFile.download.hashFormat; |  | ||||||
| 			} else { |  | ||||||
| 				hash = metadata.getHash(); |  | ||||||
| 				fileHashFormat = metadata.hashFormat; |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			Source src = metadata.getSource(indexUri); |  | ||||||
| 			GeneralHashingSource fileSource = HashUtils.getHasher(fileHashFormat).getHashingSource(src); |  | ||||||
| 			Buffer data = new Buffer(); |  | ||||||
| 			Okio.buffer(fileSource).readAll(data); |  | ||||||
|  |  | ||||||
| 			if (fileSource.hashIsEqual(hash)) { |  | ||||||
| 				Files.createDirectories(destPath.getParent()); |  | ||||||
| 				Files.copy(data.inputStream(), destPath, StandardCopyOption.REPLACE_EXISTING); |  | ||||||
| 			} else { |  | ||||||
| 				// TODO: no more SYSOUT!!!!!!!!! |  | ||||||
| 				System.out.println("Invalid hash for " + metadata.getDestURI().toString()); |  | ||||||
| 				System.out.println("Calculated: " + fileSource.getHash()); |  | ||||||
| 				System.out.println("Expected:   " + hash); |  | ||||||
| 				failure = new Exception("Hash invalid!"); |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if (cachedFile.cachedLocation != null && !destPath.equals(Paths.get(packFolder, cachedFile.cachedLocation))) { |  | ||||||
| 				// Delete old file if location changes |  | ||||||
| 				Files.delete(Paths.get(packFolder, cachedFile.cachedLocation)); |  | ||||||
| 			} |  | ||||||
| 		} catch (Exception e) { |  | ||||||
| 			failure = e; |  | ||||||
| 		} |  | ||||||
| 		if (failure == null) { |  | ||||||
| 			if (cachedFile == null) { |  | ||||||
| 				cachedFile = new ManifestFile.File(); |  | ||||||
| 			} |  | ||||||
| 			// Update the manifest file |  | ||||||
| 			try { |  | ||||||
| 				cachedFile.hash = metadata.getHash(); |  | ||||||
| 			} catch (Exception e) { |  | ||||||
| 				failure = e; |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			cachedFile.isOptional = isOptional(); |  | ||||||
| 			cachedFile.cachedLocation = metadata.getDestURI().toString(); |  | ||||||
| 			if (metadata.linkedFile != null) { |  | ||||||
| 				try { |  | ||||||
| 					cachedFile.linkedFileHash = metadata.linkedFile.getHash(); |  | ||||||
| 				} catch (Exception e) { |  | ||||||
| 					failure = e; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public Exception getException() { |  | ||||||
| 		return failure; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	boolean isOptional() { |  | ||||||
| 		if (metadata.linkedFile != null) { |  | ||||||
| 			return metadata.linkedFile.isOptional(); |  | ||||||
| 		} |  | ||||||
| 		return false; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	boolean isNewOptional() { |  | ||||||
| 		return isOptional() && this.newOptional; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	boolean correctSide() { |  | ||||||
| 		if (metadata.linkedFile != null) { |  | ||||||
| 			return metadata.linkedFile.side.hasSide(downloadSide); |  | ||||||
| 		} |  | ||||||
| 		return true; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getName() { |  | ||||||
| 		return metadata.getName(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public boolean getOptionValue() { |  | ||||||
| 		return cachedFile.optionValue; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public String getOptionDescription() { |  | ||||||
| 		if (metadata.linkedFile != null) { |  | ||||||
| 			return metadata.linkedFile.option.description; |  | ||||||
| 		} |  | ||||||
| 		return null; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public void setOptionValue(boolean value) { |  | ||||||
| 		if (value && !cachedFile.optionValue) { |  | ||||||
| 			// Ensure that an update is done if it changes from false to true, or from true to false |  | ||||||
| 			alreadyUpToDate = false; |  | ||||||
| 		} |  | ||||||
| 		cachedFile.optionValue = value; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	static List<DownloadTask> createTasksFromIndex(IndexFile index, String defaultFormat, UpdateManager.Options.Side downloadSide) { |  | ||||||
| 		ArrayList<DownloadTask> tasks = new ArrayList<>(); |  | ||||||
| 		for (IndexFile.File file : index.files) { |  | ||||||
| 			tasks.add(new DownloadTask(file, defaultFormat, downloadSide)); |  | ||||||
| 		} |  | ||||||
| 		return tasks; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,148 +0,0 @@ | |||||||
| package link.infra.packwiz.installer; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import link.infra.packwiz.installer.ui.CLIHandler; |  | ||||||
| import link.infra.packwiz.installer.ui.IUserInterface; |  | ||||||
| import link.infra.packwiz.installer.ui.InputStateHandler; |  | ||||||
| import link.infra.packwiz.installer.ui.InstallWindow; |  | ||||||
| import org.apache.commons.cli.*; |  | ||||||
|  |  | ||||||
| import javax.swing.*; |  | ||||||
| import java.awt.*; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") |  | ||||||
| public class Main { |  | ||||||
|  |  | ||||||
| 	// Actual main() is in RequiresBootstrap! |  | ||||||
| 	@SuppressWarnings("unused") |  | ||||||
| 	public Main(String[] args) { |  | ||||||
| 		// Big overarching try/catch just in case everything breaks |  | ||||||
| 		try { |  | ||||||
| 			this.startup(args); |  | ||||||
| 		} catch (Exception e) { |  | ||||||
| 			e.printStackTrace(); |  | ||||||
| 			EventQueue.invokeLater(() -> { |  | ||||||
| 				JOptionPane.showMessageDialog(null, |  | ||||||
| 						"A fatal error occurred: \n" + e.getClass().getCanonicalName() + ": " + e.getMessage(), |  | ||||||
| 						"packwiz-installer", JOptionPane.ERROR_MESSAGE); |  | ||||||
| 				System.exit(1); |  | ||||||
| 			}); |  | ||||||
| 			// In case the eventqueue is broken, exit after 1 minute |  | ||||||
| 			try { |  | ||||||
| 				Thread.sleep(60 * 1000); |  | ||||||
| 			} catch (InterruptedException e1) { |  | ||||||
| 				// Good, it was already called? |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			System.exit(1); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void startup(String[] args) { |  | ||||||
| 		Options options = new Options(); |  | ||||||
| 		addNonBootstrapOptions(options); |  | ||||||
| 		addBootstrapOptions(options); |  | ||||||
|  |  | ||||||
| 		CommandLineParser parser = new DefaultParser(); |  | ||||||
| 		CommandLine cmd = null; |  | ||||||
| 		try { |  | ||||||
| 			cmd = parser.parse(options, args); |  | ||||||
| 		} catch (ParseException e) { |  | ||||||
| 			e.printStackTrace(); |  | ||||||
| 			try { |  | ||||||
| 				UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); |  | ||||||
| 			} catch (Exception e1) { |  | ||||||
| 				// Ignore the exceptions, just continue using the ugly L&F |  | ||||||
| 			} |  | ||||||
| 			JOptionPane.showMessageDialog(null, e.getMessage(), "packwiz-installer", JOptionPane.ERROR_MESSAGE); |  | ||||||
| 			System.exit(1); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		IUserInterface ui; |  | ||||||
| 		// if "headless", GUI creation will fail anyway! |  | ||||||
| 		if (cmd.hasOption("no-gui") || GraphicsEnvironment.isHeadless()) { |  | ||||||
| 			ui = new CLIHandler(); |  | ||||||
| 		} else { |  | ||||||
| 			ui = new InstallWindow(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		String[] unparsedArgs = cmd.getArgs(); |  | ||||||
| 		if (unparsedArgs.length > 1) { |  | ||||||
| 			ui.handleExceptionAndExit(new RuntimeException("Too many arguments specified!")); |  | ||||||
| 			return; |  | ||||||
| 		} else if (unparsedArgs.length < 1) { |  | ||||||
| 			ui.handleExceptionAndExit(new RuntimeException("URI to install from must be specified!")); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		String title = cmd.getOptionValue("title"); |  | ||||||
| 		if (title != null) { |  | ||||||
| 			ui.setTitle(title); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		InputStateHandler inputStateHandler = new InputStateHandler(); |  | ||||||
| 		ui.show(inputStateHandler); |  | ||||||
|  |  | ||||||
| 		UpdateManager.Options uOptions = new UpdateManager.Options(); |  | ||||||
|  |  | ||||||
| 		String side = cmd.getOptionValue("side"); |  | ||||||
| 		if (side != null) { |  | ||||||
| 			uOptions.side = UpdateManager.Options.Side.from(side); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		String packFolder = cmd.getOptionValue("pack-folder"); |  | ||||||
| 		if (packFolder != null) { |  | ||||||
| 			uOptions.packFolder = packFolder; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		String metaFile = cmd.getOptionValue("meta-file"); |  | ||||||
| 		if (metaFile != null) { |  | ||||||
| 			uOptions.manifestFile = metaFile; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		try { |  | ||||||
| 			uOptions.downloadURI = new SpaceSafeURI(unparsedArgs[0]); |  | ||||||
| 		} catch (URISyntaxException e) { |  | ||||||
| 			// TODO: better error message? |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Start update process! |  | ||||||
| 		// TODO: start in SwingWorker? |  | ||||||
| 		try { |  | ||||||
| 			ui.executeManager(() -> { |  | ||||||
| 				try { |  | ||||||
| 					new UpdateManager(uOptions, ui, inputStateHandler); |  | ||||||
| 				} catch (Exception e) { |  | ||||||
| 					// TODO: better error message? |  | ||||||
| 					ui.handleExceptionAndExit(e); |  | ||||||
| 				} |  | ||||||
| 			}); |  | ||||||
| 		} catch (Exception e) { |  | ||||||
| 			// TODO: better error message? |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Called by packwiz-installer-bootstrap to set up the help command |  | ||||||
| 	@SuppressWarnings("WeakerAccess") |  | ||||||
| 	public static void addNonBootstrapOptions(Options options) { |  | ||||||
| 		options.addOption("s", "side", true, "Side to install mods from (client/server, defaults to client)"); |  | ||||||
| 		options.addOption(null, "title", true, "Title of the installer window"); |  | ||||||
| 		options.addOption(null, "pack-folder", true, "Folder to install the pack to (defaults to the JAR directory)"); |  | ||||||
| 		options.addOption(null, "meta-file", true, "JSON file to store pack metadata, relative to the pack folder (defaults to packwiz.json)"); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	// TODO: link these somehow so they're only defined once? |  | ||||||
| 	private static void addBootstrapOptions(Options options) { |  | ||||||
| 		options.addOption(null, "bootstrap-update-url", true, "Github API URL for checking for updates"); |  | ||||||
| 		options.addOption(null, "bootstrap-update-token", true, "Github API Access Token, for private repositories"); |  | ||||||
| 		options.addOption(null, "bootstrap-no-update", false, "Don't update packwiz-installer"); |  | ||||||
| 		options.addOption(null, "bootstrap-main-jar", true, "Location of the packwiz-installer JAR file"); |  | ||||||
| 		options.addOption("g", "no-gui", false, "Don't display a GUI to show update progress"); |  | ||||||
| 		options.addOption("h", "help", false, "Display this message"); // Implemented in packwiz-installer-bootstrap! |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| } |  | ||||||
| @@ -1,479 +0,0 @@ | |||||||
| package link.infra.packwiz.installer; |  | ||||||
|  |  | ||||||
| import com.google.gson.Gson; |  | ||||||
| import com.google.gson.GsonBuilder; |  | ||||||
| import com.google.gson.JsonIOException; |  | ||||||
| import com.google.gson.JsonSyntaxException; |  | ||||||
| import com.google.gson.annotations.SerializedName; |  | ||||||
| import com.moandjiezana.toml.Toml; |  | ||||||
| import link.infra.packwiz.installer.metadata.IndexFile; |  | ||||||
| import link.infra.packwiz.installer.metadata.ManifestFile; |  | ||||||
| import link.infra.packwiz.installer.metadata.PackFile; |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.GeneralHashingSource; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.Hash; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.HashUtils; |  | ||||||
| import link.infra.packwiz.installer.request.HandlerManager; |  | ||||||
| import link.infra.packwiz.installer.ui.IExceptionDetails; |  | ||||||
| import link.infra.packwiz.installer.ui.IUserInterface; |  | ||||||
| import link.infra.packwiz.installer.ui.InputStateHandler; |  | ||||||
| import link.infra.packwiz.installer.ui.InstallProgress; |  | ||||||
| import okio.Okio; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.io.*; |  | ||||||
| import java.nio.file.Files; |  | ||||||
| import java.nio.file.Paths; |  | ||||||
| import java.util.*; |  | ||||||
| import java.util.concurrent.*; |  | ||||||
| import java.util.stream.Collectors; |  | ||||||
|  |  | ||||||
| public class UpdateManager { |  | ||||||
|  |  | ||||||
| 	private final Options opts; |  | ||||||
| 	public final IUserInterface ui; |  | ||||||
| 	private boolean cancelled; |  | ||||||
| 	private boolean cancelledStartGame = false; |  | ||||||
| 	private InputStateHandler stateHandler; |  | ||||||
|  |  | ||||||
| 	public static class Options { |  | ||||||
| 		SpaceSafeURI downloadURI = null; |  | ||||||
| 		String manifestFile = "packwiz.json"; // TODO: make configurable |  | ||||||
| 		String packFolder = "."; |  | ||||||
| 		Side side = Side.CLIENT; |  | ||||||
|  |  | ||||||
| 		public enum Side { |  | ||||||
| 			@SerializedName("client") |  | ||||||
| 			CLIENT("client"), |  | ||||||
| 			@SerializedName("server") |  | ||||||
| 			SERVER("server"), |  | ||||||
| 			@SerializedName("both") |  | ||||||
| 			BOTH("both", new Side[] { CLIENT, SERVER }); |  | ||||||
|  |  | ||||||
| 			private final String sideName; |  | ||||||
| 			private final Side[] depSides; |  | ||||||
|  |  | ||||||
| 			Side(String sideName) { |  | ||||||
| 				this.sideName = sideName.toLowerCase(); |  | ||||||
| 				this.depSides = null; |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			Side(String sideName, Side[] depSides) { |  | ||||||
| 				this.sideName = sideName.toLowerCase(); |  | ||||||
| 				this.depSides = depSides; |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			@Override |  | ||||||
| 			public String toString() { |  | ||||||
| 				return this.sideName; |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			public boolean hasSide(Side tSide) { |  | ||||||
| 				if (this.equals(tSide)) { |  | ||||||
| 					return true; |  | ||||||
| 				} |  | ||||||
| 				if (this.depSides != null) { |  | ||||||
| 					for (Side depSide : this.depSides) { |  | ||||||
| 						if (depSide.equals(tSide)) { |  | ||||||
| 							return true; |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				return false; |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			public static Side from(String name) { |  | ||||||
| 				String lowerName = name.toLowerCase(); |  | ||||||
| 				for (Side side : Side.values()) { |  | ||||||
| 					if (side.sideName.equals(lowerName)) { |  | ||||||
| 						return side; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				return null; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	UpdateManager(Options opts, IUserInterface ui, InputStateHandler inputStateHandler) { |  | ||||||
| 		this.opts = opts; |  | ||||||
| 		this.ui = ui; |  | ||||||
| 		this.stateHandler = inputStateHandler; |  | ||||||
| 		this.start(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void start() { |  | ||||||
| 		this.checkOptions(); |  | ||||||
|  |  | ||||||
| 		ui.submitProgress(new InstallProgress("Loading manifest file...")); |  | ||||||
| 		Gson gson = new GsonBuilder().registerTypeAdapter(Hash.class, new Hash.TypeHandler()).create(); |  | ||||||
| 		ManifestFile manifest; |  | ||||||
| 		try { |  | ||||||
| 			manifest = gson.fromJson(new FileReader(Paths.get(opts.packFolder, opts.manifestFile).toString()), |  | ||||||
| 					ManifestFile.class); |  | ||||||
| 		} catch (FileNotFoundException e) { |  | ||||||
| 			manifest = new ManifestFile(); |  | ||||||
| 		} catch (JsonSyntaxException | JsonIOException e) { |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			handleCancellation(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		ui.submitProgress(new InstallProgress("Loading pack file...")); |  | ||||||
| 		GeneralHashingSource packFileSource; |  | ||||||
| 		try { |  | ||||||
| 			Source src = HandlerManager.getFileSource(opts.downloadURI); |  | ||||||
| 			packFileSource = HashUtils.getHasher("sha256").getHashingSource(src); |  | ||||||
| 		} catch (Exception e) { |  | ||||||
| 			// TODO: still launch the game if updating doesn't work? |  | ||||||
| 			// TODO: ask user if they want to launch the game, exit(1) if they don't |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		PackFile pf; |  | ||||||
| 		try { |  | ||||||
| 			pf = new Toml().read(Okio.buffer(packFileSource).inputStream()).to(PackFile.class); |  | ||||||
| 		} catch (IllegalStateException e) { |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			handleCancellation(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		ui.submitProgress(new InstallProgress("Checking local files...")); |  | ||||||
|  |  | ||||||
| 		// Invalidation checking must be done here, as it must happen before pack/index hashes are checked |  | ||||||
| 		List<SpaceSafeURI> invalidatedUris = new ArrayList<>(); |  | ||||||
| 		if (manifest.cachedFiles != null) { |  | ||||||
| 			for (Map.Entry<SpaceSafeURI, ManifestFile.File> entry : manifest.cachedFiles.entrySet()) { |  | ||||||
| 				boolean invalid = false; |  | ||||||
| 				// if isn't optional, or is optional but optionValue == true |  | ||||||
| 				if (!entry.getValue().isOptional || entry.getValue().optionValue) { |  | ||||||
| 					if (entry.getValue().cachedLocation != null) { |  | ||||||
| 						if (!Files.exists(Paths.get(opts.packFolder, entry.getValue().cachedLocation))) { |  | ||||||
| 							invalid = true; |  | ||||||
| 						} |  | ||||||
| 					} else { |  | ||||||
| 						// if cachedLocation == null, should probably be installed!! |  | ||||||
| 						invalid = true; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				if (invalid) { |  | ||||||
| 					SpaceSafeURI fileUri = entry.getKey(); |  | ||||||
| 					System.out.println("File " + fileUri.toString() + " invalidated, marked for redownloading"); |  | ||||||
| 					invalidatedUris.add(fileUri); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (manifest.packFileHash != null && packFileSource.hashIsEqual(manifest.packFileHash) && invalidatedUris.isEmpty()) { |  | ||||||
| 			System.out.println("Modpack is already up to date!"); |  | ||||||
| 			// todo: --force? |  | ||||||
| 			if (!stateHandler.getOptionsButton()) { |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		System.out.println("Modpack name: " + pf.name); |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			handleCancellation(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		try { |  | ||||||
| 			// This is badly written, I'll probably heavily refactor it at some point |  | ||||||
| 			processIndex(HandlerManager.getNewLoc(opts.downloadURI, pf.index.file), |  | ||||||
| 					HashUtils.getHash(pf.index.hashFormat, pf.index.hash), pf.index.hashFormat, manifest, invalidatedUris); |  | ||||||
| 		} catch (Exception e1) { |  | ||||||
| 			ui.handleExceptionAndExit(e1); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		handleCancellation(); |  | ||||||
|  |  | ||||||
| 		// TODO: update MMC params, java args etc |  | ||||||
|  |  | ||||||
| 		manifest.packFileHash = packFileSource.getHash(); |  | ||||||
| 		manifest.cachedSide = opts.side; |  | ||||||
| 		try (Writer writer = new FileWriter(Paths.get(opts.packFolder, opts.manifestFile).toString())) { |  | ||||||
| 			gson.toJson(manifest, writer); |  | ||||||
| 		} catch (IOException e) { |  | ||||||
| 			// TODO: add message? |  | ||||||
| 			ui.handleException(e); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void checkOptions() { |  | ||||||
| 		// TODO: implement |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void processIndex(SpaceSafeURI indexUri, Hash indexHash, String hashFormat, ManifestFile manifest, List<SpaceSafeURI> invalidatedUris) { |  | ||||||
| 		if (manifest.indexFileHash != null && manifest.indexFileHash.equals(indexHash) && invalidatedUris.isEmpty()) { |  | ||||||
| 			System.out.println("Modpack files are already up to date!"); |  | ||||||
| 			if (!stateHandler.getOptionsButton()) { |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		manifest.indexFileHash = indexHash; |  | ||||||
|  |  | ||||||
| 		GeneralHashingSource indexFileSource; |  | ||||||
| 		try { |  | ||||||
| 			Source src = HandlerManager.getFileSource(indexUri); |  | ||||||
| 			indexFileSource = HashUtils.getHasher(hashFormat).getHashingSource(src); |  | ||||||
| 		} catch (Exception e) { |  | ||||||
| 			// TODO: still launch the game if updating doesn't work? |  | ||||||
| 			// TODO: ask user if they want to launch the game, exit(1) if they don't |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		IndexFile indexFile; |  | ||||||
| 		try { |  | ||||||
| 			indexFile = new Toml().read(Okio.buffer(indexFileSource).inputStream()).to(IndexFile.class); |  | ||||||
| 		} catch (IllegalStateException e) { |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (!indexFileSource.hashIsEqual(indexHash)) { |  | ||||||
| 			// TODO: throw exception |  | ||||||
| 			System.out.println("I was meant to put an error message here but I'll do that later"); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (manifest.cachedFiles == null) { |  | ||||||
| 			manifest.cachedFiles = new HashMap<>(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		ui.submitProgress(new InstallProgress("Checking local files...")); |  | ||||||
| 		Iterator<Map.Entry<SpaceSafeURI, ManifestFile.File>> it = manifest.cachedFiles.entrySet().iterator(); |  | ||||||
| 		while (it.hasNext()) { |  | ||||||
| 			Map.Entry<SpaceSafeURI, ManifestFile.File> entry = it.next(); |  | ||||||
| 			if (entry.getValue().cachedLocation != null) { |  | ||||||
| 				boolean alreadyDeleted = false; |  | ||||||
| 				// Delete if option value has been set to false |  | ||||||
| 				if (entry.getValue().isOptional && !entry.getValue().optionValue) { |  | ||||||
| 					try { |  | ||||||
| 						Files.deleteIfExists(Paths.get(opts.packFolder, entry.getValue().cachedLocation)); |  | ||||||
| 					} catch (IOException e) { |  | ||||||
| 						// TODO: should this be shown to the user in some way? |  | ||||||
| 						e.printStackTrace(); |  | ||||||
| 					} |  | ||||||
| 					// Set to null, as it doesn't exist anymore |  | ||||||
| 					entry.getValue().cachedLocation = null; |  | ||||||
| 					alreadyDeleted = true; |  | ||||||
| 				} |  | ||||||
| 				if (indexFile.files.stream().noneMatch(f -> f.file.equals(entry.getKey()))) { |  | ||||||
| 					// File has been removed from the index |  | ||||||
| 					if (!alreadyDeleted) { |  | ||||||
| 						try { |  | ||||||
| 							Files.deleteIfExists(Paths.get(opts.packFolder, entry.getValue().cachedLocation)); |  | ||||||
| 						} catch (IOException e) { |  | ||||||
| 							// TODO: should this be shown to the user in some way? |  | ||||||
| 							e.printStackTrace(); |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 					it.remove(); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		ui.submitProgress(new InstallProgress("Comparing new files...")); |  | ||||||
|  |  | ||||||
| 		// TODO: progress bar? |  | ||||||
| 		List<DownloadTask> tasks = DownloadTask.createTasksFromIndex(indexFile, indexFile.hashFormat, opts.side); |  | ||||||
| 		// If the side changes, invalidate EVERYTHING just in case |  | ||||||
| 		// Might not be needed, but done just to be safe |  | ||||||
| 		boolean invalidateAll = !opts.side.equals(manifest.cachedSide); |  | ||||||
| 		if (invalidateAll) { |  | ||||||
| 			System.out.println("Side changed, invalidating all mods"); |  | ||||||
| 		} |  | ||||||
| 		tasks.forEach(f -> { |  | ||||||
| 			// TODO: should linkedfile be checked as well? should this be done in the download section? |  | ||||||
| 			if (invalidateAll) { |  | ||||||
| 				f.invalidate(); |  | ||||||
| 			} else if (invalidatedUris.contains(f.metadata.file)) { |  | ||||||
| 				f.invalidate(); |  | ||||||
| 			} |  | ||||||
| 			ManifestFile.File file = manifest.cachedFiles.get(f.metadata.file); |  | ||||||
| 			if (file != null) { |  | ||||||
| 				// Ensure the file can be reverted later if necessary - the DownloadTask modifies the file so if it fails we need the old version back |  | ||||||
| 				file.backup(); |  | ||||||
| 			} |  | ||||||
| 			// If it is null, the DownloadTask will make a new empty cachedFile |  | ||||||
| 			f.updateFromCache(file); |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Let's hope downloadMetadata is a pure function!!! |  | ||||||
| 		tasks.parallelStream().forEach(f -> f.downloadMetadata(indexFile, indexUri)); |  | ||||||
|  |  | ||||||
| 		List<IExceptionDetails> failedTasks = tasks.stream().filter(t -> t.getException() != null).collect(Collectors.toList()); |  | ||||||
| 		if (failedTasks.size() > 0) { |  | ||||||
| 			IExceptionDetails.ExceptionListResult exceptionListResult; |  | ||||||
| 			try { |  | ||||||
| 				exceptionListResult = ui.showExceptions(failedTasks, tasks.size(), true).get(); |  | ||||||
| 			} catch (InterruptedException | ExecutionException e) { |  | ||||||
| 				// Interrupted means cancelled??? |  | ||||||
| 				ui.handleExceptionAndExit(e); |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			switch (exceptionListResult) { |  | ||||||
| 				case CONTINUE: |  | ||||||
| 					break; |  | ||||||
| 				case CANCEL: |  | ||||||
| 					cancelled = true; |  | ||||||
| 					return; |  | ||||||
| 				case IGNORE: |  | ||||||
| 					cancelledStartGame = true; |  | ||||||
| 					return; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (stateHandler.getCancelButton()) { |  | ||||||
| 			showCancellationDialog(); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		List<DownloadTask> nonFailedFirstTasks = tasks.stream().filter(t -> t.getException() == null).collect(Collectors.toList()); |  | ||||||
| 		List<DownloadTask> optionTasks = nonFailedFirstTasks.stream().filter(DownloadTask::correctSide).filter(DownloadTask::isOptional).collect(Collectors.toList()); |  | ||||||
| 		// If options changed, present all options again |  | ||||||
| 		if (stateHandler.getOptionsButton() || optionTasks.stream().anyMatch(DownloadTask::isNewOptional)) { |  | ||||||
| 			// new ArrayList is requires so it's an IOptionDetails rather than a DownloadTask list |  | ||||||
| 			Future<Boolean> cancelledResult = ui.showOptions(new ArrayList<>(optionTasks)); |  | ||||||
| 			try { |  | ||||||
| 				if (cancelledResult.get()) { |  | ||||||
| 					cancelled = true; |  | ||||||
| 					// TODO: Should the UI be closed somehow?? |  | ||||||
| 					return; |  | ||||||
| 				} |  | ||||||
| 			} catch (InterruptedException | ExecutionException e) { |  | ||||||
| 				// Interrupted means cancelled??? |  | ||||||
| 				ui.handleExceptionAndExit(e); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		ui.disableOptionsButton(); |  | ||||||
|  |  | ||||||
| 		// TODO: different thread pool type? |  | ||||||
| 		ExecutorService threadPool = Executors.newFixedThreadPool(10); |  | ||||||
| 		CompletionService<DownloadTask> completionService = new ExecutorCompletionService<>(threadPool); |  | ||||||
|  |  | ||||||
| 		tasks.forEach(t -> completionService.submit(() -> { |  | ||||||
| 			t.download(opts.packFolder, indexUri); |  | ||||||
| 			return t; |  | ||||||
| 		})); |  | ||||||
|  |  | ||||||
| 		for (int i = 0; i < tasks.size(); i++) { |  | ||||||
| 			DownloadTask task; |  | ||||||
| 			try { |  | ||||||
| 				task = completionService.take().get(); |  | ||||||
| 			} catch (InterruptedException | ExecutionException e) { |  | ||||||
| 				ui.handleException(e); |  | ||||||
| 				task = null; |  | ||||||
| 			} |  | ||||||
| 			// Update manifest - If there were no errors cachedFile has already been modified in place (good old pass by reference) |  | ||||||
| 			if (task != null) { |  | ||||||
| 				if (task.getException() != null) { |  | ||||||
| 					ManifestFile.File file = task.cachedFile.getRevert(); |  | ||||||
| 					if (file != null) { |  | ||||||
| 						manifest.cachedFiles.putIfAbsent(task.metadata.file, file); |  | ||||||
| 					} |  | ||||||
| 				} else { |  | ||||||
| 					// idiot, if it wasn't there in the first place it won't magically appear there |  | ||||||
| 					manifest.cachedFiles.putIfAbsent(task.metadata.file, task.cachedFile); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			String progress; |  | ||||||
| 			if (task != null) { |  | ||||||
| 				if (task.getException() != null) { |  | ||||||
| 					progress = "Failed to download " + task.metadata.getName() + ": " + task.getException().getMessage(); |  | ||||||
| 					task.getException().printStackTrace(); |  | ||||||
| 				} else { |  | ||||||
| 					// TODO: should this be revised for tasks that didn't actually download it? |  | ||||||
| 					progress = "Downloaded " + task.metadata.getName(); |  | ||||||
| 				} |  | ||||||
| 			} else { |  | ||||||
| 				progress = "Failed to download, unknown reason"; |  | ||||||
| 			} |  | ||||||
| 			ui.submitProgress(new InstallProgress(progress, i + 1, tasks.size())); |  | ||||||
|  |  | ||||||
| 			if (stateHandler.getCancelButton()) { |  | ||||||
| 				// Stop all tasks, don't launch the game (it's in an invalid state!) |  | ||||||
| 				threadPool.shutdown(); |  | ||||||
| 				cancelled = true; |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		List<IExceptionDetails> failedTasks2ElectricBoogaloo = nonFailedFirstTasks.stream().filter(t -> t.getException() != null).collect(Collectors.toList()); |  | ||||||
| 		if (failedTasks2ElectricBoogaloo.size() > 0) { |  | ||||||
| 			IExceptionDetails.ExceptionListResult exceptionListResult; |  | ||||||
| 			try { |  | ||||||
| 				exceptionListResult = ui.showExceptions(failedTasks2ElectricBoogaloo, tasks.size(), false).get(); |  | ||||||
| 			} catch (InterruptedException | ExecutionException e) { |  | ||||||
| 				// Interrupted means cancelled??? |  | ||||||
| 				ui.handleExceptionAndExit(e); |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			switch (exceptionListResult) { |  | ||||||
| 				case CONTINUE: |  | ||||||
| 					break; |  | ||||||
| 				case CANCEL: |  | ||||||
| 					cancelled = true; |  | ||||||
| 					return; |  | ||||||
| 				case IGNORE: |  | ||||||
| 					cancelledStartGame = true; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void showCancellationDialog() { |  | ||||||
| 		IExceptionDetails.ExceptionListResult exceptionListResult; |  | ||||||
| 		try { |  | ||||||
| 			exceptionListResult = ui.showCancellationDialog().get(); |  | ||||||
| 		} catch (InterruptedException | ExecutionException e) { |  | ||||||
| 			// Interrupted means cancelled??? |  | ||||||
| 			ui.handleExceptionAndExit(e); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		switch (exceptionListResult) { |  | ||||||
| 			case CONTINUE: |  | ||||||
| 				throw new RuntimeException("Continuation not allowed here!"); |  | ||||||
| 			case CANCEL: |  | ||||||
| 				cancelled = true; |  | ||||||
| 				return; |  | ||||||
| 			case IGNORE: |  | ||||||
| 				cancelledStartGame = true; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private void handleCancellation() { |  | ||||||
| 		if (cancelled) { |  | ||||||
| 			System.out.println("Update cancelled by user!"); |  | ||||||
| 			System.exit(1); |  | ||||||
| 		} else if (cancelledStartGame) { |  | ||||||
| 			System.out.println("Update cancelled by user! Continuing to start game..."); |  | ||||||
| 			System.exit(0); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,105 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import com.google.gson.annotations.SerializedName; |  | ||||||
| import com.moandjiezana.toml.Toml; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.GeneralHashingSource; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.Hash; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.HashUtils; |  | ||||||
| import link.infra.packwiz.installer.request.HandlerManager; |  | ||||||
| import okio.Okio; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.nio.file.Paths; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| public class IndexFile { |  | ||||||
| 	@SerializedName("hash-format") |  | ||||||
| 	public String hashFormat; |  | ||||||
| 	public List<File> files; |  | ||||||
| 	 |  | ||||||
| 	public static class File { |  | ||||||
| 		public SpaceSafeURI file; |  | ||||||
| 		@SerializedName("hash-format") |  | ||||||
| 		public String hashFormat; |  | ||||||
| 		public String hash; |  | ||||||
| 		public SpaceSafeURI alias; |  | ||||||
| 		public boolean metafile; |  | ||||||
| 		public boolean preserve; |  | ||||||
|  |  | ||||||
| 		public transient ModFile linkedFile; |  | ||||||
| 		public transient SpaceSafeURI linkedFileURI; |  | ||||||
|  |  | ||||||
| 		public void downloadMeta(IndexFile parentIndexFile, SpaceSafeURI indexUri) throws Exception { |  | ||||||
| 			if (!metafile) { |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 			if (hashFormat == null || hashFormat.length() == 0) { |  | ||||||
| 				hashFormat = parentIndexFile.hashFormat; |  | ||||||
| 			} |  | ||||||
| 			Hash fileHash = HashUtils.getHash(hashFormat, hash); |  | ||||||
| 			linkedFileURI = HandlerManager.getNewLoc(indexUri, file); |  | ||||||
| 			Source src = HandlerManager.getFileSource(linkedFileURI); |  | ||||||
| 			GeneralHashingSource fileStream = HashUtils.getHasher(hashFormat).getHashingSource(src); |  | ||||||
|  |  | ||||||
| 			linkedFile = new Toml().read(Okio.buffer(fileStream).inputStream()).to(ModFile.class); |  | ||||||
| 			if (!fileStream.hashIsEqual(fileHash)) { |  | ||||||
| 				throw new Exception("Invalid mod file hash"); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public Source getSource(SpaceSafeURI indexUri) throws Exception { |  | ||||||
| 			if (metafile) { |  | ||||||
| 				if (linkedFile == null) { |  | ||||||
| 					throw new Exception("Linked file doesn't exist!"); |  | ||||||
| 				} |  | ||||||
| 				return linkedFile.getSource(linkedFileURI); |  | ||||||
| 			} else { |  | ||||||
| 				SpaceSafeURI newLoc = HandlerManager.getNewLoc(indexUri, file); |  | ||||||
| 				if (newLoc == null) { |  | ||||||
| 					throw new Exception("Index file URI is invalid"); |  | ||||||
| 				} |  | ||||||
| 				return HandlerManager.getFileSource(newLoc); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public Hash getHash() throws Exception { |  | ||||||
| 			if (hash == null) { |  | ||||||
| 				// TODO: should these be more specific exceptions (e.g. IndexFileException?!) |  | ||||||
| 				throw new Exception("Index file doesn't have a hash"); |  | ||||||
| 			} |  | ||||||
| 			if (hashFormat == null) { |  | ||||||
| 				throw new Exception("Index file doesn't have a hash format"); |  | ||||||
| 			} |  | ||||||
| 			return HashUtils.getHash(hashFormat, hash); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public String getName() { |  | ||||||
| 			if (metafile) { |  | ||||||
| 				if (linkedFile != null) { |  | ||||||
| 					if (linkedFile.name != null) { |  | ||||||
| 						return linkedFile.name; |  | ||||||
| 					} else if (linkedFile.filename != null) { |  | ||||||
| 						return linkedFile.filename; |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			if (file != null) { |  | ||||||
| 				return Paths.get(file.getPath()).getFileName().toString(); |  | ||||||
| 			} |  | ||||||
| 			// TODO: throw some kind of exception? |  | ||||||
| 			return "Invalid file"; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public SpaceSafeURI getDestURI() { |  | ||||||
| 			if (alias != null) { |  | ||||||
| 				return alias; |  | ||||||
| 			} |  | ||||||
| 			if (metafile && linkedFile != null) { |  | ||||||
| 				// TODO: URIs are bad |  | ||||||
| 				return file.resolve(linkedFile.filename); |  | ||||||
| 			} else { |  | ||||||
| 				return file; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.UpdateManager; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.Hash; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| public class ManifestFile { |  | ||||||
| 	public Hash packFileHash = null; |  | ||||||
| 	public Hash indexFileHash = null; |  | ||||||
| 	public Map<SpaceSafeURI, File> cachedFiles; |  | ||||||
| 	// If the side changes, EVERYTHING invalidates. FUN!!! |  | ||||||
| 	public UpdateManager.Options.Side cachedSide = UpdateManager.Options.Side.CLIENT; |  | ||||||
|  |  | ||||||
| 	public static class File { |  | ||||||
| 		private transient File revert; |  | ||||||
|  |  | ||||||
| 		public Hash hash = null; |  | ||||||
| 		public Hash linkedFileHash = null; |  | ||||||
| 		public String cachedLocation = null; |  | ||||||
|  |  | ||||||
| 		public boolean isOptional = false; |  | ||||||
| 		public boolean optionValue = true; |  | ||||||
|  |  | ||||||
| 		// When an error occurs, the state needs to be reverted. To do this, I have a crude revert system. |  | ||||||
| 		public void backup() { |  | ||||||
| 			revert = new File(); |  | ||||||
| 			revert.hash = hash; |  | ||||||
| 			revert.linkedFileHash = linkedFileHash; |  | ||||||
| 			revert.cachedLocation = cachedLocation; |  | ||||||
| 			revert.isOptional = isOptional; |  | ||||||
| 			revert.optionValue = optionValue; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public File getRevert() { |  | ||||||
| 			return revert; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,70 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import com.google.gson.annotations.SerializedName; |  | ||||||
| import link.infra.packwiz.installer.UpdateManager.Options.Side; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.Hash; |  | ||||||
| import link.infra.packwiz.installer.metadata.hash.HashUtils; |  | ||||||
| import link.infra.packwiz.installer.request.HandlerManager; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| public class ModFile { |  | ||||||
| 	public String name; |  | ||||||
| 	public String filename; |  | ||||||
| 	public Side side; |  | ||||||
|  |  | ||||||
| 	public Download download; |  | ||||||
| 	public static class Download { |  | ||||||
| 		public SpaceSafeURI url; |  | ||||||
| 		@SerializedName("hash-format") |  | ||||||
| 		public String hashFormat; |  | ||||||
| 		public String hash; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public Map<String, Object> update; |  | ||||||
|  |  | ||||||
| 	public Option option; |  | ||||||
| 	public static class Option { |  | ||||||
| 		public boolean optional; |  | ||||||
| 		public String description; |  | ||||||
| 		@SerializedName("default") |  | ||||||
| 		public boolean defaultValue; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public Source getSource(SpaceSafeURI baseLoc) throws Exception { |  | ||||||
| 		if (download == null) { |  | ||||||
| 			throw new Exception("Metadata file doesn't have download"); |  | ||||||
| 		} |  | ||||||
| 		if (download.url == null) { |  | ||||||
| 			throw new Exception("Metadata file doesn't have a download URI"); |  | ||||||
| 		} |  | ||||||
| 		SpaceSafeURI newLoc = HandlerManager.getNewLoc(baseLoc, download.url); |  | ||||||
| 		if (newLoc == null) { |  | ||||||
| 			throw new Exception("Metadata file URI is invalid"); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		return HandlerManager.getFileSource(newLoc); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public Hash getHash() throws Exception { |  | ||||||
| 		if (download == null) { |  | ||||||
| 			throw new Exception("Metadata file doesn't have download"); |  | ||||||
| 		} |  | ||||||
| 		if (download.hash == null) { |  | ||||||
| 			throw new Exception("Metadata file doesn't have a hash"); |  | ||||||
| 		} |  | ||||||
| 		if (download.hashFormat == null) { |  | ||||||
| 			throw new Exception("Metadata file doesn't have a hash format"); |  | ||||||
| 		} |  | ||||||
| 		return HashUtils.getHash(download.hashFormat, download.hash); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public boolean isOptional() { |  | ||||||
| 		if (option != null) { |  | ||||||
| 			return option.optional; |  | ||||||
| 		} |  | ||||||
| 		return false; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import com.google.gson.annotations.SerializedName; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| public class PackFile { |  | ||||||
| 	public String name; |  | ||||||
|  |  | ||||||
| 	public IndexFileLoc index; |  | ||||||
| 	public static class IndexFileLoc { |  | ||||||
| 		public SpaceSafeURI file; |  | ||||||
| 		@SerializedName("hash-format") |  | ||||||
| 		public String hashFormat; |  | ||||||
| 		public String hash; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public Map<String, String> versions; |  | ||||||
| 	public Map<String, Object> client; |  | ||||||
| 	public Map<String, Object> server; |  | ||||||
| } |  | ||||||
| @@ -1,88 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import com.google.gson.annotations.JsonAdapter; |  | ||||||
|  |  | ||||||
| import java.io.Serializable; |  | ||||||
| import java.net.MalformedURLException; |  | ||||||
| import java.net.URI; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
| import java.net.URL; |  | ||||||
|  |  | ||||||
| // The world's worst URI wrapper |  | ||||||
| @JsonAdapter(SpaceSafeURIParser.class) |  | ||||||
| public class SpaceSafeURI implements Comparable<SpaceSafeURI>, Serializable { |  | ||||||
| 	private final URI u; |  | ||||||
|  |  | ||||||
| 	public SpaceSafeURI(String str) throws URISyntaxException { |  | ||||||
| 		u = new URI(str.replace(" ", "%20")); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public SpaceSafeURI(URI uri) { |  | ||||||
| 		this.u = uri; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public SpaceSafeURI(String scheme, String authority, String path, String query, String fragment) throws URISyntaxException { |  | ||||||
| 		// TODO: do all components need to be replaced? |  | ||||||
| 		scheme = scheme.replace(" ", "%20"); |  | ||||||
| 		authority = authority.replace(" ", "%20"); |  | ||||||
| 		path = path.replace(" ", "%20"); |  | ||||||
| 		query = query.replace(" ", "%20"); |  | ||||||
| 		fragment = fragment.replace(" ", "%20"); |  | ||||||
| 		u = new URI(scheme, authority, path, query, fragment); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getPath() { |  | ||||||
| 		return u.getPath().replace("%20", " "); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String toString() { |  | ||||||
| 		return u.toString().replace("%20", " "); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@SuppressWarnings("WeakerAccess") |  | ||||||
| 	public SpaceSafeURI resolve(String path) { |  | ||||||
| 		return new SpaceSafeURI(u.resolve(path.replace(" ", "%20"))); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public SpaceSafeURI resolve(SpaceSafeURI loc) { |  | ||||||
| 		return new SpaceSafeURI(u.resolve(loc.u)); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public SpaceSafeURI relativize(SpaceSafeURI loc) { |  | ||||||
| 		return new SpaceSafeURI(u.relativize(loc.u)); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public boolean equals(Object obj) { |  | ||||||
| 		if (obj instanceof SpaceSafeURI) { |  | ||||||
| 			return u.equals(((SpaceSafeURI) obj).u); |  | ||||||
| 		} |  | ||||||
| 		return false; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public int hashCode() { |  | ||||||
| 		return u.hashCode(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public int compareTo(SpaceSafeURI uri) { |  | ||||||
| 		return u.compareTo(uri.u); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getScheme() { |  | ||||||
| 		return u.getScheme(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getAuthority() { |  | ||||||
| 		return u.getAuthority(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getHost() { |  | ||||||
| 		return u.getHost(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public URL toURL() throws MalformedURLException { |  | ||||||
| 		return u.toURL(); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata; |  | ||||||
|  |  | ||||||
| import com.google.gson.JsonDeserializationContext; |  | ||||||
| import com.google.gson.JsonDeserializer; |  | ||||||
| import com.google.gson.JsonElement; |  | ||||||
| import com.google.gson.JsonParseException; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.Type; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This class encodes spaces before parsing the URI, so the URI can actually be |  | ||||||
|  * parsed. |  | ||||||
|  */ |  | ||||||
| class SpaceSafeURIParser implements JsonDeserializer<SpaceSafeURI> { |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public SpaceSafeURI deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) |  | ||||||
| 			throws JsonParseException { |  | ||||||
| 		try { |  | ||||||
| 			return new SpaceSafeURI(json.getAsString()); |  | ||||||
| 		} catch (URISyntaxException e) { |  | ||||||
| 			throw new JsonParseException("Failed to parse URI", e); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// TODO: replace this with a better solution? |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import okio.ForwardingSource; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| public abstract class GeneralHashingSource extends ForwardingSource { |  | ||||||
|  |  | ||||||
| 	public GeneralHashingSource(Source delegate) { |  | ||||||
| 		super(delegate); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public abstract Hash getHash(); |  | ||||||
|  |  | ||||||
| 	public boolean hashIsEqual(Object compareTo) { |  | ||||||
| 		return compareTo.equals(getHash()); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.Type; |  | ||||||
|  |  | ||||||
| import com.google.gson.JsonDeserializationContext; |  | ||||||
| import com.google.gson.JsonDeserializer; |  | ||||||
| import com.google.gson.JsonElement; |  | ||||||
| import com.google.gson.JsonObject; |  | ||||||
| import com.google.gson.JsonParseException; |  | ||||||
| import com.google.gson.JsonPrimitive; |  | ||||||
| import com.google.gson.JsonSerializationContext; |  | ||||||
| import com.google.gson.JsonSerializer; |  | ||||||
|  |  | ||||||
| public abstract class Hash { |  | ||||||
| 	protected abstract String getStringValue(); |  | ||||||
|  |  | ||||||
| 	protected abstract String getType(); |  | ||||||
|  |  | ||||||
| 	public static class TypeHandler implements JsonDeserializer<Hash>, JsonSerializer<Hash> { |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public JsonElement serialize(Hash src, Type typeOfSrc, JsonSerializationContext context) { |  | ||||||
| 			JsonObject out = new JsonObject(); |  | ||||||
| 			out.add("type", new JsonPrimitive(src.getType())); |  | ||||||
| 			out.add("value", new JsonPrimitive(src.getStringValue())); |  | ||||||
| 			return out; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public Hash deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) |  | ||||||
| 				throws JsonParseException { |  | ||||||
| 			JsonObject obj = json.getAsJsonObject(); |  | ||||||
| 			String type, value; |  | ||||||
| 			try { |  | ||||||
| 				type = obj.get("type").getAsString(); |  | ||||||
| 				value = obj.get("value").getAsString(); |  | ||||||
| 			} catch (NullPointerException e) { |  | ||||||
| 				throw new JsonParseException("Invalid hash JSON data"); |  | ||||||
| 			} |  | ||||||
| 			try { |  | ||||||
| 				return HashUtils.getHash(type, value); |  | ||||||
| 			} catch (Exception e) { |  | ||||||
| 				throw new JsonParseException("Failed to create hash object", e); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| public class HashUtils { |  | ||||||
| 	private static final Map<String, IHasher> hashTypeConversion = new HashMap<String, IHasher>(); |  | ||||||
| 	static { |  | ||||||
| 		hashTypeConversion.put("sha256", new HashingSourceHasher("sha256")); |  | ||||||
| 		hashTypeConversion.put("murmur2", new Murmur2Hasher()); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public static IHasher getHasher(String type) throws Exception { |  | ||||||
| 		IHasher hasher = hashTypeConversion.get(type); |  | ||||||
| 		if (hasher == null) { |  | ||||||
| 			throw new Exception("Hash type not supported: " + type); |  | ||||||
| 		} |  | ||||||
| 		return hasher; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public static Hash getHash(String type, String value) throws Exception { |  | ||||||
| 		if (hashTypeConversion.containsKey(type)) { |  | ||||||
| 			return hashTypeConversion.get(type).getHash(value); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		throw new Exception("Hash type not supported: " + type); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,86 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import okio.HashingSource; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| public class HashingSourceHasher implements IHasher { |  | ||||||
| 	String type; |  | ||||||
|  |  | ||||||
| 	public HashingSourceHasher(String type) { |  | ||||||
| 		this.type = type; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// i love naming things |  | ||||||
| 	private class HashingSourceGeneralHashingSource extends GeneralHashingSource { |  | ||||||
| 		HashingSource delegateHashing; |  | ||||||
| 		HashingSourceHash value; |  | ||||||
|  |  | ||||||
| 		public HashingSourceGeneralHashingSource(HashingSource delegate) { |  | ||||||
| 			super(delegate); |  | ||||||
| 			delegateHashing = delegate; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public Hash getHash() { |  | ||||||
| 			if (value == null) { |  | ||||||
| 				value = new HashingSourceHash(delegateHashing.hash().hex()); |  | ||||||
| 			} |  | ||||||
| 			return value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// this some funky inner class stuff |  | ||||||
| 	// each of these classes is specific to the instance of the HasherHashingSource |  | ||||||
| 	// therefore HashingSourceHashes from different parent instances will be not instanceof each other |  | ||||||
| 	private class HashingSourceHash extends Hash { |  | ||||||
| 		String value; |  | ||||||
| 		private HashingSourceHash(String value) { |  | ||||||
| 			this.value = value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public boolean equals(Object obj) { |  | ||||||
| 			if (!(obj instanceof HashingSourceHash)) { |  | ||||||
| 				return false; |  | ||||||
| 			} |  | ||||||
| 			HashingSourceHash objHash = (HashingSourceHash) obj; |  | ||||||
| 			if (value != null) { |  | ||||||
| 				return value.equals(objHash.value); |  | ||||||
| 			} else { |  | ||||||
| 				return objHash.value == null; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public String toString() { |  | ||||||
| 			return type + ": " + value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		protected String getStringValue() { |  | ||||||
| 			return value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		protected String getType() { |  | ||||||
| 			return type; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public GeneralHashingSource getHashingSource(Source delegate) { |  | ||||||
| 		switch (type) { |  | ||||||
| 			case "sha256": |  | ||||||
| 			return new HashingSourceGeneralHashingSource(HashingSource.sha256(delegate)); |  | ||||||
| 			// TODO: support other hash types |  | ||||||
| 		} |  | ||||||
| 		throw new RuntimeException("Invalid hash type provided"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Hash getHash(String value) { |  | ||||||
| 		return new HashingSourceHash(value); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| } |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| public interface IHasher { |  | ||||||
| 	public GeneralHashingSource getHashingSource(Source delegate); |  | ||||||
| 	public Hash getHash(String value); |  | ||||||
| } |  | ||||||
| @@ -1,103 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.metadata.hash; |  | ||||||
|  |  | ||||||
| import okio.Buffer; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| public class Murmur2Hasher implements IHasher { |  | ||||||
| 	private class Murmur2GeneralHashingSource extends GeneralHashingSource { |  | ||||||
| 		Murmur2Hash value; |  | ||||||
| 		Buffer internalBuffer = new Buffer(); |  | ||||||
| 		Buffer tempBuffer = new Buffer(); |  | ||||||
| 		Source delegate; |  | ||||||
|  |  | ||||||
| 		public Murmur2GeneralHashingSource(Source delegate) { |  | ||||||
| 			super(delegate); |  | ||||||
| 			this.delegate = delegate; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public long read(Buffer sink, long byteCount) throws IOException { |  | ||||||
| 			long out = delegate.read(tempBuffer, byteCount); |  | ||||||
| 			if (out > -1) { |  | ||||||
| 				sink.write(tempBuffer.clone(), out); |  | ||||||
| 				internalBuffer.write(tempBuffer, out); |  | ||||||
| 			} |  | ||||||
| 			return out; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public Hash getHash() { |  | ||||||
| 			if (value == null) { |  | ||||||
| 				byte[] data = computeNormalizedArray(internalBuffer.readByteArray()); |  | ||||||
| 				value = new Murmur2Hash(Murmur2Lib.hash32(data, data.length, 1)); |  | ||||||
| 			} |  | ||||||
| 			return value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Credit to https://github.com/modmuss50/CAV2/blob/master/murmur.go |  | ||||||
| 		private byte[] computeNormalizedArray(byte[] input) { |  | ||||||
| 			byte[] output = new byte[input.length]; |  | ||||||
| 			int num = 0; |  | ||||||
| 			for (byte b : input) { |  | ||||||
| 				if (!(b == 9 || b == 10 || b == 13 || b == 32)) { |  | ||||||
| 					output[num] = b; |  | ||||||
| 					num++; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			byte[] outputTrimmed = new byte[num]; |  | ||||||
| 			System.arraycopy(output, 0, outputTrimmed, 0, num); |  | ||||||
| 			return outputTrimmed; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private static class Murmur2Hash extends Hash { |  | ||||||
| 		int value; |  | ||||||
| 		private Murmur2Hash(String value) { |  | ||||||
| 			// Parsing as long then casting to int converts values gt int max value but lt uint max value |  | ||||||
| 			// into negatives. I presume this is how the murmur2 code handles this. |  | ||||||
| 			this.value = (int)Long.parseLong(value); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		private Murmur2Hash(int value) { |  | ||||||
| 			this.value = value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public boolean equals(Object obj) { |  | ||||||
| 			if (!(obj instanceof Murmur2Hash)) { |  | ||||||
| 				return false; |  | ||||||
| 			} |  | ||||||
| 			Murmur2Hash objHash = (Murmur2Hash) obj; |  | ||||||
| 			return value == objHash.value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public String toString() { |  | ||||||
| 			return "murmur2: " + value; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		protected String getStringValue() { |  | ||||||
| 			return Integer.toString(value); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		protected String getType() { |  | ||||||
| 			return "murmur2"; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public GeneralHashingSource getHashingSource(Source delegate) { |  | ||||||
| 		return new Murmur2GeneralHashingSource(delegate); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Hash getHash(String value) { |  | ||||||
| 		return new Murmur2Hash(value); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| } |  | ||||||
| @@ -74,13 +74,13 @@ public class Murmur2Lib { | |||||||
|     int left = length - len_m; |     int left = length - len_m; | ||||||
|     if (left != 0) { |     if (left != 0) { | ||||||
|       if (left >= 3) { |       if (left >= 3) { | ||||||
|         h ^= (int) data[length - 3] << 16; |         h ^= (int) data[length - (left - 2)] << 16; | ||||||
|       } |       } | ||||||
|       if (left >= 2) { |       if (left >= 2) { | ||||||
|         h ^= (int) data[length - 2] << 8; |         h ^= (int) data[length - (left - 1)] << 8; | ||||||
|       } |       } | ||||||
|       if (left >= 1) { |       if (left >= 1) { | ||||||
|         h ^= (int) data[length - 1]; |         h ^= data[length - left]; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       h *= M_32; |       h *= M_32; | ||||||
| @@ -152,7 +152,7 @@ public class Murmur2Lib { | |||||||
|       case 2: |       case 2: | ||||||
|         h ^= (long) (data[tailStart + 1] & 0xff) << 8; |         h ^= (long) (data[tailStart + 1] & 0xff) << 8; | ||||||
|       case 1: |       case 1: | ||||||
|         h ^= (long) (data[tailStart] & 0xff); |         h ^= data[tailStart] & 0xff; | ||||||
|         h *= M_64; |         h *= M_64; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,59 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.request; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import link.infra.packwiz.installer.request.handlers.RequestHandlerGithub; |  | ||||||
| import link.infra.packwiz.installer.request.handlers.RequestHandlerHTTP; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| public abstract class HandlerManager { |  | ||||||
| 	 |  | ||||||
| 	private static List<IRequestHandler> handlers = new ArrayList<>(); |  | ||||||
| 	 |  | ||||||
| 	static { |  | ||||||
| 		handlers.add(new RequestHandlerGithub()); |  | ||||||
| 		handlers.add(new RequestHandlerHTTP()); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	public static SpaceSafeURI getNewLoc(SpaceSafeURI base, SpaceSafeURI loc) { |  | ||||||
| 		if (loc == null) return null; |  | ||||||
| 		if (base != null) { |  | ||||||
| 			loc = base.resolve(loc); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		for (IRequestHandler handler : handlers) { |  | ||||||
| 			if (handler.matchesHandler(loc)) { |  | ||||||
| 				return handler.getNewLoc(loc); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return loc; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	// TODO: What if files are read multiple times?? |  | ||||||
| 	// Zip handler discards once read, requesting multiple times on other handlers would cause multiple downloads |  | ||||||
| 	// Caching system? Copy from already downloaded files? |  | ||||||
|  |  | ||||||
| 	public static Source getFileSource(SpaceSafeURI loc) throws Exception { |  | ||||||
| 		for (IRequestHandler handler : handlers) { |  | ||||||
| 			if (handler.matchesHandler(loc)) { |  | ||||||
| 				Source src = handler.getFileSource(loc); |  | ||||||
| 				if (src == null) { |  | ||||||
| 					throw new Exception("Couldn't find URI: " + loc.toString()); |  | ||||||
| 				} else { |  | ||||||
| 					return src; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		// TODO: specialised exception classes?? |  | ||||||
| 		throw new Exception("No handler available for URI: " + loc.toString()); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	// github toml resolution |  | ||||||
| 	// e.g. https://github.com/comp500/Demagnetize -> demagnetize.toml |  | ||||||
| 	// https://github.com/comp500/Demagnetize/blob/master/demagnetize.toml |  | ||||||
| 	 |  | ||||||
| 	// To handle "progress", just count tasks, rather than individual progress |  | ||||||
| 	// It'll look bad, especially for zip-based things, but it should work fine |  | ||||||
| } |  | ||||||
| @@ -1,90 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.request.handlers; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
|  |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
| import java.util.concurrent.locks.ReentrantReadWriteLock; |  | ||||||
| import java.util.regex.Matcher; |  | ||||||
| import java.util.regex.Pattern; |  | ||||||
|  |  | ||||||
| public class RequestHandlerGithub extends RequestHandlerZip { |  | ||||||
| 	 |  | ||||||
| 	public RequestHandlerGithub() { |  | ||||||
| 		super(true); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public SpaceSafeURI getNewLoc(SpaceSafeURI loc) { |  | ||||||
| 		return loc; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	// TODO: is caching really needed, if HTTPURLConnection follows redirects correctly? |  | ||||||
| 	private Map<String, SpaceSafeURI> zipUriMap = new HashMap<>(); |  | ||||||
| 	private final ReentrantReadWriteLock zipUriLock = new ReentrantReadWriteLock(); |  | ||||||
| 	private static Pattern repoMatcherPattern = Pattern.compile("/([\\w.-]+/[\\w.-]+).*"); |  | ||||||
| 	 |  | ||||||
| 	private String getRepoName(SpaceSafeURI loc) { |  | ||||||
| 		Matcher matcher = repoMatcherPattern.matcher(loc.getPath()); |  | ||||||
| 		if (matcher.matches()) { |  | ||||||
| 			return matcher.group(1); |  | ||||||
| 		} else { |  | ||||||
| 			return null; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	protected SpaceSafeURI getZipUri(SpaceSafeURI loc) throws Exception { |  | ||||||
| 		String repoName = getRepoName(loc); |  | ||||||
| 		String branchName = getBranch(loc); |  | ||||||
| 		zipUriLock.readLock().lock(); |  | ||||||
| 		SpaceSafeURI zipUri = zipUriMap.get(repoName + "/" + branchName); |  | ||||||
| 		zipUriLock.readLock().unlock(); |  | ||||||
| 		if (zipUri != null) { |  | ||||||
| 			return zipUri; |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		zipUri = new SpaceSafeURI("https://api.github.com/repos/" + repoName + "/zipball/" + branchName); |  | ||||||
| 		 |  | ||||||
| 		zipUriLock.writeLock().lock(); |  | ||||||
| 		// If another thread sets the value concurrently, use the value of the |  | ||||||
| 		// thread that first acquired the lock. |  | ||||||
| 		SpaceSafeURI zipUriInserted = zipUriMap.putIfAbsent(repoName + "/" + branchName, zipUri); |  | ||||||
| 		if (zipUriInserted != null) { |  | ||||||
| 			zipUri = zipUriInserted; |  | ||||||
| 		} |  | ||||||
| 		zipUriLock.writeLock().unlock(); |  | ||||||
| 		return zipUri; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	private static Pattern branchMatcherPattern = Pattern.compile("/[\\w.-]+/[\\w.-]+/blob/([\\w.-]+).*"); |  | ||||||
| 	 |  | ||||||
| 	private String getBranch(SpaceSafeURI loc) { |  | ||||||
| 		Matcher matcher = branchMatcherPattern.matcher(loc.getPath()); |  | ||||||
| 		if (matcher.matches()) { |  | ||||||
| 			return matcher.group(1); |  | ||||||
| 		} else { |  | ||||||
| 			return null; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	protected SpaceSafeURI getLocationInZip(SpaceSafeURI loc) throws Exception { |  | ||||||
| 		String path = "/" + getRepoName(loc) + "/blob/" + getBranch(loc); |  | ||||||
| 		return new SpaceSafeURI(loc.getScheme(), loc.getAuthority(), path, null, null).relativize(loc); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public boolean matchesHandler(SpaceSafeURI loc) { |  | ||||||
| 		String scheme = loc.getScheme(); |  | ||||||
| 		if (!("http".equals(scheme) || "https".equals(scheme))) { |  | ||||||
| 			return false; |  | ||||||
| 		} |  | ||||||
| 		if (!"github.com".equals(loc.getHost())) { |  | ||||||
| 			return false; |  | ||||||
| 		} |  | ||||||
| 		// TODO: sanity checks, support for more github urls |  | ||||||
| 		return true; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.request.handlers; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import link.infra.packwiz.installer.request.IRequestHandler; |  | ||||||
| import okio.Okio; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.net.URLConnection; |  | ||||||
|  |  | ||||||
| public class RequestHandlerHTTP implements IRequestHandler { |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public boolean matchesHandler(SpaceSafeURI loc) { |  | ||||||
| 		String scheme = loc.getScheme(); |  | ||||||
| 		return "http".equals(scheme) || "https".equals(scheme); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Source getFileSource(SpaceSafeURI loc) throws Exception { |  | ||||||
| 		URLConnection conn = loc.toURL().openConnection(); |  | ||||||
| 		// TODO: when do we send specific headers??? should there be a way to signal this? |  | ||||||
| 		// github *sometimes* requires it, sometimes not! |  | ||||||
| 		//conn.addRequestProperty("Accept", "application/octet-stream"); |  | ||||||
| 		// 30 second read timeout |  | ||||||
| 		conn.setReadTimeout(30 * 1000); |  | ||||||
| 		return Okio.source(conn.getInputStream()); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,169 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.request.handlers; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; |  | ||||||
| import okio.Buffer; |  | ||||||
| import okio.BufferedSource; |  | ||||||
| import okio.Okio; |  | ||||||
| import okio.Source; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
| import java.util.concurrent.locks.ReentrantLock; |  | ||||||
| import java.util.concurrent.locks.ReentrantReadWriteLock; |  | ||||||
| import java.util.function.Predicate; |  | ||||||
| import java.util.zip.ZipEntry; |  | ||||||
| import java.util.zip.ZipInputStream; |  | ||||||
|  |  | ||||||
| public abstract class RequestHandlerZip extends RequestHandlerHTTP { |  | ||||||
| 	 |  | ||||||
| 	private final boolean modeHasFolder; |  | ||||||
| 	 |  | ||||||
| 	public RequestHandlerZip(boolean modeHasFolder) { |  | ||||||
| 		this.modeHasFolder = modeHasFolder; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	private String removeFolder(String name) { |  | ||||||
| 		if (modeHasFolder) { |  | ||||||
| 			return name.substring(name.indexOf("/")+1); |  | ||||||
| 		} else { |  | ||||||
| 			return name; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	private class ZipReader { |  | ||||||
| 		 |  | ||||||
| 		private final ZipInputStream zis; |  | ||||||
| 		private final Map<SpaceSafeURI, Buffer> readFiles = new HashMap<>(); |  | ||||||
| 		// Write lock implies access to ZipInputStream - only 1 thread must read at a time! |  | ||||||
| 		final ReentrantLock filesLock = new ReentrantLock(); |  | ||||||
| 		private ZipEntry entry; |  | ||||||
|  |  | ||||||
| 		private final BufferedSource zipSource; |  | ||||||
|  |  | ||||||
| 		ZipReader(Source zip) { |  | ||||||
| 			zis = new ZipInputStream(Okio.buffer(zip).inputStream()); |  | ||||||
| 			zipSource = Okio.buffer(Okio.source(zis)); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		// File lock must be obtained before calling this function |  | ||||||
| 		private Buffer readCurrFile() throws IOException { |  | ||||||
| 			Buffer fileBuffer = new Buffer(); |  | ||||||
| 			zipSource.readFully(fileBuffer, entry.getSize()); |  | ||||||
| 			return fileBuffer; |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		// File lock must be obtained before calling this function |  | ||||||
| 		private Buffer findFile(SpaceSafeURI loc) throws IOException, URISyntaxException { |  | ||||||
| 			while (true) { |  | ||||||
| 				entry = zis.getNextEntry(); |  | ||||||
| 				if (entry == null) { |  | ||||||
| 					return null; |  | ||||||
| 				} |  | ||||||
| 				Buffer data = readCurrFile(); |  | ||||||
| 				SpaceSafeURI fileLoc = new SpaceSafeURI(removeFolder(entry.getName())); |  | ||||||
| 				if (loc.equals(fileLoc)) { |  | ||||||
| 					return data; |  | ||||||
| 				} else { |  | ||||||
| 					readFiles.put(fileLoc, data); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		Source getFileSource(SpaceSafeURI loc) throws Exception { |  | ||||||
| 			filesLock.lock(); |  | ||||||
| 			// Assume files are only read once, allow GC by removing |  | ||||||
| 			Buffer file = readFiles.remove(loc); |  | ||||||
| 			if (file != null) { |  | ||||||
| 				filesLock.unlock(); |  | ||||||
| 				return file; |  | ||||||
| 			} |  | ||||||
| 			 |  | ||||||
| 			file = findFile(loc); |  | ||||||
| 			filesLock.unlock(); |  | ||||||
| 			return file; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		SpaceSafeURI findInZip(Predicate<SpaceSafeURI> matches) throws Exception { |  | ||||||
| 			filesLock.lock(); |  | ||||||
| 			for (SpaceSafeURI file : readFiles.keySet()) { |  | ||||||
| 				if (matches.test(file)) { |  | ||||||
| 					filesLock.unlock(); |  | ||||||
| 					return file; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			 |  | ||||||
| 			while (true) { |  | ||||||
| 				entry = zis.getNextEntry(); |  | ||||||
| 				if (entry == null) { |  | ||||||
| 					filesLock.unlock(); |  | ||||||
| 					return null; |  | ||||||
| 				} |  | ||||||
| 				Buffer data = readCurrFile(); |  | ||||||
| 				SpaceSafeURI fileLoc = new SpaceSafeURI(removeFolder(entry.getName())); |  | ||||||
| 				readFiles.put(fileLoc, data); |  | ||||||
| 				if (matches.test(fileLoc)) { |  | ||||||
| 					filesLock.unlock(); |  | ||||||
| 					return fileLoc; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	private final Map<SpaceSafeURI, ZipReader> cache = new HashMap<>(); |  | ||||||
| 	private final ReentrantReadWriteLock cacheLock = new ReentrantReadWriteLock(); |  | ||||||
| 	 |  | ||||||
| 	protected abstract SpaceSafeURI getZipUri(SpaceSafeURI loc) throws Exception; |  | ||||||
| 	 |  | ||||||
| 	protected abstract SpaceSafeURI getLocationInZip(SpaceSafeURI loc) throws Exception; |  | ||||||
| 	 |  | ||||||
| 	@Override |  | ||||||
| 	public abstract boolean matchesHandler(SpaceSafeURI loc); |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Source getFileSource(SpaceSafeURI loc) throws Exception { |  | ||||||
| 		SpaceSafeURI zipUri = getZipUri(loc); |  | ||||||
| 		cacheLock.readLock().lock(); |  | ||||||
| 		ZipReader zr = cache.get(zipUri); |  | ||||||
| 		cacheLock.readLock().unlock(); |  | ||||||
| 		if (zr == null) { |  | ||||||
| 			cacheLock.writeLock().lock(); |  | ||||||
| 			// Recheck, because unlocking read lock allows another thread to modify it |  | ||||||
| 			zr = cache.get(zipUri); |  | ||||||
| 			if (zr == null) { |  | ||||||
| 				Source src = super.getFileSource(zipUri); |  | ||||||
| 				if (src == null) { |  | ||||||
| 					cacheLock.writeLock().unlock(); |  | ||||||
| 					return null; |  | ||||||
| 				} |  | ||||||
| 				zr = new ZipReader(src); |  | ||||||
| 				cache.put(zipUri, zr); |  | ||||||
| 			} |  | ||||||
| 			cacheLock.writeLock().unlock(); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		return zr.getFileSource(getLocationInZip(loc)); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	protected SpaceSafeURI findInZip(SpaceSafeURI loc, Predicate<SpaceSafeURI> matches) throws Exception { |  | ||||||
| 		SpaceSafeURI zipUri = getZipUri(loc); |  | ||||||
| 		cacheLock.readLock().lock(); |  | ||||||
| 		ZipReader zr = cache.get(zipUri); |  | ||||||
| 		cacheLock.readLock().unlock(); |  | ||||||
| 		if (zr == null) { |  | ||||||
| 			cacheLock.writeLock().lock(); |  | ||||||
| 			// Recheck, because unlocking read lock allows another thread to modify it |  | ||||||
| 			zr = cache.get(zipUri); |  | ||||||
| 			if (zr == null) { |  | ||||||
| 				zr = new ZipReader(super.getFileSource(zipUri)); |  | ||||||
| 				cache.put(zipUri, zr); |  | ||||||
| 			} |  | ||||||
| 			cacheLock.writeLock().unlock(); |  | ||||||
| 		} |  | ||||||
| 		 |  | ||||||
| 		return zr.findInZip(matches); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,49 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.CompletableFuture; |  | ||||||
| import java.util.concurrent.Future; |  | ||||||
|  |  | ||||||
| public class CLIHandler implements IUserInterface { |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void handleException(Exception e) { |  | ||||||
| 		e.printStackTrace(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void show(InputStateHandler h) {} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void submitProgress(InstallProgress progress) { |  | ||||||
| 		StringBuilder sb = new StringBuilder(); |  | ||||||
| 		if (progress.hasProgress) { |  | ||||||
| 			sb.append('('); |  | ||||||
| 			sb.append(progress.progress); |  | ||||||
| 			sb.append('/'); |  | ||||||
| 			sb.append(progress.progressTotal); |  | ||||||
| 			sb.append(") "); |  | ||||||
| 		} |  | ||||||
| 		sb.append(progress.message); |  | ||||||
| 		System.out.println(sb.toString()); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void executeManager(Runnable task) { |  | ||||||
| 		task.run(); |  | ||||||
| 		System.out.println("Finished successfully!"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Future<Boolean> showOptions(List<IOptionDetails> option) { |  | ||||||
| 		throw new RuntimeException("Optional mods not implemented for CLI! Make sure your optional mods are only on the client side!"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Future<IExceptionDetails.ExceptionListResult> showExceptions(List<IExceptionDetails> opts, int numTotal, boolean allowsIgnore) { |  | ||||||
| 		CompletableFuture<IExceptionDetails.ExceptionListResult> future = new CompletableFuture<>(); |  | ||||||
| 		future.complete(IExceptionDetails.ExceptionListResult.CANCEL); |  | ||||||
| 		return future; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,183 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import link.infra.packwiz.installer.ui.IExceptionDetails.ExceptionListResult; |  | ||||||
|  |  | ||||||
| import javax.swing.*; |  | ||||||
| import javax.swing.border.EmptyBorder; |  | ||||||
| import java.awt.*; |  | ||||||
| import java.awt.event.WindowAdapter; |  | ||||||
| import java.awt.event.WindowEvent; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.io.PrintWriter; |  | ||||||
| import java.io.StringWriter; |  | ||||||
| import java.net.URI; |  | ||||||
| import java.net.URISyntaxException; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.CompletableFuture; |  | ||||||
|  |  | ||||||
| class ExceptionListWindow extends JDialog { |  | ||||||
|  |  | ||||||
| 	private static final long serialVersionUID = 1L; |  | ||||||
| 	private final JTextArea lblExceptionStacktrace; |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Create the dialog. |  | ||||||
| 	 */ |  | ||||||
| 	ExceptionListWindow(List<IExceptionDetails> eList, CompletableFuture<ExceptionListResult> future, int numTotal, boolean allowsIgnore, JFrame parentWindow) { |  | ||||||
| 		super(parentWindow, "Failed file downloads", true); |  | ||||||
|  |  | ||||||
| 		setBounds(100, 100, 540, 340); |  | ||||||
| 		setLocationRelativeTo(parentWindow); |  | ||||||
| 		getContentPane().setLayout(new BorderLayout()); |  | ||||||
| 		{ |  | ||||||
| 			JPanel errorPanel = new JPanel(); |  | ||||||
| 			getContentPane().add(errorPanel, BorderLayout.NORTH); |  | ||||||
| 			{ |  | ||||||
| 				JLabel lblWarning = new JLabel("One or more errors were encountered while installing the modpack!"); |  | ||||||
| 				lblWarning.setIcon(UIManager.getIcon("OptionPane.warningIcon")); |  | ||||||
| 				errorPanel.add(lblWarning); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		JPanel contentPanel = new JPanel(); |  | ||||||
| 		contentPanel.setBorder(new EmptyBorder(5, 5, 5, 5)); |  | ||||||
| 		getContentPane().add(contentPanel, BorderLayout.CENTER); |  | ||||||
| 		contentPanel.setLayout(new BorderLayout(0, 0)); |  | ||||||
| 		{ |  | ||||||
| 			JSplitPane splitPane = new JSplitPane(); |  | ||||||
| 			splitPane.setResizeWeight(0.3); |  | ||||||
| 			contentPanel.add(splitPane); |  | ||||||
| 			{ |  | ||||||
| 				lblExceptionStacktrace = new JTextArea("Select a file"); |  | ||||||
| 				lblExceptionStacktrace.setBackground(UIManager.getColor("List.background")); |  | ||||||
| 				lblExceptionStacktrace.setOpaque(true); |  | ||||||
| 				lblExceptionStacktrace.setWrapStyleWord(true); |  | ||||||
| 				lblExceptionStacktrace.setLineWrap(true); |  | ||||||
| 				lblExceptionStacktrace.setEditable(false); |  | ||||||
| 				lblExceptionStacktrace.setFocusable(true); |  | ||||||
| 				lblExceptionStacktrace.setFont(UIManager.getFont("Label.font")); |  | ||||||
| 				lblExceptionStacktrace.setBorder(new EmptyBorder(5, 5, 5, 5)); |  | ||||||
| 				JScrollPane scrollPane = new JScrollPane(lblExceptionStacktrace); |  | ||||||
| 				scrollPane.setBorder(new EmptyBorder(0, 0, 0, 0)); |  | ||||||
| 				splitPane.setRightComponent(scrollPane); |  | ||||||
| 			} |  | ||||||
| 			{ |  | ||||||
| 				JList<String> list = new JList<>(); |  | ||||||
| 				list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION); |  | ||||||
| 				list.setBorder(new EmptyBorder(5, 5, 5, 5)); |  | ||||||
| 				ExceptionListModel listModel = new ExceptionListModel(eList); |  | ||||||
| 				list.setModel(listModel); |  | ||||||
| 				list.addListSelectionListener(e -> { |  | ||||||
| 					int i = list.getSelectedIndex(); |  | ||||||
| 					if (i > -1) { |  | ||||||
| 						StringWriter sw = new StringWriter(); |  | ||||||
| 						listModel.getExceptionAt(i).printStackTrace(new PrintWriter(sw)); |  | ||||||
| 						lblExceptionStacktrace.setText(sw.toString()); |  | ||||||
| 						// Scroll to the top |  | ||||||
| 						lblExceptionStacktrace.setCaretPosition(0); |  | ||||||
| 					} else { |  | ||||||
| 						lblExceptionStacktrace.setText("Select a file"); |  | ||||||
| 					} |  | ||||||
| 				}); |  | ||||||
| 				JScrollPane scrollPane = new JScrollPane(list); |  | ||||||
| 				scrollPane.setBorder(new EmptyBorder(0, 0, 0, 0)); |  | ||||||
| 				splitPane.setLeftComponent(scrollPane); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		{ |  | ||||||
| 			JPanel buttonPane = new JPanel(); |  | ||||||
| 			getContentPane().add(buttonPane, BorderLayout.SOUTH); |  | ||||||
| 			buttonPane.setLayout(new BorderLayout(0, 0)); |  | ||||||
| 			{ |  | ||||||
| 				JPanel rightButtons = new JPanel(); |  | ||||||
| 				buttonPane.add(rightButtons, BorderLayout.EAST); |  | ||||||
| 				{ |  | ||||||
| 					JButton btnContinue = new JButton("Continue"); |  | ||||||
| 					btnContinue.setToolTipText("Attempt to continue installing, excluding the failed downloads"); |  | ||||||
| 					btnContinue.addActionListener(e -> { |  | ||||||
| 						future.complete(ExceptionListResult.CONTINUE); |  | ||||||
| 						ExceptionListWindow.this.dispose(); |  | ||||||
| 					}); |  | ||||||
| 					rightButtons.add(btnContinue); |  | ||||||
| 				} |  | ||||||
| 				{ |  | ||||||
| 					JButton btnCancelLaunch = new JButton("Cancel launch"); |  | ||||||
| 					btnCancelLaunch.setToolTipText("Stop launching the game"); |  | ||||||
| 					btnCancelLaunch.addActionListener(e -> { |  | ||||||
| 						future.complete(ExceptionListResult.CANCEL); |  | ||||||
| 						ExceptionListWindow.this.dispose(); |  | ||||||
| 					}); |  | ||||||
| 					rightButtons.add(btnCancelLaunch); |  | ||||||
| 				} |  | ||||||
| 				{ |  | ||||||
| 					JButton btnIgnoreUpdate = new JButton("Ignore update"); |  | ||||||
| 					btnIgnoreUpdate.setEnabled(allowsIgnore); |  | ||||||
| 					btnIgnoreUpdate.setToolTipText("Start the game without attempting to update"); |  | ||||||
| 					btnIgnoreUpdate.addActionListener(e -> { |  | ||||||
| 						future.complete(ExceptionListResult.IGNORE); |  | ||||||
| 						ExceptionListWindow.this.dispose(); |  | ||||||
| 					}); |  | ||||||
| 					rightButtons.add(btnIgnoreUpdate); |  | ||||||
| 					{ |  | ||||||
| 						JLabel lblErrored = new JLabel(eList.size() + "/" + numTotal + " errored"); |  | ||||||
| 						lblErrored.setHorizontalAlignment(SwingConstants.CENTER); |  | ||||||
| 						buttonPane.add(lblErrored, BorderLayout.CENTER); |  | ||||||
| 					} |  | ||||||
| 					{ |  | ||||||
| 						JPanel leftButtons = new JPanel(); |  | ||||||
| 						buttonPane.add(leftButtons, BorderLayout.WEST); |  | ||||||
| 						{ |  | ||||||
| 							JButton btnReportIssue = new JButton("Report issue"); |  | ||||||
| 							boolean supported = Desktop.isDesktopSupported() && Desktop.getDesktop().isSupported(Desktop.Action.BROWSE); |  | ||||||
| 							btnReportIssue.setEnabled(supported); |  | ||||||
| 							if (supported) { |  | ||||||
| 								btnReportIssue.addActionListener(e -> { |  | ||||||
| 									try { |  | ||||||
| 										Desktop.getDesktop().browse(new URI("https://github.com/comp500/packwiz-installer/issues/new")); |  | ||||||
| 									} catch (IOException | URISyntaxException e1) { |  | ||||||
| 										// lol the button just won't work i guess |  | ||||||
| 									} |  | ||||||
| 								}); |  | ||||||
| 							} |  | ||||||
| 							leftButtons.add(btnReportIssue); |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		addWindowListener(new WindowAdapter() { |  | ||||||
| 			@Override |  | ||||||
| 			public void windowClosing(WindowEvent e) { |  | ||||||
| 				future.complete(ExceptionListResult.CANCEL); |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			@Override |  | ||||||
| 			public void windowClosed(WindowEvent e) { |  | ||||||
| 				// Just in case closing didn't get triggered - if something else called dispose() the |  | ||||||
| 				// future will have already completed |  | ||||||
| 				future.complete(ExceptionListResult.CANCEL); |  | ||||||
| 			} |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private static class ExceptionListModel extends AbstractListModel<String> { |  | ||||||
| 		private static final long serialVersionUID = 1L; |  | ||||||
| 		private final List<IExceptionDetails> details; |  | ||||||
|  |  | ||||||
| 		ExceptionListModel(List<IExceptionDetails> details) { |  | ||||||
| 			this.details = details; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public int getSize() { |  | ||||||
| 			return details.size(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public String getElementAt(int index) { |  | ||||||
| 			return details.get(index).getName(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		Exception getExceptionAt(int index) { |  | ||||||
| 			return details.get(index).getException(); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| public interface IExceptionDetails { |  | ||||||
| 	Exception getException(); |  | ||||||
| 	String getName(); |  | ||||||
|  |  | ||||||
| 	enum ExceptionListResult { |  | ||||||
| 		CONTINUE, CANCEL, IGNORE |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| public interface IOptionDetails { |  | ||||||
| 	String getName(); |  | ||||||
| 	boolean getOptionValue(); |  | ||||||
| 	String getOptionDescription(); |  | ||||||
| 	void setOptionValue(boolean value); |  | ||||||
| } |  | ||||||
| @@ -1,38 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.CompletableFuture; |  | ||||||
| import java.util.concurrent.Future; |  | ||||||
|  |  | ||||||
| public interface IUserInterface { |  | ||||||
| 	 |  | ||||||
| 	void show(InputStateHandler handler); |  | ||||||
|  |  | ||||||
| 	void handleException(Exception e); |  | ||||||
|  |  | ||||||
| 	default void handleExceptionAndExit(Exception e) { |  | ||||||
| 		handleException(e); |  | ||||||
| 		System.exit(1); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	default void setTitle(String title) {} |  | ||||||
|  |  | ||||||
| 	void submitProgress(InstallProgress progress); |  | ||||||
|  |  | ||||||
| 	void executeManager(Runnable task); |  | ||||||
|  |  | ||||||
| 	// Return true if the installation was cancelled! |  | ||||||
| 	Future<Boolean> showOptions(List<IOptionDetails> option); |  | ||||||
|  |  | ||||||
| 	Future<IExceptionDetails.ExceptionListResult> showExceptions(List<IExceptionDetails> opts, int numTotal, boolean allowsIgnore); |  | ||||||
|  |  | ||||||
| 	default void disableOptionsButton() {} |  | ||||||
|  |  | ||||||
| 	// Should not return CONTINUE |  | ||||||
| 	default Future<IExceptionDetails.ExceptionListResult> showCancellationDialog() { |  | ||||||
| 		CompletableFuture<IExceptionDetails.ExceptionListResult> future = new CompletableFuture<>(); |  | ||||||
| 		future.complete(IExceptionDetails.ExceptionListResult.CANCEL); |  | ||||||
| 		return future; |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| } |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| public class InputStateHandler { |  | ||||||
| 	private boolean optionsButtonPressed = false; |  | ||||||
| 	private boolean cancelButtonPressed = false; |  | ||||||
|  |  | ||||||
| 	synchronized void pressCancelButton() { |  | ||||||
| 		this.cancelButtonPressed = true; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	synchronized void pressOptionsButton() { |  | ||||||
| 		this.optionsButtonPressed = true; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public synchronized boolean getCancelButton() { |  | ||||||
| 		return cancelButtonPressed; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public synchronized boolean getOptionsButton() { |  | ||||||
| 		return optionsButtonPressed; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| public class InstallProgress { |  | ||||||
| 	public final String message; |  | ||||||
| 	public final boolean hasProgress; |  | ||||||
| 	public final int progress; |  | ||||||
| 	public final int progressTotal; |  | ||||||
|  |  | ||||||
| 	public InstallProgress(String message) { |  | ||||||
| 		this.message = message; |  | ||||||
| 		hasProgress = false; |  | ||||||
| 		progress = 0; |  | ||||||
| 		progressTotal = 0; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public InstallProgress(String message, int progress, int progressTotal) { |  | ||||||
| 		this.message = message; |  | ||||||
| 		hasProgress = true; |  | ||||||
| 		this.progress = progress; |  | ||||||
| 		this.progressTotal = progressTotal; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,228 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import javax.swing.*; |  | ||||||
| import javax.swing.border.EmptyBorder; |  | ||||||
| import java.awt.*; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.CompletableFuture; |  | ||||||
| import java.util.concurrent.Future; |  | ||||||
| import java.util.concurrent.atomic.AtomicBoolean; |  | ||||||
|  |  | ||||||
| public class InstallWindow implements IUserInterface { |  | ||||||
|  |  | ||||||
| 	private JFrame frmPackwizlauncher; |  | ||||||
| 	private JLabel lblProgresslabel; |  | ||||||
| 	private JProgressBar progressBar; |  | ||||||
| 	private InputStateHandler inputStateHandler; |  | ||||||
|  |  | ||||||
| 	private String title = "Updating modpack..."; |  | ||||||
| 	private SwingWorkerButWithPublicPublish<Void, InstallProgress> worker; |  | ||||||
| 	private AtomicBoolean aboutToCrash = new AtomicBoolean(); |  | ||||||
| 	private JButton btnOptions; |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void show(InputStateHandler handler) { |  | ||||||
| 		this.inputStateHandler = handler; |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			try { |  | ||||||
| 				UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); |  | ||||||
| 				initialize(); |  | ||||||
| 				frmPackwizlauncher.setVisible(true); |  | ||||||
| 			} catch (Exception e) { |  | ||||||
| 				e.printStackTrace(); |  | ||||||
| 			} |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Initialize the contents of the frame. |  | ||||||
| 	 * @wbp.parser.entryPoint |  | ||||||
| 	 */ |  | ||||||
| 	private void initialize() { |  | ||||||
| 		frmPackwizlauncher = new JFrame(); |  | ||||||
| 		frmPackwizlauncher.setTitle(title); |  | ||||||
| 		frmPackwizlauncher.setBounds(100, 100, 493, 95); |  | ||||||
| 		frmPackwizlauncher.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); |  | ||||||
| 		frmPackwizlauncher.setLocationRelativeTo(null); |  | ||||||
|  |  | ||||||
| 		JPanel panel = new JPanel(); |  | ||||||
| 		panel.setBorder(new EmptyBorder(10, 10, 10, 10)); |  | ||||||
| 		frmPackwizlauncher.getContentPane().add(panel, BorderLayout.CENTER); |  | ||||||
| 		panel.setLayout(new BorderLayout(0, 0)); |  | ||||||
|  |  | ||||||
| 		progressBar = new JProgressBar(); |  | ||||||
| 		progressBar.setIndeterminate(true); |  | ||||||
| 		panel.add(progressBar, BorderLayout.CENTER); |  | ||||||
|  |  | ||||||
| 		lblProgresslabel = new JLabel("Loading..."); |  | ||||||
| 		panel.add(lblProgresslabel, BorderLayout.SOUTH); |  | ||||||
|  |  | ||||||
| 		JPanel panel_1 = new JPanel(); |  | ||||||
| 		panel_1.setBorder(new EmptyBorder(0, 5, 0, 5)); |  | ||||||
| 		frmPackwizlauncher.getContentPane().add(panel_1, BorderLayout.EAST); |  | ||||||
| 		GridBagLayout gbl_panel_1 = new GridBagLayout(); |  | ||||||
| 		panel_1.setLayout(gbl_panel_1); |  | ||||||
|  |  | ||||||
| 		btnOptions = new JButton("Optional mods..."); |  | ||||||
| 		btnOptions.addActionListener(e -> { |  | ||||||
| 			btnOptions.setText("Loading..."); |  | ||||||
| 			btnOptions.setEnabled(false); |  | ||||||
| 			inputStateHandler.pressOptionsButton(); |  | ||||||
| 		}); |  | ||||||
| 		btnOptions.setAlignmentX(Component.CENTER_ALIGNMENT); |  | ||||||
| 		GridBagConstraints gbc_btnOptions = new GridBagConstraints(); |  | ||||||
| 		gbc_btnOptions.gridx = 0; |  | ||||||
| 		gbc_btnOptions.gridy = 0; |  | ||||||
| 		panel_1.add(btnOptions, gbc_btnOptions); |  | ||||||
|  |  | ||||||
| 		JButton btnCancel = new JButton("Cancel"); |  | ||||||
| 		btnCancel.addActionListener(e -> { |  | ||||||
| 			btnCancel.setEnabled(false); |  | ||||||
| 			inputStateHandler.pressCancelButton(); |  | ||||||
| 		}); |  | ||||||
| 		btnCancel.setAlignmentX(Component.CENTER_ALIGNMENT); |  | ||||||
| 		GridBagConstraints gbc_btnCancel = new GridBagConstraints(); |  | ||||||
| 		gbc_btnCancel.gridx = 0; |  | ||||||
| 		gbc_btnCancel.gridy = 1; |  | ||||||
| 		panel_1.add(btnCancel, gbc_btnCancel); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void handleException(Exception e) { |  | ||||||
| 		e.printStackTrace(); |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			JOptionPane.showMessageDialog(null, "An error occurred: \n" + e.getClass().getCanonicalName() + ": " + e.getMessage(), title, JOptionPane.ERROR_MESSAGE); |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void handleExceptionAndExit(Exception e) { |  | ||||||
| 		e.printStackTrace(); |  | ||||||
| 		// Used to prevent the done() handler of SwingWorker executing if the invokeLater hasn't happened yet |  | ||||||
| 		aboutToCrash.set(true); |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			JOptionPane.showMessageDialog(null, "A fatal error occurred: \n" + e.getClass().getCanonicalName() + ": " + e.getMessage(), title, JOptionPane.ERROR_MESSAGE); |  | ||||||
| 			System.exit(1); |  | ||||||
| 		}); |  | ||||||
| 		// Pause forever, so it blocks while we wait for System.exit to take effect |  | ||||||
| 		try { |  | ||||||
| 			Thread.currentThread().join(); |  | ||||||
| 		} catch (InterruptedException ex) { |  | ||||||
| 			// no u |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void setTitle(String title) { |  | ||||||
| 		this.title = title; |  | ||||||
| 		if (frmPackwizlauncher != null) { |  | ||||||
| 			EventQueue.invokeLater(() -> frmPackwizlauncher.setTitle(title)); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void submitProgress(InstallProgress progress) { |  | ||||||
| 		StringBuilder sb = new StringBuilder(); |  | ||||||
| 		if (progress.hasProgress) { |  | ||||||
| 			sb.append('('); |  | ||||||
| 			sb.append(progress.progress); |  | ||||||
| 			sb.append('/'); |  | ||||||
| 			sb.append(progress.progressTotal); |  | ||||||
| 			sb.append(") "); |  | ||||||
| 		} |  | ||||||
| 		sb.append(progress.message); |  | ||||||
| 		// TODO: better logging library? |  | ||||||
| 		System.out.println(sb.toString()); |  | ||||||
| 		if (worker != null) { |  | ||||||
| 			worker.publishPublic(progress); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void executeManager(Runnable task) { |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			worker = new SwingWorkerButWithPublicPublish<Void, InstallProgress>() { |  | ||||||
|  |  | ||||||
| 				@Override |  | ||||||
| 				protected Void doInBackground() { |  | ||||||
| 					task.run(); |  | ||||||
| 					return null; |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				@Override |  | ||||||
| 				protected void process(List<InstallProgress> chunks) { |  | ||||||
| 					// Only process last chunk |  | ||||||
| 					if (chunks.size() > 0) { |  | ||||||
| 						InstallProgress prog = chunks.get(chunks.size() - 1); |  | ||||||
| 						if (prog.hasProgress) { |  | ||||||
| 							progressBar.setIndeterminate(false); |  | ||||||
| 							progressBar.setValue(prog.progress); |  | ||||||
| 							progressBar.setMaximum(prog.progressTotal); |  | ||||||
| 						} else { |  | ||||||
| 							progressBar.setIndeterminate(true); |  | ||||||
| 							progressBar.setValue(0); |  | ||||||
| 						} |  | ||||||
| 						lblProgresslabel.setText(prog.message); |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				@Override |  | ||||||
| 				protected void done() { |  | ||||||
| 					if (aboutToCrash.get()) { |  | ||||||
| 						return; |  | ||||||
| 					} |  | ||||||
| 					// TODO: a better way to do this? |  | ||||||
| 					frmPackwizlauncher.dispose(); |  | ||||||
| 					System.out.println("Finished successfully!"); |  | ||||||
| 					System.exit(0); |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 			}; |  | ||||||
| 			worker.execute(); |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Future<Boolean> showOptions(List<IOptionDetails> opts) { |  | ||||||
| 		CompletableFuture<Boolean> future = new CompletableFuture<>(); |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			OptionsSelectWindow dialog = new OptionsSelectWindow(opts, future, frmPackwizlauncher); |  | ||||||
| 			dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE); |  | ||||||
| 			dialog.setVisible(true); |  | ||||||
| 		}); |  | ||||||
| 		return future; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Future<IExceptionDetails.ExceptionListResult> showExceptions(List<IExceptionDetails> opts, int numTotal, boolean allowsIgnore) { |  | ||||||
| 		CompletableFuture<IExceptionDetails.ExceptionListResult> future = new CompletableFuture<>(); |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			ExceptionListWindow dialog = new ExceptionListWindow(opts, future, numTotal, allowsIgnore, frmPackwizlauncher); |  | ||||||
| 			dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE); |  | ||||||
| 			dialog.setVisible(true); |  | ||||||
| 		}); |  | ||||||
| 		return future; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void disableOptionsButton() { |  | ||||||
| 		if (btnOptions != null) { |  | ||||||
| 			btnOptions.setText("Optional mods..."); |  | ||||||
| 			btnOptions.setEnabled(false); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public Future<IExceptionDetails.ExceptionListResult> showCancellationDialog() { |  | ||||||
| 		CompletableFuture<IExceptionDetails.ExceptionListResult> future = new CompletableFuture<>(); |  | ||||||
| 		EventQueue.invokeLater(() -> { |  | ||||||
| 			Object[] buttons = {"Quit", "Ignore"}; |  | ||||||
| 			int result = JOptionPane.showOptionDialog(frmPackwizlauncher, |  | ||||||
| 					"The installation was cancelled. Would you like to quit the game, or ignore the update and start the game?", |  | ||||||
| 					"Cancelled installation", |  | ||||||
| 					JOptionPane.YES_NO_OPTION, JOptionPane.QUESTION_MESSAGE, null, buttons, buttons[0]); |  | ||||||
| 			future.complete(result == 0 ? IExceptionDetails.ExceptionListResult.CANCEL : IExceptionDetails.ExceptionListResult.IGNORE); |  | ||||||
| 		}); |  | ||||||
| 		return future; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| // Serves as a proxy for IOptionDetails, so that setOptionValue isn't called until OK is clicked |  | ||||||
| class OptionTempHandler implements IOptionDetails { |  | ||||||
| 	private final IOptionDetails opt; |  | ||||||
| 	private boolean tempValue; |  | ||||||
|  |  | ||||||
| 	OptionTempHandler(IOptionDetails opt) { |  | ||||||
| 		this.opt = opt; |  | ||||||
| 		tempValue = opt.getOptionValue(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getName() { |  | ||||||
| 		return opt.getName(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public String getOptionDescription() { |  | ||||||
| 		return opt.getOptionDescription(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public boolean getOptionValue() { |  | ||||||
| 		return tempValue; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	public void setOptionValue(boolean value) { |  | ||||||
| 		tempValue = value; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void finalise() { |  | ||||||
| 		opt.setOptionValue(tempValue); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,205 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import javax.swing.*; |  | ||||||
| import javax.swing.border.EmptyBorder; |  | ||||||
| import javax.swing.event.ListSelectionEvent; |  | ||||||
| import javax.swing.event.ListSelectionListener; |  | ||||||
| import javax.swing.event.TableModelListener; |  | ||||||
| import javax.swing.table.TableModel; |  | ||||||
| import java.awt.*; |  | ||||||
| import java.awt.event.ActionEvent; |  | ||||||
| import java.awt.event.ActionListener; |  | ||||||
| import java.awt.event.WindowAdapter; |  | ||||||
| import java.awt.event.WindowEvent; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.CompletableFuture; |  | ||||||
|  |  | ||||||
| public class OptionsSelectWindow extends JDialog implements ActionListener { |  | ||||||
|  |  | ||||||
| 	private static final long serialVersionUID = 1L; |  | ||||||
| 	private final JTextArea lblOptionDescription; |  | ||||||
| 	private final OptionTableModel tableModel; |  | ||||||
| 	private final CompletableFuture<Boolean> future; |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Create the dialog. |  | ||||||
| 	 */ |  | ||||||
| 	OptionsSelectWindow(List<IOptionDetails> optList, CompletableFuture<Boolean> future, JFrame parentWindow) { |  | ||||||
| 		super(parentWindow, "Select optional mods...", true); |  | ||||||
|  |  | ||||||
| 		tableModel = new OptionTableModel(optList); |  | ||||||
| 		this.future = future; |  | ||||||
|  |  | ||||||
| 		setBounds(100, 100, 450, 300); |  | ||||||
| 		setLocationRelativeTo(parentWindow); |  | ||||||
| 		getContentPane().setLayout(new BorderLayout()); |  | ||||||
| 		JPanel contentPanel = new JPanel(); |  | ||||||
| 		contentPanel.setBorder(new EmptyBorder(5, 5, 5, 5)); |  | ||||||
| 		getContentPane().add(contentPanel, BorderLayout.CENTER); |  | ||||||
| 		contentPanel.setLayout(new BorderLayout(0, 0)); |  | ||||||
| 		{ |  | ||||||
| 			JSplitPane splitPane = new JSplitPane(); |  | ||||||
| 			splitPane.setResizeWeight(0.5); |  | ||||||
| 			contentPanel.add(splitPane); |  | ||||||
| 			{ |  | ||||||
| 				JTable table = new JTable(); |  | ||||||
| 				table.setShowVerticalLines(false); |  | ||||||
| 				table.setShowHorizontalLines(false); |  | ||||||
| 				table.setSelectionMode(ListSelectionModel.SINGLE_SELECTION); |  | ||||||
| 				table.setShowGrid(false); |  | ||||||
| 				table.setModel(tableModel); |  | ||||||
| 				table.getColumnModel().getColumn(0).setResizable(false); |  | ||||||
| 				table.getColumnModel().getColumn(0).setPreferredWidth(15); |  | ||||||
| 				table.getColumnModel().getColumn(0).setMaxWidth(15); |  | ||||||
| 				table.getColumnModel().getColumn(1).setResizable(false); |  | ||||||
| 				table.getSelectionModel().addListSelectionListener(new ListSelectionListener() { |  | ||||||
| 					@Override |  | ||||||
| 					public void valueChanged(ListSelectionEvent e) { |  | ||||||
| 						int i = table.getSelectedRow(); |  | ||||||
| 						if (i > -1) { |  | ||||||
| 							lblOptionDescription.setText(tableModel.getDescription(i)); |  | ||||||
| 						} else { |  | ||||||
| 							lblOptionDescription.setText("Select an option..."); |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				}); |  | ||||||
| 				table.setTableHeader(null); |  | ||||||
| 				JScrollPane scrollPane = new JScrollPane(table); |  | ||||||
| 				scrollPane.getViewport().setBackground(UIManager.getColor("List.background")); |  | ||||||
| 				scrollPane.setHorizontalScrollBarPolicy(ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER); |  | ||||||
| 				scrollPane.setBorder(new EmptyBorder(0, 0, 0, 0)); |  | ||||||
| 				splitPane.setLeftComponent(scrollPane); |  | ||||||
| 			} |  | ||||||
| 			{ |  | ||||||
| 				lblOptionDescription = new JTextArea("Select an option..."); |  | ||||||
| 				lblOptionDescription.setBackground(UIManager.getColor("List.background")); |  | ||||||
| 				lblOptionDescription.setOpaque(true); |  | ||||||
| 				lblOptionDescription.setWrapStyleWord(true); |  | ||||||
| 				lblOptionDescription.setLineWrap(true); |  | ||||||
| 				lblOptionDescription.setEditable(false); |  | ||||||
| 				lblOptionDescription.setFocusable(false); |  | ||||||
| 				lblOptionDescription.setFont(UIManager.getFont("Label.font")); |  | ||||||
| 				lblOptionDescription.setBorder(new EmptyBorder(10, 10, 10, 10)); |  | ||||||
| 				JScrollPane scrollPane = new JScrollPane(lblOptionDescription); |  | ||||||
| 				scrollPane.setBorder(new EmptyBorder(0, 0, 0, 0)); |  | ||||||
| 				splitPane.setRightComponent(scrollPane); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		{ |  | ||||||
| 			JPanel buttonPane = new JPanel(); |  | ||||||
| 			buttonPane.setLayout(new FlowLayout(FlowLayout.RIGHT)); |  | ||||||
| 			getContentPane().add(buttonPane, BorderLayout.SOUTH); |  | ||||||
| 			{ |  | ||||||
| 				JButton okButton = new JButton("OK"); |  | ||||||
| 				okButton.setActionCommand("OK"); |  | ||||||
| 				okButton.addActionListener(this); |  | ||||||
| 				buttonPane.add(okButton); |  | ||||||
| 				getRootPane().setDefaultButton(okButton); |  | ||||||
| 			} |  | ||||||
| 			{ |  | ||||||
| 				JButton cancelButton = new JButton("Cancel"); |  | ||||||
| 				cancelButton.setActionCommand("Cancel"); |  | ||||||
| 				cancelButton.addActionListener(this); |  | ||||||
| 				buttonPane.add(cancelButton); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		addWindowListener(new WindowAdapter() { |  | ||||||
| 			@Override |  | ||||||
| 			public void windowClosing(WindowEvent e) { |  | ||||||
| 				future.complete(true); |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			@Override |  | ||||||
| 			public void windowClosed(WindowEvent e) { |  | ||||||
| 				// Just in case closing didn't get triggered - if something else called dispose() the |  | ||||||
| 				// future will have already completed |  | ||||||
| 				future.complete(true); |  | ||||||
| 			} |  | ||||||
| 		}); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	private static class OptionTableModel implements TableModel { |  | ||||||
| 		private List<OptionTempHandler> opts = new ArrayList<>(); |  | ||||||
|  |  | ||||||
| 		OptionTableModel(List<IOptionDetails> givenOpts) { |  | ||||||
| 			for (IOptionDetails opt : givenOpts) { |  | ||||||
| 				opts.add(new OptionTempHandler(opt)); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public int getRowCount() { |  | ||||||
| 			return opts.size(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public int getColumnCount() { |  | ||||||
| 			return 2; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		private final String[] columnNames = {"Enabled", "Mod name"}; |  | ||||||
| 		private final Class<?>[] columnTypes = {Boolean.class, String.class}; |  | ||||||
| 		private final boolean[] columnEditables = {true, false}; |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public String getColumnName(int columnIndex) { |  | ||||||
| 			return columnNames[columnIndex]; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public Class<?> getColumnClass(int columnIndex) { |  | ||||||
| 			return columnTypes[columnIndex]; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public boolean isCellEditable(int rowIndex, int columnIndex) { |  | ||||||
| 			return columnEditables[columnIndex]; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public Object getValueAt(int rowIndex, int columnIndex) { |  | ||||||
| 			OptionTempHandler opt = opts.get(rowIndex); |  | ||||||
| 			return columnIndex == 0 ? opt.getOptionValue() : opt.getName(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public void setValueAt(Object aValue, int rowIndex, int columnIndex) { |  | ||||||
| 			if (columnIndex == 0) { |  | ||||||
| 				OptionTempHandler opt = opts.get(rowIndex); |  | ||||||
| 				opt.setOptionValue((boolean) aValue); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Noop, the table model doesn't change! |  | ||||||
| 		@Override |  | ||||||
| 		public void addTableModelListener(TableModelListener l) {} |  | ||||||
|  |  | ||||||
| 		@Override |  | ||||||
| 		public void removeTableModelListener(TableModelListener l) {} |  | ||||||
|  |  | ||||||
| 		String getDescription(int rowIndex) { |  | ||||||
| 			return opts.get(rowIndex).getOptionDescription(); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		void finalise() { |  | ||||||
| 			for (OptionTempHandler opt : opts) { |  | ||||||
| 				opt.finalise(); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@Override |  | ||||||
| 	public void actionPerformed(ActionEvent e) { |  | ||||||
| 		if (e.getActionCommand().equals("OK")) { |  | ||||||
| 			tableModel.finalise(); |  | ||||||
| 			future.complete(false); |  | ||||||
| 			dispose(); |  | ||||||
| 		} else if (e.getActionCommand().equals("Cancel")) { |  | ||||||
| 			future.complete(true); |  | ||||||
| 			dispose(); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| package link.infra.packwiz.installer.ui; |  | ||||||
|  |  | ||||||
| import javax.swing.SwingWorker; |  | ||||||
|  |  | ||||||
| // Q: AAA WHAT HAVE YOU DONE THIS IS DISGUSTING |  | ||||||
| // A: it just makes things easier, so i can easily have one interface for CLI/GUI |  | ||||||
| //    if someone has a better way to do this please PR it |  | ||||||
| public abstract class SwingWorkerButWithPublicPublish<T,V> extends SwingWorker<T,V> { |  | ||||||
| 	@SafeVarargs |  | ||||||
| 	public final void publishPublic(V... chunks) { |  | ||||||
| 		publish(chunks); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
							
								
								
									
										252
									
								
								src/main/kotlin/link/infra/packwiz/installer/DownloadTask.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								src/main/kotlin/link/infra/packwiz/installer/DownloadTask.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,252 @@ | |||||||
|  | package link.infra.packwiz.installer | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.IndexFile | ||||||
|  | import link.infra.packwiz.installer.metadata.ManifestFile | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.Hash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHasher | ||||||
|  | import link.infra.packwiz.installer.ui.data.ExceptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  | import link.infra.packwiz.installer.util.Log | ||||||
|  | import okio.Buffer | ||||||
|  | import okio.HashingSink | ||||||
|  | import okio.buffer | ||||||
|  | import java.io.IOException | ||||||
|  | import java.nio.file.Files | ||||||
|  | import java.nio.file.Paths | ||||||
|  | import java.nio.file.StandardCopyOption | ||||||
|  | import java.util.* | ||||||
|  |  | ||||||
|  | internal class DownloadTask private constructor(val metadata: IndexFile.File, defaultFormat: String, private val downloadSide: UpdateManager.Options.Side) : IOptionDetails { | ||||||
|  | 	var cachedFile: ManifestFile.File? = null | ||||||
|  |  | ||||||
|  | 	private var err: Exception? = null | ||||||
|  | 	val exceptionDetails get() = err?.let { e -> ExceptionDetails(name, e) } | ||||||
|  |  | ||||||
|  | 	fun failed() = err != null | ||||||
|  |  | ||||||
|  | 	private var alreadyUpToDate = false | ||||||
|  | 	private var metadataRequired = true | ||||||
|  | 	private var invalidated = false | ||||||
|  | 	// If file is new or isOptional changed to true, the option needs to be presented again | ||||||
|  | 	private var newOptional = true | ||||||
|  |  | ||||||
|  | 	val isOptional get() = metadata.linkedFile?.isOptional ?: false | ||||||
|  |  | ||||||
|  | 	fun isNewOptional() = isOptional && newOptional | ||||||
|  |  | ||||||
|  | 	fun correctSide() = metadata.linkedFile?.side?.hasSide(downloadSide) ?: true | ||||||
|  |  | ||||||
|  | 	override val name get() = metadata.name | ||||||
|  |  | ||||||
|  | 	// Ensure that an update is done if it changes from false to true, or from true to false | ||||||
|  | 	override var optionValue: Boolean | ||||||
|  | 		get() = cachedFile?.optionValue ?: true | ||||||
|  | 		set(value) { | ||||||
|  | 			if (value && !optionValue) { // Ensure that an update is done if it changes from false to true, or from true to false | ||||||
|  | 				alreadyUpToDate = false | ||||||
|  | 			} | ||||||
|  | 			cachedFile?.optionValue = value | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	override val optionDescription get() = metadata.linkedFile?.option?.description ?: "" | ||||||
|  |  | ||||||
|  | 	init { | ||||||
|  | 		if (metadata.hashFormat?.isEmpty() != false) { | ||||||
|  | 			metadata.hashFormat = defaultFormat | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun invalidate() { | ||||||
|  | 		invalidated = true | ||||||
|  | 		alreadyUpToDate = false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun updateFromCache(cachedFile: ManifestFile.File?) { | ||||||
|  | 		if (err != null) return | ||||||
|  |  | ||||||
|  | 		if (cachedFile == null) { | ||||||
|  | 			this.cachedFile = ManifestFile.File() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		this.cachedFile = cachedFile | ||||||
|  | 		if (!invalidated) { | ||||||
|  | 			val currHash = try { | ||||||
|  | 				getHash(metadata.hashFormat!!, metadata.hash!!) | ||||||
|  | 			} catch (e: Exception) { | ||||||
|  | 				err = e | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			if (currHash == cachedFile.hash) { // Already up to date | ||||||
|  | 				alreadyUpToDate = true | ||||||
|  | 				metadataRequired = false | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (cachedFile.isOptional) { | ||||||
|  | 			// Because option selection dialog might set this task to true/false, metadata is always needed to download | ||||||
|  | 			// the file, and to show the description and name | ||||||
|  | 			metadataRequired = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun downloadMetadata(parentIndexFile: IndexFile, indexUri: SpaceSafeURI) { | ||||||
|  | 		if (err != null) return | ||||||
|  |  | ||||||
|  | 		if (metadataRequired) { | ||||||
|  | 			try { | ||||||
|  | 				// Retrieve the linked metadata file | ||||||
|  | 				metadata.downloadMeta(parentIndexFile, indexUri) | ||||||
|  | 			} catch (e: Exception) { | ||||||
|  | 				err = e | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			cachedFile?.let { cachedFile -> | ||||||
|  | 				val linkedFile = metadata.linkedFile | ||||||
|  | 				if (linkedFile != null) { | ||||||
|  | 					linkedFile.option?.let { opt -> | ||||||
|  | 						if (opt.optional) { | ||||||
|  | 							if (cachedFile.isOptional) { | ||||||
|  | 								// isOptional didn't change | ||||||
|  | 								newOptional = false | ||||||
|  | 							} else { | ||||||
|  | 								// isOptional false -> true, set option to it's default value | ||||||
|  | 								// TODO: preserve previous option value, somehow?? | ||||||
|  | 								cachedFile.optionValue = opt.defaultValue | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 					cachedFile.isOptional = isOptional | ||||||
|  | 					cachedFile.onlyOtherSide = !correctSide() | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun download(packFolder: String, indexUri: SpaceSafeURI) { | ||||||
|  | 		if (err != null) return | ||||||
|  |  | ||||||
|  | 		// TODO: is this necessary if we overwrite? | ||||||
|  | 		// Ensure it is removed | ||||||
|  | 		cachedFile?.let { | ||||||
|  | 			if (!it.optionValue || !correctSide()) { | ||||||
|  | 				if (it.cachedLocation == null) return | ||||||
|  |  | ||||||
|  | 				try { | ||||||
|  | 					Files.deleteIfExists(Paths.get(packFolder, it.cachedLocation)) | ||||||
|  | 				} catch (e: IOException) { | ||||||
|  | 					Log.warn("Failed to delete file before downloading", e) | ||||||
|  | 				} | ||||||
|  | 				it.cachedLocation = null | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (alreadyUpToDate) return | ||||||
|  |  | ||||||
|  | 		// TODO: should I be validating JSON properly, or this fine!!!!!!!?? | ||||||
|  | 		assert(metadata.destURI != null) | ||||||
|  | 		val destPath = Paths.get(packFolder, metadata.destURI.toString()) | ||||||
|  |  | ||||||
|  | 		// Don't update files marked with preserve if they already exist on disk | ||||||
|  | 		if (metadata.preserve) { | ||||||
|  | 			if (destPath.toFile().exists()) { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// TODO: if already exists and has correct hash, ignore? | ||||||
|  | 		// TODO: add .disabled support? | ||||||
|  |  | ||||||
|  | 		try { | ||||||
|  | 			val hash: Hash | ||||||
|  | 			val fileHashFormat: String | ||||||
|  | 			val linkedFile = metadata.linkedFile | ||||||
|  |  | ||||||
|  | 			if (linkedFile != null) { | ||||||
|  | 				hash = linkedFile.hash | ||||||
|  | 				fileHashFormat = Objects.requireNonNull(Objects.requireNonNull(linkedFile.download)!!.hashFormat)!! | ||||||
|  | 			} else { | ||||||
|  | 				hash = metadata.getHashObj() | ||||||
|  | 				fileHashFormat = Objects.requireNonNull(metadata.hashFormat)!! | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			val src = metadata.getSource(indexUri) | ||||||
|  | 			val fileSource = getHasher(fileHashFormat).getHashingSource(src) | ||||||
|  | 			val data = Buffer() | ||||||
|  |  | ||||||
|  | 			// Read all the data into a buffer (very inefficient for large files! but allows rollback if hash check fails) | ||||||
|  | 			// TODO: should we instead rename the existing file, then stream straight to the file and rollback from the renamed file? | ||||||
|  | 			fileSource.buffer().use { | ||||||
|  | 				it.readAll(data) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (fileSource.hashIsEqual(hash)) { | ||||||
|  | 				// isDirectory follows symlinks, but createDirectories doesn't | ||||||
|  | 				try { | ||||||
|  | 					Files.createDirectories(destPath.parent) | ||||||
|  | 				} catch (e: java.nio.file.FileAlreadyExistsException) { | ||||||
|  | 					if (!Files.isDirectory(destPath.parent)) { | ||||||
|  | 						throw e | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				Files.copy(data.inputStream(), destPath, StandardCopyOption.REPLACE_EXISTING) | ||||||
|  | 				data.clear() | ||||||
|  | 			} else { | ||||||
|  | 				// TODO: move println to something visible in the error window | ||||||
|  | 				println("Invalid hash for " + metadata.destURI.toString()) | ||||||
|  | 				println("Calculated: " + fileSource.hash) | ||||||
|  | 				println("Expected:   $hash") | ||||||
|  | 				// Attempt to get the SHA256 hash | ||||||
|  | 				val sha256 = HashingSink.sha256(okio.blackholeSink()) | ||||||
|  | 				data.readAll(sha256) | ||||||
|  | 				println("SHA256 hash value: " + sha256.hash) | ||||||
|  | 				err = Exception("Hash invalid!") | ||||||
|  | 				data.clear() | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			cachedFile?.cachedLocation?.let { | ||||||
|  | 				if (destPath != Paths.get(packFolder, it)) { | ||||||
|  | 					// Delete old file if location changes | ||||||
|  | 					try { | ||||||
|  | 						Files.delete(Paths.get(packFolder, cachedFile!!.cachedLocation)) | ||||||
|  | 					} catch (e: IOException) { | ||||||
|  | 						// Continue, as it was probably already deleted? | ||||||
|  | 						// TODO: log it | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} catch (e: Exception) { | ||||||
|  | 			err = e | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Update the manifest file | ||||||
|  | 		cachedFile = (cachedFile ?: ManifestFile.File()).also { | ||||||
|  | 			try { | ||||||
|  | 				it.hash = metadata.getHashObj() | ||||||
|  | 			} catch (e: Exception) { | ||||||
|  | 				err = e | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			it.isOptional = isOptional | ||||||
|  | 			it.cachedLocation = metadata.destURI.toString() | ||||||
|  | 			metadata.linkedFile?.let { linked -> | ||||||
|  | 				try { | ||||||
|  | 					it.linkedFileHash = linked.hash | ||||||
|  | 				} catch (e: Exception) { | ||||||
|  | 					err = e | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	companion object { | ||||||
|  | 		@JvmStatic | ||||||
|  | 		fun createTasksFromIndex(index: IndexFile, defaultFormat: String, downloadSide: UpdateManager.Options.Side): List<DownloadTask> { | ||||||
|  | 			val tasks = ArrayList<DownloadTask>() | ||||||
|  | 			for (file in Objects.requireNonNull(index.files)) { | ||||||
|  | 				tasks.add(DownloadTask(file, defaultFormat, downloadSide)) | ||||||
|  | 			} | ||||||
|  | 			return tasks | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										131
									
								
								src/main/kotlin/link/infra/packwiz/installer/Main.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								src/main/kotlin/link/infra/packwiz/installer/Main.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | |||||||
|  | @file:JvmName("Main") | ||||||
|  |  | ||||||
|  | package link.infra.packwiz.installer | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.ui.cli.CLIHandler | ||||||
|  | import link.infra.packwiz.installer.ui.gui.GUIHandler | ||||||
|  | import link.infra.packwiz.installer.util.Log | ||||||
|  | import org.apache.commons.cli.DefaultParser | ||||||
|  | import org.apache.commons.cli.Options | ||||||
|  | import org.apache.commons.cli.ParseException | ||||||
|  | import java.awt.EventQueue | ||||||
|  | import java.awt.GraphicsEnvironment | ||||||
|  | import java.net.URISyntaxException | ||||||
|  | import javax.swing.JOptionPane | ||||||
|  | import javax.swing.UIManager | ||||||
|  | import kotlin.system.exitProcess | ||||||
|  |  | ||||||
|  | @Suppress("unused") | ||||||
|  | class Main(args: Array<String>) { | ||||||
|  | 	// Don't attempt to start a GUI if we are headless | ||||||
|  | 	private var guiEnabled = !GraphicsEnvironment.isHeadless() | ||||||
|  |  | ||||||
|  | 	private fun startup(args: Array<String>) { | ||||||
|  | 		val options = Options() | ||||||
|  | 		addNonBootstrapOptions(options) | ||||||
|  | 		addBootstrapOptions(options) | ||||||
|  |  | ||||||
|  | 		val parser = DefaultParser() | ||||||
|  | 		val cmd = try { | ||||||
|  | 			parser.parse(options, args) | ||||||
|  | 		} catch (e: ParseException) { | ||||||
|  | 			Log.fatal("Failed to parse command line arguments", e) | ||||||
|  | 			if (guiEnabled) { | ||||||
|  | 				EventQueue.invokeAndWait { | ||||||
|  | 					try { | ||||||
|  | 						UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()) | ||||||
|  | 					} catch (ignored: Exception) { | ||||||
|  | 						// Ignore the exceptions, just continue using the ugly L&F | ||||||
|  | 					} | ||||||
|  | 					JOptionPane.showMessageDialog(null, "Failed to parse command line arguments: $e", | ||||||
|  | 						"packwiz-installer", JOptionPane.ERROR_MESSAGE) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			exitProcess(1) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (guiEnabled && cmd.hasOption("no-gui")) { | ||||||
|  | 			guiEnabled = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		val ui = if (guiEnabled) GUIHandler() else CLIHandler() | ||||||
|  |  | ||||||
|  | 		val unparsedArgs = cmd.args | ||||||
|  | 		if (unparsedArgs.size > 1) { | ||||||
|  | 			ui.showErrorAndExit("Too many arguments specified!") | ||||||
|  | 		} else if (unparsedArgs.isEmpty()) { | ||||||
|  | 			ui.showErrorAndExit("pack.toml URI to install from must be specified!") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		val title = cmd.getOptionValue("title") | ||||||
|  | 		if (title != null) { | ||||||
|  | 			ui.title = title | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ui.show() | ||||||
|  |  | ||||||
|  | 		val uOptions = try { | ||||||
|  | 			UpdateManager.Options.construct( | ||||||
|  | 				downloadURI = SpaceSafeURI(unparsedArgs[0]), | ||||||
|  | 				side = cmd.getOptionValue("side")?.let((UpdateManager.Options.Side)::from), | ||||||
|  | 				packFolder = cmd.getOptionValue("pack-folder"), | ||||||
|  | 				manifestFile = cmd.getOptionValue("meta-file") | ||||||
|  | 			) | ||||||
|  | 		} catch (e: URISyntaxException) { | ||||||
|  | 			ui.showErrorAndExit("Failed to read pack.toml URI", e) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Start update process! | ||||||
|  | 		try { | ||||||
|  | 			UpdateManager(uOptions, ui) | ||||||
|  | 		} catch (e: Exception) { | ||||||
|  | 			ui.showErrorAndExit("Update process failed", e) | ||||||
|  | 		} | ||||||
|  | 		println("Finished successfully!") | ||||||
|  | 		ui.dispose() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	companion object { | ||||||
|  | 		// Called by packwiz-installer-bootstrap to set up the help command | ||||||
|  | 		@JvmStatic | ||||||
|  | 		fun addNonBootstrapOptions(options: Options) { | ||||||
|  | 			options.addOption("s", "side", true, "Side to install mods from (client/server, defaults to client)") | ||||||
|  | 			options.addOption(null, "title", true, "Title of the installer window") | ||||||
|  | 			options.addOption(null, "pack-folder", true, "Folder to install the pack to (defaults to the JAR directory)") | ||||||
|  | 			options.addOption(null, "meta-file", true, "JSON file to store pack metadata, relative to the pack folder (defaults to packwiz.json)") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// TODO: link these somehow so they're only defined once? | ||||||
|  | 		@JvmStatic | ||||||
|  | 		private fun addBootstrapOptions(options: Options) { | ||||||
|  | 			options.addOption(null, "bootstrap-update-url", true, "Github API URL for checking for updates") | ||||||
|  | 			options.addOption(null, "bootstrap-update-token", true, "Github API Access Token, for private repositories") | ||||||
|  | 			options.addOption(null, "bootstrap-no-update", false, "Don't update packwiz-installer") | ||||||
|  | 			options.addOption(null, "bootstrap-main-jar", true, "Location of the packwiz-installer JAR file") | ||||||
|  | 			options.addOption("g", "no-gui", false, "Don't display a GUI to show update progress") | ||||||
|  | 			options.addOption("h", "help", false, "Display this message") // Implemented in packwiz-installer-bootstrap! | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Actual main() is in RequiresBootstrap! | ||||||
|  | 	init { | ||||||
|  | 		// Big overarching try/catch just in case everything breaks | ||||||
|  | 		try { | ||||||
|  | 			startup(args) | ||||||
|  | 		} catch (e: Exception) { | ||||||
|  | 			Log.fatal("Error from main", e) | ||||||
|  | 			if (guiEnabled) { | ||||||
|  | 				EventQueue.invokeLater { | ||||||
|  | 					JOptionPane.showMessageDialog(null, | ||||||
|  | 						"A fatal error occurred: \n$e", | ||||||
|  | 						"packwiz-installer", JOptionPane.ERROR_MESSAGE) | ||||||
|  | 					exitProcess(1) | ||||||
|  | 				} | ||||||
|  | 				// In case the EventQueue is broken, exit after 1 minute | ||||||
|  | 				Thread.sleep(60 * 1000.toLong()) | ||||||
|  | 			} | ||||||
|  | 			exitProcess(1) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										447
									
								
								src/main/kotlin/link/infra/packwiz/installer/UpdateManager.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										447
									
								
								src/main/kotlin/link/infra/packwiz/installer/UpdateManager.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,447 @@ | |||||||
|  | package link.infra.packwiz.installer | ||||||
|  |  | ||||||
|  | import com.google.gson.GsonBuilder | ||||||
|  | import com.google.gson.JsonIOException | ||||||
|  | import com.google.gson.JsonSyntaxException | ||||||
|  | import com.google.gson.annotations.SerializedName | ||||||
|  | import com.moandjiezana.toml.Toml | ||||||
|  | import link.infra.packwiz.installer.DownloadTask.Companion.createTasksFromIndex | ||||||
|  | import link.infra.packwiz.installer.metadata.IndexFile | ||||||
|  | import link.infra.packwiz.installer.metadata.ManifestFile | ||||||
|  | import link.infra.packwiz.installer.metadata.PackFile | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.Hash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHasher | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getFileSource | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getNewLoc | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface.CancellationResult | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface.ExceptionListResult | ||||||
|  | import link.infra.packwiz.installer.ui.data.InstallProgress | ||||||
|  | import link.infra.packwiz.installer.util.Log | ||||||
|  | import link.infra.packwiz.installer.util.ifletOrErr | ||||||
|  | import okio.buffer | ||||||
|  | import java.io.FileNotFoundException | ||||||
|  | import java.io.FileReader | ||||||
|  | import java.io.FileWriter | ||||||
|  | import java.io.IOException | ||||||
|  | import java.nio.file.Files | ||||||
|  | import java.nio.file.Paths | ||||||
|  | import java.util.concurrent.CompletionService | ||||||
|  | import java.util.concurrent.ExecutionException | ||||||
|  | import java.util.concurrent.ExecutorCompletionService | ||||||
|  | import java.util.concurrent.Executors | ||||||
|  | import kotlin.system.exitProcess | ||||||
|  |  | ||||||
|  | class UpdateManager internal constructor(private val opts: Options, val ui: IUserInterface) { | ||||||
|  | 	private var cancelled = false | ||||||
|  | 	private var cancelledStartGame = false | ||||||
|  | 	private var errorsOccurred = false | ||||||
|  |  | ||||||
|  | 	init { | ||||||
|  | 		start() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	data class Options( | ||||||
|  | 		val downloadURI: SpaceSafeURI, | ||||||
|  | 		val manifestFile: String, | ||||||
|  | 		val packFolder: String, | ||||||
|  | 		val side: Side | ||||||
|  | 	) { | ||||||
|  | 		// Horrible workaround for default params not working cleanly with nullable values | ||||||
|  | 		companion object { | ||||||
|  | 			fun construct(downloadURI: SpaceSafeURI, manifestFile: String?, packFolder: String?, side: Side?) = | ||||||
|  | 				Options(downloadURI, manifestFile ?: "packwiz.json", packFolder ?: ".", side ?: Side.CLIENT) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		enum class Side { | ||||||
|  | 			@SerializedName("client") | ||||||
|  | 			CLIENT("client"), | ||||||
|  | 			@SerializedName("server") | ||||||
|  | 			SERVER("server"), | ||||||
|  | 			@SerializedName("both") | ||||||
|  | 			@Suppress("unused") | ||||||
|  | 			BOTH("both", arrayOf(CLIENT, SERVER)); | ||||||
|  |  | ||||||
|  | 			private val sideName: String | ||||||
|  | 			private val depSides: Array<Side>? | ||||||
|  |  | ||||||
|  | 			constructor(sideName: String) { | ||||||
|  | 				this.sideName = sideName.toLowerCase() | ||||||
|  | 				depSides = null | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			constructor(sideName: String, depSides: Array<Side>) { | ||||||
|  | 				this.sideName = sideName.toLowerCase() | ||||||
|  | 				this.depSides = depSides | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			override fun toString() = sideName | ||||||
|  |  | ||||||
|  | 			fun hasSide(tSide: Side): Boolean { | ||||||
|  | 				if (this == tSide) { | ||||||
|  | 					return true | ||||||
|  | 				} | ||||||
|  | 				if (depSides != null) { | ||||||
|  | 					for (depSide in depSides) { | ||||||
|  | 						if (depSide == tSide) { | ||||||
|  | 							return true | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				return false | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			companion object { | ||||||
|  | 				fun from(name: String): Side? { | ||||||
|  | 					val lowerName = name.toLowerCase() | ||||||
|  | 					for (side in values()) { | ||||||
|  | 						if (side.sideName == lowerName) { | ||||||
|  | 							return side | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 					return null | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private fun start() { | ||||||
|  | 		checkOptions() | ||||||
|  |  | ||||||
|  | 		ui.submitProgress(InstallProgress("Loading manifest file...")) | ||||||
|  | 		val gson = GsonBuilder().registerTypeAdapter(Hash::class.java, Hash.TypeHandler()).create() | ||||||
|  | 		val manifest = try { | ||||||
|  | 			gson.fromJson(FileReader(Paths.get(opts.packFolder, opts.manifestFile).toString()), | ||||||
|  | 					ManifestFile::class.java) | ||||||
|  | 		} catch (e: FileNotFoundException) { | ||||||
|  | 			ui.firstInstall = true | ||||||
|  | 			ManifestFile() | ||||||
|  | 		} catch (e: JsonSyntaxException) { | ||||||
|  | 			ui.showErrorAndExit("Invalid local manifest file, try deleting ${opts.manifestFile}", e) | ||||||
|  | 		} catch (e: JsonIOException) { | ||||||
|  | 			ui.showErrorAndExit("Failed to read local manifest file, try deleting ${opts.manifestFile}", e) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			handleCancellation() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ui.submitProgress(InstallProgress("Loading pack file...")) | ||||||
|  | 		val packFileSource = try { | ||||||
|  | 			val src = getFileSource(opts.downloadURI) | ||||||
|  | 			getHasher("sha256").getHashingSource(src) | ||||||
|  | 		} catch (e: Exception) { | ||||||
|  | 			ui.showErrorAndExit("Failed to download pack.toml", e) | ||||||
|  | 		} | ||||||
|  | 		val pf = packFileSource.buffer().use { | ||||||
|  | 			try { | ||||||
|  | 				Toml().read(it.inputStream()).to(PackFile::class.java) | ||||||
|  | 			} catch (e: IllegalStateException) { | ||||||
|  | 				ui.showErrorAndExit("Failed to parse pack.toml", e) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			handleCancellation() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ui.submitProgress(InstallProgress("Checking local files...")) | ||||||
|  |  | ||||||
|  | 		// Invalidation checking must be done here, as it must happen before pack/index hashes are checked | ||||||
|  | 		val invalidatedUris: MutableList<SpaceSafeURI> = ArrayList() | ||||||
|  | 		for ((fileUri, file) in manifest.cachedFiles) { | ||||||
|  | 			// ignore onlyOtherSide files | ||||||
|  | 			if (file.onlyOtherSide) { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			var invalid = false | ||||||
|  | 			// if isn't optional, or is optional but optionValue == true | ||||||
|  | 			if (!file.isOptional || file.optionValue) { | ||||||
|  | 				if (file.cachedLocation != null) { | ||||||
|  | 					if (!Paths.get(opts.packFolder, file.cachedLocation).toFile().exists()) { | ||||||
|  | 						invalid = true | ||||||
|  | 					} | ||||||
|  | 				} else { | ||||||
|  | 					// if cachedLocation == null, should probably be installed!! | ||||||
|  | 					invalid = true | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			if (invalid) { | ||||||
|  | 				Log.info("File $fileUri invalidated, marked for redownloading") | ||||||
|  | 				invalidatedUris.add(fileUri) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (manifest.packFileHash?.let { packFileSource.hashIsEqual(it) } == true && invalidatedUris.isEmpty()) { | ||||||
|  | 			Log.info("Modpack is already up to date!") | ||||||
|  | 			// todo: --force? | ||||||
|  | 			if (!ui.optionsButtonPressed) { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		Log.info("Modpack name: ${pf.name}") | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			handleCancellation() | ||||||
|  | 		} | ||||||
|  | 		try { | ||||||
|  | 			// TODO: switch to OkHttp for better redirect handling | ||||||
|  | 			ui.ifletOrErr(pf.index, "No index file found, or the pack file is empty; note that Java doesn't automatically follow redirects from HTTP to HTTPS (and may cause this error)") { index -> | ||||||
|  | 				ui.ifletOrErr(index.hashFormat, index.hash, "Pack has no hash or hashFormat for index") { hashFormat, hash -> | ||||||
|  | 					ui.ifletOrErr(getNewLoc(opts.downloadURI, index.file), "Pack has invalid index file: " + index.file) { newLoc -> | ||||||
|  | 						processIndex( | ||||||
|  | 							newLoc, | ||||||
|  | 							getHash(hashFormat, hash), | ||||||
|  | 							hashFormat, | ||||||
|  | 							manifest, | ||||||
|  | 							invalidatedUris | ||||||
|  | 						) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} catch (e1: Exception) { | ||||||
|  | 			ui.showErrorAndExit("Failed to process index file", e1) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		handleCancellation() | ||||||
|  |  | ||||||
|  | 		// TODO: update MMC params, java args etc | ||||||
|  |  | ||||||
|  | 		// If there were errors, don't write the manifest/index hashes, to ensure they are rechecked later | ||||||
|  | 		if (errorsOccurred) { | ||||||
|  | 			manifest.indexFileHash = null | ||||||
|  | 			manifest.packFileHash = null | ||||||
|  | 		} else { | ||||||
|  | 			manifest.packFileHash = packFileSource.hash | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		manifest.cachedSide = opts.side | ||||||
|  | 		try { | ||||||
|  | 			FileWriter(Paths.get(opts.packFolder, opts.manifestFile).toString()).use { writer -> gson.toJson(manifest, writer) } | ||||||
|  | 		} catch (e: IOException) { | ||||||
|  | 			ui.showErrorAndExit("Failed to save local manifest file", e) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private fun checkOptions() { | ||||||
|  | 		// TODO: implement | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private fun processIndex(indexUri: SpaceSafeURI, indexHash: Hash, hashFormat: String, manifest: ManifestFile, invalidatedUris: List<SpaceSafeURI>) { | ||||||
|  | 		if (manifest.indexFileHash == indexHash && invalidatedUris.isEmpty()) { | ||||||
|  | 			Log.info("Modpack files are already up to date!") | ||||||
|  | 			if (!ui.optionsButtonPressed) { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		manifest.indexFileHash = indexHash | ||||||
|  |  | ||||||
|  | 		val indexFileSource = try { | ||||||
|  | 			val src = getFileSource(indexUri) | ||||||
|  | 			getHasher(hashFormat).getHashingSource(src) | ||||||
|  | 		} catch (e: Exception) { | ||||||
|  | 			ui.showErrorAndExit("Failed to download index file", e) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		val indexFile = try { | ||||||
|  | 			Toml().read(indexFileSource.buffer().inputStream()).to(IndexFile::class.java) | ||||||
|  | 		} catch (e: IllegalStateException) { | ||||||
|  | 			ui.showErrorAndExit("Failed to parse index file", e) | ||||||
|  | 		} | ||||||
|  | 		if (!indexFileSource.hashIsEqual(indexHash)) { | ||||||
|  | 			ui.showErrorAndExit("Your index file hash is invalid! The pack developer should packwiz refresh on the pack again") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ui.submitProgress(InstallProgress("Checking local files...")) | ||||||
|  | 		// TODO: use kotlin filtering/FP rather than an iterator? | ||||||
|  | 		val it: MutableIterator<Map.Entry<SpaceSafeURI, ManifestFile.File>> = manifest.cachedFiles.entries.iterator() | ||||||
|  | 		while (it.hasNext()) { | ||||||
|  | 			val (uri, file) = it.next() | ||||||
|  | 			if (file.cachedLocation != null) { | ||||||
|  | 				var alreadyDeleted = false | ||||||
|  | 				// Delete if option value has been set to false | ||||||
|  | 				if (file.isOptional && !file.optionValue) { | ||||||
|  | 					try { | ||||||
|  | 						Files.deleteIfExists(Paths.get(opts.packFolder, file.cachedLocation)) | ||||||
|  | 					} catch (e: IOException) { | ||||||
|  | 						Log.warn("Failed to delete optional disabled file", e) | ||||||
|  | 					} | ||||||
|  | 					// Set to null, as it doesn't exist anymore | ||||||
|  | 					file.cachedLocation = null | ||||||
|  | 					alreadyDeleted = true | ||||||
|  | 				} | ||||||
|  | 				if (indexFile.files.none { it.file == uri }) { // File has been removed from the index | ||||||
|  | 					if (!alreadyDeleted) { | ||||||
|  | 						try { | ||||||
|  | 							Files.deleteIfExists(Paths.get(opts.packFolder, file.cachedLocation)) | ||||||
|  | 						} catch (e: IOException) { | ||||||
|  | 							Log.warn("Failed to delete file removed from index", e) | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 					it.remove() | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		ui.submitProgress(InstallProgress("Comparing new files...")) | ||||||
|  |  | ||||||
|  | 		// TODO: progress bar? | ||||||
|  | 		if (indexFile.files.isEmpty()) { | ||||||
|  | 			Log.warn("Index is empty!") | ||||||
|  | 		} | ||||||
|  | 		val tasks = createTasksFromIndex(indexFile, indexFile.hashFormat, opts.side) | ||||||
|  | 		// If the side changes, invalidate EVERYTHING just in case | ||||||
|  | 		// Might not be needed, but done just to be safe | ||||||
|  | 		val invalidateAll = opts.side != manifest.cachedSide | ||||||
|  | 		if (invalidateAll) { | ||||||
|  | 			Log.info("Side changed, invalidating all mods") | ||||||
|  | 		} | ||||||
|  | 		tasks.forEach{ f -> | ||||||
|  | 			// TODO: should linkedfile be checked as well? should this be done in the download section? | ||||||
|  | 			if (invalidateAll) { | ||||||
|  | 				f.invalidate() | ||||||
|  | 			} else if (invalidatedUris.contains(f.metadata.file)) { | ||||||
|  | 				f.invalidate() | ||||||
|  | 			} | ||||||
|  | 			val file = manifest.cachedFiles[f.metadata.file] | ||||||
|  | 			// Ensure the file can be reverted later if necessary - the DownloadTask modifies the file so if it fails we need the old version back | ||||||
|  | 			file?.backup() | ||||||
|  | 			// If it is null, the DownloadTask will make a new empty cachedFile | ||||||
|  | 			f.updateFromCache(file) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Let's hope downloadMetadata is a pure function!!! | ||||||
|  | 		tasks.parallelStream().forEach { f -> f.downloadMetadata(indexFile, indexUri) } | ||||||
|  |  | ||||||
|  | 		val failedTaskDetails = tasks.asSequence().map(DownloadTask::exceptionDetails).filterNotNull().toList() | ||||||
|  | 		if (failedTaskDetails.isNotEmpty()) { | ||||||
|  | 			errorsOccurred = true | ||||||
|  | 			when (ui.showExceptions(failedTaskDetails, tasks.size, true)) { | ||||||
|  | 				ExceptionListResult.CONTINUE -> {} | ||||||
|  | 				ExceptionListResult.CANCEL -> { | ||||||
|  | 					cancelled = true | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
|  | 				ExceptionListResult.IGNORE -> { | ||||||
|  | 					cancelledStartGame = true | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (ui.cancelButtonPressed) { | ||||||
|  | 			showCancellationDialog() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// TODO: task failed function? | ||||||
|  | 		val nonFailedFirstTasks = tasks.filter { t -> !t.failed() }.toList() | ||||||
|  | 		val optionTasks = nonFailedFirstTasks.filter(DownloadTask::correctSide).filter(DownloadTask::isOptional).toList() | ||||||
|  | 		// If options changed, present all options again | ||||||
|  | 		if (ui.optionsButtonPressed || optionTasks.any(DownloadTask::isNewOptional)) { | ||||||
|  | 			// new ArrayList is required so it's an IOptionDetails rather than a DownloadTask list | ||||||
|  | 			if (ui.showOptions(ArrayList(optionTasks))) { | ||||||
|  | 				cancelled = true | ||||||
|  | 				handleCancellation() | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		// TODO: keep this enabled? then apply changes after download process? | ||||||
|  | 		ui.disableOptionsButton(optionTasks.isNotEmpty()) | ||||||
|  |  | ||||||
|  | 		// TODO: different thread pool type? | ||||||
|  | 		val threadPool = Executors.newFixedThreadPool(10) | ||||||
|  | 		val completionService: CompletionService<DownloadTask> = ExecutorCompletionService(threadPool) | ||||||
|  | 		tasks.forEach { t -> | ||||||
|  | 			completionService.submit { | ||||||
|  | 				t.download(opts.packFolder, indexUri) | ||||||
|  | 				t | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		for (i in tasks.indices) { | ||||||
|  | 			val task: DownloadTask = try { | ||||||
|  | 				completionService.take().get() | ||||||
|  | 			} catch (e: InterruptedException) { | ||||||
|  | 				ui.showErrorAndExit("Interrupted when consuming download tasks", e) | ||||||
|  | 			} catch (e: ExecutionException) { | ||||||
|  | 				ui.showErrorAndExit("Failed to execute download task", e) | ||||||
|  | 			} | ||||||
|  | 			// Update manifest - If there were no errors cachedFile has already been modified in place (good old pass by reference) | ||||||
|  | 			task.cachedFile?.let { file -> | ||||||
|  | 				if (task.failed()) { | ||||||
|  | 					val oldFile = file.revert | ||||||
|  | 					if (oldFile != null) { | ||||||
|  | 						task.metadata.file?.let { uri -> manifest.cachedFiles.putIfAbsent(uri, oldFile) } | ||||||
|  | 					} else { null } | ||||||
|  | 				} else { | ||||||
|  | 					task.metadata.file?.let { uri -> manifest.cachedFiles.putIfAbsent(uri, file) } | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			val exDetails = task.exceptionDetails | ||||||
|  | 			val progress = if (exDetails != null) { | ||||||
|  | 				"Failed to download ${exDetails.name}: ${exDetails.exception.message}" | ||||||
|  | 			} else { | ||||||
|  | 				"Downloaded ${task.name}" | ||||||
|  | 			} | ||||||
|  | 			ui.submitProgress(InstallProgress(progress, i + 1, tasks.size)) | ||||||
|  |  | ||||||
|  | 			if (ui.cancelButtonPressed) { // Stop all tasks, don't launch the game (it's in an invalid state!) | ||||||
|  | 				threadPool.shutdown() | ||||||
|  | 				cancelled = true | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Shut down the thread pool when the update is done | ||||||
|  | 		threadPool.shutdown() | ||||||
|  |  | ||||||
|  | 		val failedTasks2ElectricBoogaloo = nonFailedFirstTasks.asSequence().map(DownloadTask::exceptionDetails).filterNotNull().toList() | ||||||
|  | 		if (failedTasks2ElectricBoogaloo.isNotEmpty()) { | ||||||
|  | 			errorsOccurred = true | ||||||
|  | 			when (ui.showExceptions(failedTasks2ElectricBoogaloo, tasks.size, false)) { | ||||||
|  | 				ExceptionListResult.CONTINUE -> {} | ||||||
|  | 				ExceptionListResult.CANCEL -> cancelled = true | ||||||
|  | 				ExceptionListResult.IGNORE -> cancelledStartGame = true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private fun showCancellationDialog() { | ||||||
|  | 		when (ui.showCancellationDialog()) { | ||||||
|  | 			CancellationResult.QUIT -> cancelled = true | ||||||
|  | 			CancellationResult.CONTINUE -> cancelledStartGame = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: move to UI? | ||||||
|  | 	private fun handleCancellation() { | ||||||
|  | 		if (cancelled) { | ||||||
|  | 			println("Update cancelled by user!") | ||||||
|  | 			exitProcess(1) | ||||||
|  | 		} else if (cancelledStartGame) { | ||||||
|  | 			println("Update cancelled by user! Continuing to start game...") | ||||||
|  | 			exitProcess(0) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,27 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.TypeAdapter | ||||||
|  | import com.google.gson.stream.JsonReader | ||||||
|  | import com.google.gson.stream.JsonToken | ||||||
|  | import com.google.gson.stream.JsonWriter | ||||||
|  | import java.io.IOException | ||||||
|  |  | ||||||
|  | class EfficientBooleanAdapter : TypeAdapter<Boolean?>() { | ||||||
|  | 	@Throws(IOException::class) | ||||||
|  | 	override fun write(out: JsonWriter, value: Boolean?) { | ||||||
|  | 		if (value == null || !value) { | ||||||
|  | 			out.nullValue() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		out.value(true) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Throws(IOException::class) | ||||||
|  | 	override fun read(reader: JsonReader): Boolean? { | ||||||
|  | 		if (reader.peek() == JsonToken.NULL) { | ||||||
|  | 			reader.nextNull() | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 		return reader.nextBoolean() | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,99 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.annotations.SerializedName | ||||||
|  | import com.moandjiezana.toml.Toml | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.Hash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHasher | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getFileSource | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getNewLoc | ||||||
|  | import okio.Source | ||||||
|  | import okio.buffer | ||||||
|  | import java.nio.file.Paths | ||||||
|  |  | ||||||
|  | class IndexFile { | ||||||
|  | 	@SerializedName("hash-format") | ||||||
|  | 	var hashFormat: String = "sha-256" | ||||||
|  | 	var files: MutableList<File> = ArrayList() | ||||||
|  |  | ||||||
|  | 	class File { | ||||||
|  | 		var file: SpaceSafeURI? = null | ||||||
|  | 		@SerializedName("hash-format") | ||||||
|  | 		var hashFormat: String? = null | ||||||
|  | 		var hash: String? = null | ||||||
|  | 		var alias: SpaceSafeURI? = null | ||||||
|  | 		var metafile = false | ||||||
|  | 		var preserve = false | ||||||
|  |  | ||||||
|  | 		@Transient | ||||||
|  | 		var linkedFile: ModFile? = null | ||||||
|  | 		@Transient | ||||||
|  | 		var linkedFileURI: SpaceSafeURI? = null | ||||||
|  |  | ||||||
|  | 		@Throws(Exception::class) | ||||||
|  | 		fun downloadMeta(parentIndexFile: IndexFile, indexUri: SpaceSafeURI?) { | ||||||
|  | 			if (!metafile) { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			if (hashFormat?.length ?: 0 == 0) { | ||||||
|  | 				hashFormat = parentIndexFile.hashFormat | ||||||
|  | 			} | ||||||
|  | 			// TODO: throw a proper exception instead of allowing NPE? | ||||||
|  | 			val fileHash = getHash(hashFormat!!, hash!!) | ||||||
|  | 			linkedFileURI = getNewLoc(indexUri, file) | ||||||
|  | 			val src = getFileSource(linkedFileURI!!) | ||||||
|  | 			val fileStream = getHasher(hashFormat!!).getHashingSource(src) | ||||||
|  | 			linkedFile = Toml().read(fileStream.buffer().inputStream()).to(ModFile::class.java) | ||||||
|  | 			if (!fileStream.hashIsEqual(fileHash)) { | ||||||
|  | 				throw Exception("Invalid mod file hash") | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		@Throws(Exception::class) | ||||||
|  | 		fun getSource(indexUri: SpaceSafeURI?): Source { | ||||||
|  | 			return if (metafile) { | ||||||
|  | 				if (linkedFile == null) { | ||||||
|  | 					throw Exception("Linked file doesn't exist!") | ||||||
|  | 				} | ||||||
|  | 				linkedFile!!.getSource(linkedFileURI) | ||||||
|  | 			} else { | ||||||
|  | 				val newLoc = getNewLoc(indexUri, file) ?: throw Exception("Index file URI is invalid") | ||||||
|  | 				getFileSource(newLoc) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		@Throws(Exception::class) | ||||||
|  | 		fun getHashObj(): Hash { | ||||||
|  | 			if (hash == null) { // TODO: should these be more specific exceptions (e.g. IndexFileException?!) | ||||||
|  | 				throw Exception("Index file doesn't have a hash") | ||||||
|  | 			} | ||||||
|  | 			if (hashFormat == null) { | ||||||
|  | 				throw Exception("Index file doesn't have a hash format") | ||||||
|  | 			} | ||||||
|  | 			return getHash(hashFormat!!, hash!!) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// TODO: throw some kind of exception? | ||||||
|  | 		val name: String | ||||||
|  | 			get() { | ||||||
|  | 				if (metafile) { | ||||||
|  | 					return linkedFile?.name ?: linkedFile?.filename ?: | ||||||
|  | 					file?.run { Paths.get(path).fileName.toString() } ?: "Invalid file" | ||||||
|  | 				} | ||||||
|  | 				return file?.run { Paths.get(path).fileName.toString() } ?: "Invalid file" | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 		// TODO: URIs are bad | ||||||
|  | 		val destURI: SpaceSafeURI? | ||||||
|  | 			get() { | ||||||
|  | 				if (alias != null) { | ||||||
|  | 					return alias | ||||||
|  | 				} | ||||||
|  | 				return if (metafile && linkedFile != null) { | ||||||
|  | 					linkedFile?.filename?.let { file?.resolve(it) } | ||||||
|  | 				} else { | ||||||
|  | 					file | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.annotations.JsonAdapter | ||||||
|  | import link.infra.packwiz.installer.UpdateManager | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.Hash | ||||||
|  |  | ||||||
|  | class ManifestFile { | ||||||
|  | 	var packFileHash: Hash? = null | ||||||
|  | 	var indexFileHash: Hash? = null | ||||||
|  | 	var cachedFiles: MutableMap<SpaceSafeURI, File> = HashMap() | ||||||
|  | 	// If the side changes, EVERYTHING invalidates. FUN!!! | ||||||
|  | 	var cachedSide = UpdateManager.Options.Side.CLIENT | ||||||
|  |  | ||||||
|  | 	// TODO: switch to Kotlin-friendly JSON/TOML libs? | ||||||
|  | 	class File { | ||||||
|  | 		@Transient | ||||||
|  | 		var revert: File? = null | ||||||
|  | 			private set | ||||||
|  |  | ||||||
|  | 		var hash: Hash? = null | ||||||
|  | 		var linkedFileHash: Hash? = null | ||||||
|  | 		var cachedLocation: String? = null | ||||||
|  |  | ||||||
|  | 		@JsonAdapter(EfficientBooleanAdapter::class) | ||||||
|  | 		var isOptional = false | ||||||
|  | 		var optionValue = true | ||||||
|  |  | ||||||
|  | 		@JsonAdapter(EfficientBooleanAdapter::class) | ||||||
|  | 		var onlyOtherSide = false | ||||||
|  |  | ||||||
|  | 		// When an error occurs, the state needs to be reverted. To do this, I have a crude revert system. | ||||||
|  | 		fun backup() { | ||||||
|  | 			revert = File().also { | ||||||
|  | 				it.hash = hash | ||||||
|  | 				it.linkedFileHash = linkedFileHash | ||||||
|  | 				it.cachedLocation = cachedLocation | ||||||
|  | 				it.isOptional = isOptional | ||||||
|  | 				it.optionValue = optionValue | ||||||
|  | 				it.onlyOtherSide = onlyOtherSide | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,57 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.annotations.SerializedName | ||||||
|  | import link.infra.packwiz.installer.UpdateManager | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.Hash | ||||||
|  | import link.infra.packwiz.installer.metadata.hash.HashUtils.getHash | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getFileSource | ||||||
|  | import link.infra.packwiz.installer.request.HandlerManager.getNewLoc | ||||||
|  | import okio.Source | ||||||
|  |  | ||||||
|  | class ModFile { | ||||||
|  | 	var name: String? = null | ||||||
|  | 	var filename: String? = null | ||||||
|  | 	var side: UpdateManager.Options.Side? = null | ||||||
|  | 	var download: Download? = null | ||||||
|  |  | ||||||
|  | 	class Download { | ||||||
|  | 		var url: SpaceSafeURI? = null | ||||||
|  | 		@SerializedName("hash-format") | ||||||
|  | 		var hashFormat: String? = null | ||||||
|  | 		var hash: String? = null | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var update: Map<String, Any>? = null | ||||||
|  | 	var option: Option? = null | ||||||
|  |  | ||||||
|  | 	class Option { | ||||||
|  | 		var optional = false | ||||||
|  | 		var description: String? = null | ||||||
|  | 		@SerializedName("default") | ||||||
|  | 		var defaultValue = false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Throws(Exception::class) | ||||||
|  | 	fun getSource(baseLoc: SpaceSafeURI?): Source { | ||||||
|  | 		download?.let { | ||||||
|  | 			if (it.url == null) { | ||||||
|  | 				throw Exception("Metadata file doesn't have a download URI") | ||||||
|  | 			} | ||||||
|  | 			val newLoc = getNewLoc(baseLoc, it.url) ?: throw Exception("Metadata file URI is invalid") | ||||||
|  | 			return getFileSource(newLoc) | ||||||
|  | 		} ?: throw Exception("Metadata file doesn't have download") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@get:Throws(Exception::class) | ||||||
|  | 	val hash: Hash | ||||||
|  | 		get() { | ||||||
|  | 			download?.let { | ||||||
|  | 				return getHash( | ||||||
|  | 						it.hashFormat ?: throw Exception("Metadata file doesn't have a hash format"), | ||||||
|  | 						it.hash ?: throw Exception("Metadata file doesn't have a hash") | ||||||
|  | 				) | ||||||
|  | 			} ?: throw Exception("Metadata file doesn't have download") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	val isOptional: Boolean get() = option?.optional ?: false | ||||||
|  | } | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.annotations.SerializedName | ||||||
|  |  | ||||||
|  | class PackFile { | ||||||
|  | 	var name: String? = null | ||||||
|  | 	var index: IndexFileLoc? = null | ||||||
|  |  | ||||||
|  | 	class IndexFileLoc { | ||||||
|  | 		var file: SpaceSafeURI? = null | ||||||
|  | 		@SerializedName("hash-format") | ||||||
|  | 		var hashFormat: String? = null | ||||||
|  | 		var hash: String? = null | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var versions: Map<String, String>? = null | ||||||
|  | 	var client: Map<String, Any>? = null | ||||||
|  | 	var server: Map<String, Any>? = null | ||||||
|  | } | ||||||
| @@ -0,0 +1,61 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.annotations.JsonAdapter | ||||||
|  | import java.io.Serializable | ||||||
|  | import java.net.MalformedURLException | ||||||
|  | import java.net.URI | ||||||
|  | import java.net.URISyntaxException | ||||||
|  | import java.net.URL | ||||||
|  |  | ||||||
|  | // The world's worst URI wrapper | ||||||
|  | @JsonAdapter(SpaceSafeURIParser::class) | ||||||
|  | class SpaceSafeURI : Comparable<SpaceSafeURI>, Serializable { | ||||||
|  | 	private val u: URI | ||||||
|  |  | ||||||
|  | 	@Throws(URISyntaxException::class) | ||||||
|  | 	constructor(str: String) { | ||||||
|  | 		u = URI(str.replace(" ", "%20")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	constructor(uri: URI) { | ||||||
|  | 		u = uri | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Throws(URISyntaxException::class) | ||||||
|  | 	constructor(scheme: String?, authority: String?, path: String?, query: String?, fragment: String?) { // TODO: do all components need to be replaced? | ||||||
|  | 		u = URI( | ||||||
|  | 				scheme?.replace(" ", "%20"), | ||||||
|  | 				authority?.replace(" ", "%20"), | ||||||
|  | 				path?.replace(" ", "%20"), | ||||||
|  | 				query?.replace(" ", "%20"), | ||||||
|  | 				fragment?.replace(" ", "%20") | ||||||
|  | 		) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	val path: String get() = u.path.replace("%20", " ") | ||||||
|  |  | ||||||
|  | 	override fun toString(): String = u.toString().replace("%20", " ") | ||||||
|  |  | ||||||
|  | 	fun resolve(path: String): SpaceSafeURI = SpaceSafeURI(u.resolve(path.replace(" ", "%20"))) | ||||||
|  |  | ||||||
|  | 	fun resolve(loc: SpaceSafeURI): SpaceSafeURI = SpaceSafeURI(u.resolve(loc.u)) | ||||||
|  |  | ||||||
|  | 	fun relativize(loc: SpaceSafeURI): SpaceSafeURI = SpaceSafeURI(u.relativize(loc.u)) | ||||||
|  |  | ||||||
|  | 	override fun equals(other: Any?): Boolean { | ||||||
|  | 		return if (other is SpaceSafeURI) { | ||||||
|  | 			u == other.u | ||||||
|  | 		} else false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun hashCode() = u.hashCode() | ||||||
|  |  | ||||||
|  | 	override fun compareTo(other: SpaceSafeURI): Int = u.compareTo(other.u) | ||||||
|  |  | ||||||
|  | 	val scheme: String get() = u.scheme | ||||||
|  | 	val authority: String get() = u.authority | ||||||
|  | 	val host: String get() = u.host | ||||||
|  |  | ||||||
|  | 	@Throws(MalformedURLException::class) | ||||||
|  | 	fun toURL(): URL = u.toURL() | ||||||
|  | } | ||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata | ||||||
|  |  | ||||||
|  | import com.google.gson.JsonDeserializationContext | ||||||
|  | import com.google.gson.JsonDeserializer | ||||||
|  | import com.google.gson.JsonElement | ||||||
|  | import com.google.gson.JsonParseException | ||||||
|  | import java.lang.reflect.Type | ||||||
|  | import java.net.URISyntaxException | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * This class encodes spaces before parsing the URI, so the URI can actually be | ||||||
|  |  * parsed. | ||||||
|  |  */ | ||||||
|  | internal class SpaceSafeURIParser : JsonDeserializer<SpaceSafeURI> { | ||||||
|  | 	@Throws(JsonParseException::class) | ||||||
|  | 	override fun deserialize(json: JsonElement, typeOfT: Type, context: JsonDeserializationContext): SpaceSafeURI { | ||||||
|  | 		return try { | ||||||
|  | 			SpaceSafeURI(json.asString) | ||||||
|  | 		} catch (e: URISyntaxException) { | ||||||
|  | 			throw JsonParseException("Failed to parse URI", e) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: replace this with a better solution? | ||||||
|  | } | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | import okio.ForwardingSource | ||||||
|  | import okio.Source | ||||||
|  |  | ||||||
|  | abstract class GeneralHashingSource(delegate: Source) : ForwardingSource(delegate) { | ||||||
|  | 	abstract val hash: Hash | ||||||
|  |  | ||||||
|  | 	fun hashIsEqual(compareTo: Any) = compareTo == hash | ||||||
|  | } | ||||||
| @@ -0,0 +1,34 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | import com.google.gson.* | ||||||
|  | import java.lang.reflect.Type | ||||||
|  |  | ||||||
|  | abstract class Hash { | ||||||
|  | 	protected abstract val stringValue: String | ||||||
|  | 	protected abstract val type: String | ||||||
|  |  | ||||||
|  | 	class TypeHandler : JsonDeserializer<Hash>, JsonSerializer<Hash> { | ||||||
|  | 		override fun serialize(src: Hash, typeOfSrc: Type, context: JsonSerializationContext): JsonElement = JsonObject().apply { | ||||||
|  | 			add("type", JsonPrimitive(src.type)) | ||||||
|  | 			add("value", JsonPrimitive(src.stringValue)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		@Throws(JsonParseException::class) | ||||||
|  | 		override fun deserialize(json: JsonElement, typeOfT: Type, context: JsonDeserializationContext): Hash { | ||||||
|  | 			val obj = json.asJsonObject | ||||||
|  | 			val type: String | ||||||
|  | 			val value: String | ||||||
|  | 			try { | ||||||
|  | 				type = obj["type"].asString | ||||||
|  | 				value = obj["value"].asString | ||||||
|  | 			} catch (e: NullPointerException) { | ||||||
|  | 				throw JsonParseException("Invalid hash JSON data") | ||||||
|  | 			} | ||||||
|  | 			return try { | ||||||
|  | 				HashUtils.getHash(type, value) | ||||||
|  | 			} catch (e: Exception) { | ||||||
|  | 				throw JsonParseException("Failed to create hash object", e) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | object HashUtils { | ||||||
|  | 	private val hashTypeConversion: Map<String, IHasher> = mapOf( | ||||||
|  | 			"sha256" to HashingSourceHasher("sha256"), | ||||||
|  | 			"sha512" to HashingSourceHasher("sha512"), | ||||||
|  | 			"murmur2" to Murmur2Hasher(), | ||||||
|  | 			"sha1" to HashingSourceHasher("sha1") | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	@JvmStatic | ||||||
|  | 	@Throws(Exception::class) | ||||||
|  | 	fun getHasher(type: String): IHasher { | ||||||
|  | 		return hashTypeConversion[type] ?: throw Exception("Hash type not supported: $type") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@JvmStatic | ||||||
|  | 	@Throws(Exception::class) | ||||||
|  | 	fun getHash(type: String, value: String): Hash { | ||||||
|  | 		return hashTypeConversion[type]?.getHash(value) ?: throw Exception("Hash type not supported: $type") | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,46 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | import okio.HashingSource | ||||||
|  | import okio.Source | ||||||
|  |  | ||||||
|  | class HashingSourceHasher internal constructor(private val type: String) : IHasher { | ||||||
|  | 	// i love naming things | ||||||
|  | 	private inner class HashingSourceGeneralHashingSource(val delegateHashing: HashingSource) : GeneralHashingSource(delegateHashing) { | ||||||
|  | 		override val hash: Hash by lazy(LazyThreadSafetyMode.NONE) { | ||||||
|  | 			HashingSourceHash(delegateHashing.hash.hex()) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// this some funky inner class stuff | ||||||
|  | 	// each of these classes is specific to the instance of the HasherHashingSource | ||||||
|  | 	// therefore HashingSourceHashes from different parent instances will be not instanceof each other | ||||||
|  | 	private inner class HashingSourceHash(val value: String) : Hash() { | ||||||
|  | 		override val stringValue get() = value | ||||||
|  |  | ||||||
|  | 		override fun equals(other: Any?): Boolean { | ||||||
|  | 			if (other !is HashingSourceHash) { | ||||||
|  | 				return false | ||||||
|  | 			} | ||||||
|  | 			return stringValue.equals(other.stringValue, ignoreCase = true) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		override fun toString(): String = "$type: $stringValue" | ||||||
|  | 		override fun hashCode(): Int = value.hashCode() | ||||||
|  |  | ||||||
|  | 		override val type: String get() = this@HashingSourceHasher.type | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getHashingSource(delegate: Source): GeneralHashingSource { | ||||||
|  | 		when (type) { | ||||||
|  | 			"md5" -> return HashingSourceGeneralHashingSource(HashingSource.md5(delegate)) | ||||||
|  | 			"sha256" -> return HashingSourceGeneralHashingSource(HashingSource.sha256(delegate)) | ||||||
|  | 			"sha512" -> return HashingSourceGeneralHashingSource(HashingSource.sha512(delegate)) | ||||||
|  | 			"sha1" -> return HashingSourceGeneralHashingSource(HashingSource.sha1(delegate)) | ||||||
|  | 		} | ||||||
|  | 		throw RuntimeException("Invalid hash type provided") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getHash(value: String): Hash { | ||||||
|  | 		return HashingSourceHash(value) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | import okio.Source | ||||||
|  |  | ||||||
|  | interface IHasher { | ||||||
|  | 	fun getHashingSource(delegate: Source): GeneralHashingSource | ||||||
|  | 	fun getHash(value: String): Hash | ||||||
|  | } | ||||||
| @@ -0,0 +1,91 @@ | |||||||
|  | package link.infra.packwiz.installer.metadata.hash | ||||||
|  |  | ||||||
|  | import okio.Buffer | ||||||
|  | import okio.Source | ||||||
|  | import java.io.IOException | ||||||
|  |  | ||||||
|  | class Murmur2Hasher : IHasher { | ||||||
|  | 	private inner class Murmur2GeneralHashingSource(delegate: Source) : GeneralHashingSource(delegate) { | ||||||
|  | 		val internalBuffer = Buffer() | ||||||
|  | 		val tempBuffer = Buffer() | ||||||
|  |  | ||||||
|  | 		override val hash: Hash by lazy(LazyThreadSafetyMode.NONE) { | ||||||
|  | 			val data = internalBuffer.readByteArray() | ||||||
|  | 			Murmur2Hash(Murmur2Lib.hash32(data, data.size, 1)) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		@Throws(IOException::class) | ||||||
|  | 		override fun read(sink: Buffer, byteCount: Long): Long { | ||||||
|  | 			val out = delegate.read(tempBuffer, byteCount) | ||||||
|  | 			if (out > -1) { | ||||||
|  | 				sink.write(tempBuffer.clone(), out) | ||||||
|  | 				computeNormalizedBufferFaster(tempBuffer, internalBuffer) | ||||||
|  | 			} | ||||||
|  | 			return out | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Credit to https://github.com/modmuss50/CAV2/blob/master/murmur.go | ||||||
|  | //		private fun computeNormalizedArray(input: ByteArray): ByteArray { | ||||||
|  | //			val output = ByteArray(input.size) | ||||||
|  | //			var index = 0 | ||||||
|  | //			for (b in input) { | ||||||
|  | //				when (b) { | ||||||
|  | //					9.toByte(), 10.toByte(), 13.toByte(), 32.toByte() -> {} | ||||||
|  | //					else -> { | ||||||
|  | //						output[index] = b | ||||||
|  | //						index++ | ||||||
|  | //					} | ||||||
|  | //				} | ||||||
|  | //			} | ||||||
|  | //			val outputTrimmed = ByteArray(index) | ||||||
|  | //			System.arraycopy(output, 0, outputTrimmed, 0, index) | ||||||
|  | //			return outputTrimmed | ||||||
|  | //		} | ||||||
|  |  | ||||||
|  | 		private fun computeNormalizedBufferFaster(input: Buffer, output: Buffer) { | ||||||
|  | 			var index = 0 | ||||||
|  | 			val arr = input.readByteArray() | ||||||
|  | 			for (b in arr) { | ||||||
|  | 				when (b) { | ||||||
|  | 					9.toByte(), 10.toByte(), 13.toByte(), 32.toByte() -> {} | ||||||
|  | 					else -> { | ||||||
|  | 						arr[index] = b | ||||||
|  | 						index++ | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			output.write(arr, 0, index) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private class Murmur2Hash : Hash { | ||||||
|  | 		val value: Int | ||||||
|  |  | ||||||
|  | 		constructor(value: String) { | ||||||
|  | 			// Parsing as long then casting to int converts values gt int max value but lt uint max value | ||||||
|  | 			// into negatives. I presume this is how the murmur2 code handles this. | ||||||
|  | 			this.value = value.toLong().toInt() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		constructor(value: Int) { | ||||||
|  | 			this.value = value | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		override val stringValue get() = value.toString() | ||||||
|  | 		override val type get() = "murmur2" | ||||||
|  |  | ||||||
|  | 		override fun equals(other: Any?): Boolean { | ||||||
|  | 			if (other !is Murmur2Hash) { | ||||||
|  | 				return false | ||||||
|  | 			} | ||||||
|  | 			return value == other.value | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		override fun toString(): String = "murmur2: $value" | ||||||
|  | 		override fun hashCode(): Int = value | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getHashingSource(delegate: Source): GeneralHashingSource = Murmur2GeneralHashingSource(delegate) | ||||||
|  | 	override fun getHash(value: String): Hash = Murmur2Hash(value) | ||||||
|  | } | ||||||
| @@ -0,0 +1,52 @@ | |||||||
|  | package link.infra.packwiz.installer.request | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.request.handlers.RequestHandlerFile | ||||||
|  | import link.infra.packwiz.installer.request.handlers.RequestHandlerGithub | ||||||
|  | import link.infra.packwiz.installer.request.handlers.RequestHandlerHTTP | ||||||
|  | import okio.Source | ||||||
|  |  | ||||||
|  | object HandlerManager { | ||||||
|  |  | ||||||
|  | 	private val handlers: List<IRequestHandler> = listOf( | ||||||
|  | 			RequestHandlerGithub(), | ||||||
|  | 			RequestHandlerHTTP(), | ||||||
|  | 			RequestHandlerFile() | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	 // TODO: get rid of nullable stuff here | ||||||
|  | 	@JvmStatic | ||||||
|  | 	fun getNewLoc(base: SpaceSafeURI?, loc: SpaceSafeURI?): SpaceSafeURI? { | ||||||
|  | 		if (loc == null) { | ||||||
|  | 			return null | ||||||
|  | 		} | ||||||
|  | 		val dest = base?.run { resolve(loc) } ?: loc | ||||||
|  | 		for (handler in handlers) with (handler) { | ||||||
|  | 			if (matchesHandler(dest)) { | ||||||
|  | 				return getNewLoc(dest) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return dest | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: What if files are read multiple times?? | ||||||
|  | 	// Zip handler discards once read, requesting multiple times on other handlers would cause multiple downloads | ||||||
|  | 	// Caching system? Copy from already downloaded files? | ||||||
|  |  | ||||||
|  | 	// TODO: change to use something more idiomatic than exceptions? | ||||||
|  |  | ||||||
|  | 	@JvmStatic | ||||||
|  | 	@Throws(Exception::class) | ||||||
|  | 	fun getFileSource(loc: SpaceSafeURI): Source { | ||||||
|  | 		for (handler in handlers) { | ||||||
|  | 			if (handler.matchesHandler(loc)) { | ||||||
|  | 				return handler.getFileSource(loc) ?: throw Exception("Couldn't find URI: $loc") | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		throw Exception("No handler available for URI: $loc") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: github toml resolution? | ||||||
|  | 	// e.g. https://github.com/comp500/Demagnetize -> demagnetize.toml | ||||||
|  | 	// https://github.com/comp500/Demagnetize/blob/master/demagnetize.toml | ||||||
|  | } | ||||||
| @@ -1,19 +1,18 @@ | |||||||
| package link.infra.packwiz.installer.request; | package link.infra.packwiz.installer.request | ||||||
| 
 | 
 | ||||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
| import okio.Source; | import okio.Source | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * IRequestHandler handles requests for locations specified in modpack metadata. |  * IRequestHandler handles requests for locations specified in modpack metadata. | ||||||
|  */ |  */ | ||||||
| public interface IRequestHandler { | interface IRequestHandler { | ||||||
| 	 | 	fun matchesHandler(loc: SpaceSafeURI): Boolean | ||||||
| 	boolean matchesHandler(SpaceSafeURI loc); | 
 | ||||||
| 	 | 	fun getNewLoc(loc: SpaceSafeURI): SpaceSafeURI { | ||||||
| 	default SpaceSafeURI getNewLoc(SpaceSafeURI loc) { | 		return loc | ||||||
| 		return loc; |  | ||||||
| 	} | 	} | ||||||
| 	 | 
 | ||||||
| 	/** | 	/** | ||||||
| 	 * Gets the Source for a location. Must be threadsafe. | 	 * Gets the Source for a location. Must be threadsafe. | ||||||
| 	 * It is assumed that each location is read only once for the duration of an IRequestHandler. | 	 * It is assumed that each location is read only once for the duration of an IRequestHandler. | ||||||
| @@ -21,6 +20,5 @@ public interface IRequestHandler { | |||||||
| 	 * @return The Source containing the data of the file | 	 * @return The Source containing the data of the file | ||||||
| 	 * @throws Exception Exception if it failed to download a file!!! | 	 * @throws Exception Exception if it failed to download a file!!! | ||||||
| 	 */ | 	 */ | ||||||
| 	Source getFileSource(SpaceSafeURI loc) throws Exception; | 	fun getFileSource(loc: SpaceSafeURI): Source? | ||||||
| 
 | } | ||||||
| } |  | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | package link.infra.packwiz.installer.request.handlers | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.request.IRequestHandler | ||||||
|  | import okio.Source | ||||||
|  | import okio.source | ||||||
|  | import java.nio.file.Paths | ||||||
|  |  | ||||||
|  | open class RequestHandlerFile : IRequestHandler { | ||||||
|  | 	override fun matchesHandler(loc: SpaceSafeURI): Boolean { | ||||||
|  | 		return "file" == loc.scheme | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getFileSource(loc: SpaceSafeURI): Source? { | ||||||
|  | 		val path = Paths.get(loc.toURL().toURI()) | ||||||
|  | 		return path.source() | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,71 @@ | |||||||
|  | package link.infra.packwiz.installer.request.handlers | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import java.util.* | ||||||
|  | import java.util.concurrent.locks.ReentrantReadWriteLock | ||||||
|  | import java.util.regex.Pattern | ||||||
|  | import kotlin.concurrent.read | ||||||
|  | import kotlin.concurrent.write | ||||||
|  |  | ||||||
|  | class RequestHandlerGithub : RequestHandlerZip(true) { | ||||||
|  | 	override fun getNewLoc(loc: SpaceSafeURI): SpaceSafeURI { | ||||||
|  | 		return loc | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	companion object { | ||||||
|  | 		private val repoMatcherPattern = Pattern.compile("/([\\w.-]+/[\\w.-]+).*") | ||||||
|  | 		private val branchMatcherPattern = Pattern.compile("/[\\w.-]+/[\\w.-]+/blob/([\\w.-]+).*") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: is caching really needed, if HTTPURLConnection follows redirects correctly? | ||||||
|  | 	private val zipUriMap: MutableMap<String, SpaceSafeURI> = HashMap() | ||||||
|  | 	private val zipUriLock = ReentrantReadWriteLock() | ||||||
|  | 	private fun getRepoName(loc: SpaceSafeURI): String? { | ||||||
|  | 		val matcher = repoMatcherPattern.matcher(loc.path) | ||||||
|  | 		return if (matcher.matches()) { | ||||||
|  | 			matcher.group(1) | ||||||
|  | 		} else { | ||||||
|  | 			null | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getZipUri(loc: SpaceSafeURI): SpaceSafeURI { | ||||||
|  | 		val repoName = getRepoName(loc) | ||||||
|  | 		val branchName = getBranch(loc) | ||||||
|  |  | ||||||
|  | 		zipUriLock.read { | ||||||
|  | 			zipUriMap["$repoName/$branchName"] | ||||||
|  | 		}?.let { return it } | ||||||
|  |  | ||||||
|  | 		var zipUri = SpaceSafeURI("https://api.github.com/repos/$repoName/zipball/$branchName") | ||||||
|  | 		zipUriLock.write { | ||||||
|  | 			// If another thread sets the value concurrently, use the existing value from the | ||||||
|  | 			// thread that first acquired the lock. | ||||||
|  | 			zipUri = zipUriMap.putIfAbsent("$repoName/$branchName", zipUri) ?: zipUri | ||||||
|  | 		} | ||||||
|  | 		return zipUri | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private fun getBranch(loc: SpaceSafeURI): String? { | ||||||
|  | 		val matcher = branchMatcherPattern.matcher(loc.path) | ||||||
|  | 		return if (matcher.matches()) { | ||||||
|  | 			matcher.group(1) | ||||||
|  | 		} else { | ||||||
|  | 			null | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getLocationInZip(loc: SpaceSafeURI): SpaceSafeURI { | ||||||
|  | 		val path = "/" + getRepoName(loc) + "/blob/" + getBranch(loc) | ||||||
|  | 		return SpaceSafeURI(loc.scheme, loc.authority, path, null, null).relativize(loc) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun matchesHandler(loc: SpaceSafeURI): Boolean { | ||||||
|  | 		val scheme = loc.scheme | ||||||
|  | 		if (!("http" == scheme || "https" == scheme)) { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 		// TODO: more match testing? | ||||||
|  | 		return "github.com" == loc.host && branchMatcherPattern.matcher(loc.path).matches() | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,29 @@ | |||||||
|  | package link.infra.packwiz.installer.request.handlers | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import link.infra.packwiz.installer.request.IRequestHandler | ||||||
|  | import okio.Source | ||||||
|  | import okio.source | ||||||
|  | import java.net.HttpURLConnection | ||||||
|  |  | ||||||
|  | open class RequestHandlerHTTP : IRequestHandler { | ||||||
|  | 	override fun matchesHandler(loc: SpaceSafeURI): Boolean { | ||||||
|  | 		val scheme = loc.scheme | ||||||
|  | 		return "http" == scheme || "https" == scheme | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun getFileSource(loc: SpaceSafeURI): Source? { | ||||||
|  | 		val conn = loc.toURL().openConnection() as HttpURLConnection | ||||||
|  | 		// TODO: when do we send specific headers??? should there be a way to signal this? | ||||||
|  | 		conn.addRequestProperty("Accept", "application/octet-stream") | ||||||
|  | 		// TODO: include version? | ||||||
|  | 		conn.addRequestProperty("User-Agent", "packwiz-installer") | ||||||
|  |  | ||||||
|  | 		conn.apply { | ||||||
|  | 			// 30 second read timeout | ||||||
|  | 			readTimeout = 30 * 1000 | ||||||
|  | 			requestMethod = "GET" | ||||||
|  | 		} | ||||||
|  | 		return conn.inputStream.source() | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,123 @@ | |||||||
|  | package link.infra.packwiz.installer.request.handlers | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||||
|  | import okio.Buffer | ||||||
|  | import okio.Source | ||||||
|  | import okio.buffer | ||||||
|  | import okio.source | ||||||
|  | import java.util.* | ||||||
|  | import java.util.concurrent.locks.ReentrantLock | ||||||
|  | import java.util.concurrent.locks.ReentrantReadWriteLock | ||||||
|  | import java.util.function.Predicate | ||||||
|  | import java.util.zip.ZipEntry | ||||||
|  | import java.util.zip.ZipInputStream | ||||||
|  | import kotlin.concurrent.read | ||||||
|  | import kotlin.concurrent.withLock | ||||||
|  | import kotlin.concurrent.write | ||||||
|  |  | ||||||
|  | abstract class RequestHandlerZip(private val modeHasFolder: Boolean) : RequestHandlerHTTP() { | ||||||
|  | 	private fun removeFolder(name: String): String { | ||||||
|  | 		return if (modeHasFolder) { | ||||||
|  | 			// TODO: replace with proper path checks once switched to Path?? | ||||||
|  | 			name.substring(name.indexOf("/") + 1) | ||||||
|  | 		} else { | ||||||
|  | 			name | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private inner class ZipReader(zip: Source) { | ||||||
|  | 		private val zis = ZipInputStream(zip.buffer().inputStream()) | ||||||
|  | 		private val readFiles: MutableMap<SpaceSafeURI, Buffer> = HashMap() | ||||||
|  | 		// Write lock implies access to ZipInputStream - only 1 thread must read at a time! | ||||||
|  | 		val filesLock = ReentrantLock() | ||||||
|  | 		private var entry: ZipEntry? = null | ||||||
|  |  | ||||||
|  | 		private val zipSource = zis.source().buffer() | ||||||
|  |  | ||||||
|  | 		// File lock must be obtained before calling this function | ||||||
|  | 		private fun readCurrFile(): Buffer { | ||||||
|  | 			val fileBuffer = Buffer() | ||||||
|  | 			zipSource.readFully(fileBuffer, entry!!.size) | ||||||
|  | 			return fileBuffer | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// File lock must be obtained before calling this function | ||||||
|  | 		private fun findFile(loc: SpaceSafeURI): Buffer? { | ||||||
|  | 			while (true) { | ||||||
|  | 				entry = zis.nextEntry | ||||||
|  | 				entry?.also { | ||||||
|  | 					val data = readCurrFile() | ||||||
|  | 					val fileLoc = SpaceSafeURI(removeFolder(it.name)) | ||||||
|  | 					if (loc == fileLoc) { | ||||||
|  | 						return data | ||||||
|  | 					} else { | ||||||
|  | 						readFiles[fileLoc] = data | ||||||
|  | 					} | ||||||
|  | 				} ?: return null | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		fun getFileSource(loc: SpaceSafeURI): Source? { | ||||||
|  | 			filesLock.withLock { | ||||||
|  | 				// Assume files are only read once, allow GC by removing | ||||||
|  | 				readFiles.remove(loc)?.also { return it } | ||||||
|  | 				return findFile(loc) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		fun findInZip(matches: Predicate<SpaceSafeURI>): SpaceSafeURI? { | ||||||
|  | 			filesLock.withLock { | ||||||
|  | 				readFiles.keys.find { matches.test(it) }?.let { return it } | ||||||
|  |  | ||||||
|  | 				do { | ||||||
|  | 					val entry = zis.nextEntry?.also { | ||||||
|  | 						val data = readCurrFile() | ||||||
|  | 						val fileLoc = SpaceSafeURI(removeFolder(it.name)) | ||||||
|  | 						readFiles[fileLoc] = data | ||||||
|  | 						if (matches.test(fileLoc)) { | ||||||
|  | 							return fileLoc | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} while (entry != null) | ||||||
|  | 				return null | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private val cache: MutableMap<SpaceSafeURI, ZipReader> = HashMap() | ||||||
|  | 	private val cacheLock = ReentrantReadWriteLock() | ||||||
|  |  | ||||||
|  | 	protected abstract fun getZipUri(loc: SpaceSafeURI): SpaceSafeURI | ||||||
|  | 	protected abstract fun getLocationInZip(loc: SpaceSafeURI): SpaceSafeURI | ||||||
|  | 	abstract override fun matchesHandler(loc: SpaceSafeURI): Boolean | ||||||
|  |  | ||||||
|  | 	override fun getFileSource(loc: SpaceSafeURI): Source? { | ||||||
|  | 		val zipUri = getZipUri(loc) | ||||||
|  | 		var zr = cacheLock.read { cache[zipUri] } | ||||||
|  | 		if (zr == null) { | ||||||
|  | 			cacheLock.write { | ||||||
|  | 				// Recheck, because unlocking read lock allows another thread to modify it | ||||||
|  | 				zr = cache[zipUri] | ||||||
|  |  | ||||||
|  | 				if (zr == null) { | ||||||
|  | 					val src = super.getFileSource(zipUri) ?: return null | ||||||
|  | 					zr = ZipReader(src).also { cache[zipUri] = it } | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return zr?.getFileSource(getLocationInZip(loc)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	protected fun findInZip(loc: SpaceSafeURI, matches: Predicate<SpaceSafeURI>): SpaceSafeURI? { | ||||||
|  | 		val zipUri = getZipUri(loc) | ||||||
|  | 		return (cacheLock.read { cache[zipUri] } ?: cacheLock.write { | ||||||
|  | 			// Recheck, because unlocking read lock allows another thread to modify it | ||||||
|  | 			cache[zipUri] ?: run { | ||||||
|  | 				// Create the ZipReader if it doesn't exist, return null if getFileSource returns null | ||||||
|  | 				super.getFileSource(zipUri)?.let { ZipReader(it) } | ||||||
|  | 						?.also { cache[zipUri] = it } | ||||||
|  | 			} | ||||||
|  | 		})?.findInZip(matches) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | package link.infra.packwiz.installer.ui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.data.ExceptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.InstallProgress | ||||||
|  |  | ||||||
|  | interface IUserInterface { | ||||||
|  | 	fun show() | ||||||
|  | 	fun dispose() | ||||||
|  |  | ||||||
|  | 	fun showErrorAndExit(message: String): Nothing { | ||||||
|  | 		showErrorAndExit(message, null) | ||||||
|  | 	} | ||||||
|  | 	fun showErrorAndExit(message: String, e: Exception?): Nothing | ||||||
|  |  | ||||||
|  | 	var title: String | ||||||
|  | 	fun submitProgress(progress: InstallProgress) | ||||||
|  | 	// Return true if the installation was cancelled! | ||||||
|  | 	fun showOptions(options: List<IOptionDetails>): Boolean | ||||||
|  |  | ||||||
|  | 	fun showExceptions(exceptions: List<ExceptionDetails>, numTotal: Int, allowsIgnore: Boolean): ExceptionListResult | ||||||
|  | 	fun disableOptionsButton(hasOptions: Boolean) {} | ||||||
|  |  | ||||||
|  | 	fun showCancellationDialog(): CancellationResult = CancellationResult.QUIT | ||||||
|  |  | ||||||
|  | 	enum class ExceptionListResult { | ||||||
|  | 		CONTINUE, CANCEL, IGNORE | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	enum class CancellationResult { | ||||||
|  | 		QUIT, CONTINUE | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var optionsButtonPressed: Boolean | ||||||
|  | 	var cancelButtonPressed: Boolean | ||||||
|  |  | ||||||
|  | 	var firstInstall: Boolean | ||||||
|  | } | ||||||
| @@ -0,0 +1,62 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.cli | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface.ExceptionListResult | ||||||
|  | import link.infra.packwiz.installer.ui.data.ExceptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.InstallProgress | ||||||
|  | import link.infra.packwiz.installer.util.Log | ||||||
|  | import kotlin.system.exitProcess | ||||||
|  |  | ||||||
|  | class CLIHandler : IUserInterface { | ||||||
|  | 	@Volatile | ||||||
|  | 	override var optionsButtonPressed = false | ||||||
|  | 	@Volatile | ||||||
|  | 	override var cancelButtonPressed = false | ||||||
|  | 	@Volatile | ||||||
|  | 	override var firstInstall = false | ||||||
|  |  | ||||||
|  | 	override var title: String = "" | ||||||
|  |  | ||||||
|  | 	override fun showErrorAndExit(message: String, e: Exception?): Nothing { | ||||||
|  | 		if (e != null) { | ||||||
|  | 			Log.fatal(message, e) | ||||||
|  | 		} else { | ||||||
|  | 			Log.fatal(message) | ||||||
|  | 		} | ||||||
|  | 		exitProcess(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun show() {} | ||||||
|  | 	override fun dispose() {} | ||||||
|  | 	override fun submitProgress(progress: InstallProgress) { | ||||||
|  | 		val sb = StringBuilder() | ||||||
|  | 		if (progress.hasProgress) { | ||||||
|  | 			sb.append('(') | ||||||
|  | 			sb.append(progress.progress) | ||||||
|  | 			sb.append('/') | ||||||
|  | 			sb.append(progress.progressTotal) | ||||||
|  | 			sb.append(") ") | ||||||
|  | 		} | ||||||
|  | 		sb.append(progress.message) | ||||||
|  | 		println(sb.toString()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showOptions(options: List<IOptionDetails>): Boolean { | ||||||
|  | 		for (opt in options) { | ||||||
|  | 			opt.optionValue = true | ||||||
|  | 			// TODO: implement option choice in the CLI? | ||||||
|  | 			Log.warn("Accepting option ${opt.name} as option choosing is not implemented in the CLI") | ||||||
|  | 		} | ||||||
|  | 		return false // Can't be cancelled! | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showExceptions(exceptions: List<ExceptionDetails>, numTotal: Int, allowsIgnore: Boolean): ExceptionListResult { | ||||||
|  | 		println("Failed to download modpack, the following errors were encountered:") | ||||||
|  | 		for (ex in exceptions) { | ||||||
|  | 			print(ex.name + ": ") | ||||||
|  | 			ex.exception.printStackTrace() | ||||||
|  | 		} | ||||||
|  | 		return ExceptionListResult.CANCEL | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.data | ||||||
|  |  | ||||||
|  | data class ExceptionDetails( | ||||||
|  | 		val name: String, | ||||||
|  | 		val exception: Exception | ||||||
|  | ) | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.data | ||||||
|  |  | ||||||
|  | interface IOptionDetails { | ||||||
|  | 	val name: String | ||||||
|  | 	var optionValue: Boolean | ||||||
|  | 	val optionDescription: String | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.data | ||||||
|  |  | ||||||
|  | data class InstallProgress( | ||||||
|  | 		val message: String, | ||||||
|  | 		val hasProgress: Boolean = false, | ||||||
|  | 		val progress: Int = 0, | ||||||
|  | 		val progressTotal: Int = 0 | ||||||
|  | ) { | ||||||
|  | 	constructor(message: String, progress: Int, progressTotal: Int) : this(message, true, progress, progressTotal) | ||||||
|  |  | ||||||
|  | 	constructor(message: String) : this(message, false) | ||||||
|  | } | ||||||
| @@ -0,0 +1,153 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.gui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface | ||||||
|  | import link.infra.packwiz.installer.ui.data.ExceptionDetails | ||||||
|  | import java.awt.BorderLayout | ||||||
|  | import java.awt.Desktop | ||||||
|  | import java.awt.event.WindowAdapter | ||||||
|  | import java.awt.event.WindowEvent | ||||||
|  | import java.io.IOException | ||||||
|  | import java.io.PrintWriter | ||||||
|  | import java.io.StringWriter | ||||||
|  | import java.net.URI | ||||||
|  | import java.net.URISyntaxException | ||||||
|  | import java.util.concurrent.CompletableFuture | ||||||
|  | import javax.swing.* | ||||||
|  | import javax.swing.border.EmptyBorder | ||||||
|  |  | ||||||
|  | class ExceptionListWindow(eList: List<ExceptionDetails>, future: CompletableFuture<IUserInterface.ExceptionListResult>, numTotal: Int, allowsIgnore: Boolean, parentWindow: JFrame?) : JDialog(parentWindow, "Failed file downloads", true) { | ||||||
|  | 	private val lblExceptionStacktrace: JTextArea | ||||||
|  |  | ||||||
|  | 	private class ExceptionListModel(private val details: List<ExceptionDetails>) : AbstractListModel<String>() { | ||||||
|  | 		override fun getSize() = details.size | ||||||
|  | 		override fun getElementAt(index: Int) = details[index].name | ||||||
|  | 		fun getExceptionAt(index: Int) = details[index].exception | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Create the dialog. | ||||||
|  | 	 */ | ||||||
|  | 	init { | ||||||
|  | 		setBounds(100, 100, 540, 340) | ||||||
|  | 		setLocationRelativeTo(parentWindow) | ||||||
|  |  | ||||||
|  | 		contentPane.apply { | ||||||
|  | 			layout = BorderLayout() | ||||||
|  |  | ||||||
|  | 			// Error panel | ||||||
|  | 			add(JPanel().apply { | ||||||
|  | 				add(JLabel("One or more errors were encountered while installing the modpack!").apply { | ||||||
|  | 					icon = UIManager.getIcon("OptionPane.warningIcon") | ||||||
|  | 				}) | ||||||
|  | 			}, BorderLayout.NORTH) | ||||||
|  |  | ||||||
|  | 			// Content panel | ||||||
|  | 			add(JPanel().apply { | ||||||
|  | 				border = EmptyBorder(5, 5, 5, 5) | ||||||
|  | 				layout = BorderLayout(0, 0) | ||||||
|  |  | ||||||
|  | 				add(JSplitPane().apply { | ||||||
|  | 					resizeWeight = 0.3 | ||||||
|  |  | ||||||
|  | 					lblExceptionStacktrace = JTextArea("Select a file") | ||||||
|  | 					lblExceptionStacktrace.background = UIManager.getColor("List.background") | ||||||
|  | 					lblExceptionStacktrace.isOpaque = true | ||||||
|  | 					lblExceptionStacktrace.wrapStyleWord = true | ||||||
|  | 					lblExceptionStacktrace.lineWrap = true | ||||||
|  | 					lblExceptionStacktrace.isEditable = false | ||||||
|  | 					lblExceptionStacktrace.isFocusable = true | ||||||
|  | 					lblExceptionStacktrace.font = UIManager.getFont("Label.font") | ||||||
|  | 					lblExceptionStacktrace.border = EmptyBorder(5, 5, 5, 5) | ||||||
|  |  | ||||||
|  | 					rightComponent = JScrollPane(lblExceptionStacktrace) | ||||||
|  |  | ||||||
|  | 					leftComponent = JScrollPane(JList<String>().apply { | ||||||
|  | 						selectionMode = ListSelectionModel.SINGLE_SELECTION | ||||||
|  | 						border = EmptyBorder(5, 5, 5, 5) | ||||||
|  | 						val listModel = ExceptionListModel(eList) | ||||||
|  | 						model = listModel | ||||||
|  | 						addListSelectionListener { | ||||||
|  | 							val i = selectedIndex | ||||||
|  | 							if (i > -1) { | ||||||
|  | 								val sw = StringWriter() | ||||||
|  | 								listModel.getExceptionAt(i).printStackTrace(PrintWriter(sw)) | ||||||
|  | 								lblExceptionStacktrace.text = sw.toString() | ||||||
|  | 								// Scroll to the top | ||||||
|  | 								lblExceptionStacktrace.caretPosition = 0 | ||||||
|  | 							} else { | ||||||
|  | 								lblExceptionStacktrace.text = "Select a file" | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  | 				}) | ||||||
|  | 			}, BorderLayout.CENTER) | ||||||
|  |  | ||||||
|  | 			// Button pane | ||||||
|  | 			add(JPanel().apply { | ||||||
|  | 				layout = BorderLayout(0, 0) | ||||||
|  |  | ||||||
|  | 				// Right buttons | ||||||
|  | 				add(JPanel().apply { | ||||||
|  | 					add(JButton("Continue").apply { | ||||||
|  | 						toolTipText = "Attempt to continue installing, excluding the failed downloads" | ||||||
|  | 						addActionListener { | ||||||
|  | 							future.complete(IUserInterface.ExceptionListResult.CONTINUE) | ||||||
|  | 							this@ExceptionListWindow.dispose() | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  |  | ||||||
|  | 					add(JButton("Cancel launch").apply { | ||||||
|  | 						toolTipText = "Stop launching the game" | ||||||
|  | 						addActionListener { | ||||||
|  | 							future.complete(IUserInterface.ExceptionListResult.CANCEL) | ||||||
|  | 							this@ExceptionListWindow.dispose() | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  |  | ||||||
|  | 					add(JButton("Ignore update").apply { | ||||||
|  | 						toolTipText = "Start the game without attempting to update" | ||||||
|  | 						isEnabled = allowsIgnore | ||||||
|  | 						addActionListener { | ||||||
|  | 							future.complete(IUserInterface.ExceptionListResult.IGNORE) | ||||||
|  | 							this@ExceptionListWindow.dispose() | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  | 				}, BorderLayout.EAST) | ||||||
|  |  | ||||||
|  | 				// Errored label | ||||||
|  | 				add(JLabel(eList.size.toString() + "/" + numTotal + " errored").apply { | ||||||
|  | 					horizontalAlignment = SwingConstants.CENTER | ||||||
|  | 				}, BorderLayout.CENTER) | ||||||
|  |  | ||||||
|  | 				// Left buttons | ||||||
|  | 				add(JPanel().apply { | ||||||
|  | 					add(JButton("Report issue").apply { | ||||||
|  | 						if (Desktop.isDesktopSupported() && Desktop.getDesktop().isSupported(Desktop.Action.BROWSE)) { | ||||||
|  | 							addActionListener { | ||||||
|  | 								try { | ||||||
|  | 									Desktop.getDesktop().browse(URI("https://github.com/comp500/packwiz-installer/issues/new")) | ||||||
|  | 								} catch (e: IOException) { | ||||||
|  | 									// lol the button just won't work i guess | ||||||
|  | 								} catch (e: URISyntaxException) {} | ||||||
|  | 							} | ||||||
|  | 						} else { | ||||||
|  | 							isEnabled = false | ||||||
|  | 						} | ||||||
|  | 					}) | ||||||
|  | 				}, BorderLayout.WEST) | ||||||
|  | 			}, BorderLayout.SOUTH) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		addWindowListener(object : WindowAdapter() { | ||||||
|  | 			override fun windowClosing(e: WindowEvent) { | ||||||
|  | 				future.complete(IUserInterface.ExceptionListResult.CANCEL) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			override fun windowClosed(e: WindowEvent) { | ||||||
|  | 				// Just in case closing didn't get triggered - if something else called dispose() the | ||||||
|  | 				// future will have already completed | ||||||
|  | 				future.complete(IUserInterface.ExceptionListResult.CANCEL) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,150 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.gui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface.ExceptionListResult | ||||||
|  | import link.infra.packwiz.installer.ui.data.ExceptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  | import link.infra.packwiz.installer.ui.data.InstallProgress | ||||||
|  | import link.infra.packwiz.installer.util.Log | ||||||
|  | import java.awt.EventQueue | ||||||
|  | import java.util.concurrent.CompletableFuture | ||||||
|  | import javax.swing.JDialog | ||||||
|  | import javax.swing.JOptionPane | ||||||
|  | import javax.swing.UIManager | ||||||
|  | import kotlin.system.exitProcess | ||||||
|  |  | ||||||
|  | class GUIHandler : IUserInterface { | ||||||
|  | 	private lateinit var frmPackwizlauncher: InstallWindow | ||||||
|  |  | ||||||
|  | 	@Volatile | ||||||
|  | 	override var optionsButtonPressed = false | ||||||
|  | 	@Volatile | ||||||
|  | 	override var cancelButtonPressed = false | ||||||
|  | 	@Volatile | ||||||
|  | 	override var firstInstall = false | ||||||
|  |  | ||||||
|  | 	override var title = "packwiz-installer" | ||||||
|  | 		set(value) { | ||||||
|  | 			field = value | ||||||
|  | 			EventQueue.invokeLater { frmPackwizlauncher.title = value } | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	init { | ||||||
|  | 		EventQueue.invokeAndWait { | ||||||
|  | 			try { | ||||||
|  | 				UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()) | ||||||
|  | 			} catch (e: Exception) { | ||||||
|  | 				Log.warn("Failed to set look and feel", e) | ||||||
|  | 			} | ||||||
|  | 			frmPackwizlauncher = InstallWindow(this).apply { | ||||||
|  | 				title = this@GUIHandler.title | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun show() = EventQueue.invokeLater { | ||||||
|  | 		frmPackwizlauncher.isVisible = true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun dispose() = EventQueue.invokeAndWait { | ||||||
|  | 		frmPackwizlauncher.dispose() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showErrorAndExit(message: String, e: Exception?): Nothing { | ||||||
|  | 		val buttons = arrayOf("Quit", if (firstInstall) "Continue without installing" else "Continue without updating") | ||||||
|  | 		if (e != null) { | ||||||
|  | 			Log.fatal(message, e) | ||||||
|  | 			EventQueue.invokeAndWait { | ||||||
|  | 				val result = JOptionPane.showOptionDialog(frmPackwizlauncher, | ||||||
|  | 					"$message: $e", | ||||||
|  | 					title, | ||||||
|  | 					JOptionPane.YES_NO_OPTION, JOptionPane.ERROR_MESSAGE, null, buttons, buttons[0]) | ||||||
|  | 				if (result == 1) { | ||||||
|  | 					Log.info("User selected to continue without installing/updating, exiting with code 0...") | ||||||
|  | 					exitProcess(0) | ||||||
|  | 				} else { | ||||||
|  | 					Log.info("User selected to quit, exiting with code 1...") | ||||||
|  | 					exitProcess(1) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			Log.fatal(message) | ||||||
|  | 			EventQueue.invokeAndWait { | ||||||
|  | 				val result = JOptionPane.showOptionDialog(frmPackwizlauncher, | ||||||
|  | 					message, | ||||||
|  | 					title, | ||||||
|  | 					JOptionPane.YES_NO_OPTION, JOptionPane.ERROR_MESSAGE, null, buttons, buttons[0]) | ||||||
|  | 				if (result == 1) { | ||||||
|  | 					Log.info("User selected to continue without installing/updating, exiting with code 0...") | ||||||
|  | 					exitProcess(0) | ||||||
|  | 				} else { | ||||||
|  | 					Log.info("User selected to quit, exiting with code 1...") | ||||||
|  | 					exitProcess(1) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		exitProcess(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun submitProgress(progress: InstallProgress) { | ||||||
|  | 		val sb = StringBuilder() | ||||||
|  | 		if (progress.hasProgress) { | ||||||
|  | 			sb.append('(') | ||||||
|  | 			sb.append(progress.progress) | ||||||
|  | 			sb.append('/') | ||||||
|  | 			sb.append(progress.progressTotal) | ||||||
|  | 			sb.append(") ") | ||||||
|  | 		} | ||||||
|  | 		sb.append(progress.message) | ||||||
|  | 		Log.info(sb.toString()) | ||||||
|  | 		EventQueue.invokeLater { | ||||||
|  | 			frmPackwizlauncher.displayProgress(progress) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showOptions(options: List<IOptionDetails>): Boolean { | ||||||
|  | 		val future = CompletableFuture<Boolean>() | ||||||
|  | 		EventQueue.invokeAndWait { | ||||||
|  | 			if (options.isEmpty()) { | ||||||
|  | 				JOptionPane.showMessageDialog(null, | ||||||
|  | 					"This modpack has no optional mods!", | ||||||
|  | 					"Optional mods", JOptionPane.INFORMATION_MESSAGE) | ||||||
|  | 				future.complete(false) | ||||||
|  | 			} else { | ||||||
|  | 				OptionsSelectWindow(options, future, frmPackwizlauncher).apply { | ||||||
|  | 					defaultCloseOperation = JDialog.DISPOSE_ON_CLOSE | ||||||
|  | 					isVisible = true | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return future.get() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showExceptions(exceptions: List<ExceptionDetails>, numTotal: Int, allowsIgnore: Boolean): ExceptionListResult { | ||||||
|  | 		val future = CompletableFuture<ExceptionListResult>() | ||||||
|  | 		EventQueue.invokeLater { | ||||||
|  | 			ExceptionListWindow(exceptions, future, numTotal, allowsIgnore, frmPackwizlauncher).apply { | ||||||
|  | 				defaultCloseOperation = JDialog.DISPOSE_ON_CLOSE | ||||||
|  | 				isVisible = true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return future.get() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun disableOptionsButton(hasOptions: Boolean) = EventQueue.invokeLater { | ||||||
|  | 		frmPackwizlauncher.disableOptionsButton(hasOptions) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun showCancellationDialog(): IUserInterface.CancellationResult { | ||||||
|  | 		val future = CompletableFuture<IUserInterface.CancellationResult>() | ||||||
|  | 		EventQueue.invokeLater { | ||||||
|  | 			val buttons = arrayOf("Quit", "Ignore") | ||||||
|  | 			val result = JOptionPane.showOptionDialog(frmPackwizlauncher, | ||||||
|  | 					"The installation was cancelled. Would you like to quit the game, or ignore the update and start the game?", | ||||||
|  | 					"Cancelled installation", | ||||||
|  | 					JOptionPane.YES_NO_OPTION, JOptionPane.QUESTION_MESSAGE, null, buttons, buttons[0]) | ||||||
|  | 			future.complete(if (result == 0) IUserInterface.CancellationResult.QUIT else IUserInterface.CancellationResult.CONTINUE) | ||||||
|  | 		} | ||||||
|  | 		return future.get() | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,86 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.gui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.data.InstallProgress | ||||||
|  | import java.awt.BorderLayout | ||||||
|  | import java.awt.Component | ||||||
|  | import java.awt.GridBagConstraints | ||||||
|  | import java.awt.GridBagLayout | ||||||
|  | import javax.swing.* | ||||||
|  | import javax.swing.border.EmptyBorder | ||||||
|  |  | ||||||
|  | class InstallWindow(private val handler: GUIHandler) : JFrame() { | ||||||
|  | 	private var lblProgresslabel: JLabel | ||||||
|  | 	private var progressBar: JProgressBar | ||||||
|  | 	private var btnOptions: JButton | ||||||
|  |  | ||||||
|  | 	init { | ||||||
|  | 		setBounds(100, 100, 493, 95) | ||||||
|  | 		// Works better with tiling window managers - there isn't any reason to change window size currently anyway | ||||||
|  | 		isResizable = false | ||||||
|  | 		defaultCloseOperation = EXIT_ON_CLOSE | ||||||
|  | 		setLocationRelativeTo(null) | ||||||
|  |  | ||||||
|  | 		// Progress bar and loading text | ||||||
|  | 		add(JPanel().apply { | ||||||
|  | 			border = EmptyBorder(10, 10, 10, 10) | ||||||
|  | 			layout = BorderLayout(0, 0) | ||||||
|  |  | ||||||
|  | 			progressBar = JProgressBar().apply { | ||||||
|  | 				isIndeterminate = true | ||||||
|  | 			} | ||||||
|  | 			add(progressBar, BorderLayout.CENTER) | ||||||
|  |  | ||||||
|  | 			lblProgresslabel = JLabel("Loading...") | ||||||
|  | 			add(lblProgresslabel, BorderLayout.SOUTH) | ||||||
|  | 		}, BorderLayout.CENTER) | ||||||
|  |  | ||||||
|  | 		// Buttons | ||||||
|  | 		add(JPanel().apply { | ||||||
|  | 			border = EmptyBorder(0, 5, 0, 5) | ||||||
|  | 			layout = GridBagLayout() | ||||||
|  |  | ||||||
|  | 			btnOptions = JButton("Optional mods...").apply { | ||||||
|  | 				alignmentX = Component.CENTER_ALIGNMENT | ||||||
|  |  | ||||||
|  | 				addActionListener { | ||||||
|  | 					text = "Loading..." | ||||||
|  | 					isEnabled = false | ||||||
|  | 					handler.optionsButtonPressed = true | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			add(btnOptions, GridBagConstraints().apply { | ||||||
|  | 				gridx = 0 | ||||||
|  | 				gridy = 0 | ||||||
|  | 			}) | ||||||
|  |  | ||||||
|  | 			add(JButton("Cancel").apply { | ||||||
|  | 				addActionListener { | ||||||
|  | 					isEnabled = false | ||||||
|  | 					handler.cancelButtonPressed = true | ||||||
|  | 				} | ||||||
|  | 			}, GridBagConstraints().apply { | ||||||
|  | 				gridx = 0 | ||||||
|  | 				gridy = 1 | ||||||
|  | 			}) | ||||||
|  | 		}, BorderLayout.EAST) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun displayProgress(progress: InstallProgress) { | ||||||
|  | 		if (progress.hasProgress) { | ||||||
|  | 			progressBar.isIndeterminate = false | ||||||
|  | 			progressBar.value = progress.progress | ||||||
|  | 			progressBar.maximum = progress.progressTotal | ||||||
|  | 		} else { | ||||||
|  | 			progressBar.isIndeterminate = true | ||||||
|  | 			progressBar.value = 0 | ||||||
|  | 		} | ||||||
|  | 		lblProgresslabel.text = progress.message | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fun disableOptionsButton(hasOptions: Boolean) { | ||||||
|  | 		btnOptions.apply { | ||||||
|  | 			text = if (hasOptions) { "Optional mods..." } else { "No optional mods" } | ||||||
|  |  			isEnabled = false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.gui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  |  | ||||||
|  | // Serves as a proxy for IOptionDetails, so that setOptionValue isn't called until OK is clicked | ||||||
|  | internal class OptionTempHandler(private val opt: IOptionDetails) : IOptionDetails { | ||||||
|  | 	override var optionValue = opt.optionValue | ||||||
|  |  | ||||||
|  | 	override val name get() = opt.name | ||||||
|  | 	override val optionDescription get() = opt.optionDescription | ||||||
|  |  | ||||||
|  | 	fun finalise() { | ||||||
|  | 		opt.optionValue = optionValue | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,167 @@ | |||||||
|  | package link.infra.packwiz.installer.ui.gui | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.data.IOptionDetails | ||||||
|  | import java.awt.BorderLayout | ||||||
|  | import java.awt.FlowLayout | ||||||
|  | import java.awt.event.ActionEvent | ||||||
|  | import java.awt.event.ActionListener | ||||||
|  | import java.awt.event.WindowAdapter | ||||||
|  | import java.awt.event.WindowEvent | ||||||
|  | import java.util.* | ||||||
|  | import java.util.concurrent.CompletableFuture | ||||||
|  | import javax.swing.* | ||||||
|  | import javax.swing.border.EmptyBorder | ||||||
|  | import javax.swing.event.TableModelListener | ||||||
|  | import javax.swing.table.TableModel | ||||||
|  |  | ||||||
|  | class OptionsSelectWindow internal constructor(optList: List<IOptionDetails>, future: CompletableFuture<Boolean>, parentWindow: JFrame?) : JDialog(parentWindow, "Select optional mods...", true), ActionListener { | ||||||
|  | 	private val lblOptionDescription: JTextArea | ||||||
|  | 	private val tableModel: OptionTableModel | ||||||
|  | 	private val future: CompletableFuture<Boolean> | ||||||
|  |  | ||||||
|  | 	private class OptionTableModel(givenOpts: List<IOptionDetails>) : TableModel { | ||||||
|  | 		private val opts: List<OptionTempHandler> | ||||||
|  |  | ||||||
|  | 		init { | ||||||
|  | 			val mutOpts = ArrayList<OptionTempHandler>() | ||||||
|  | 			for (opt in givenOpts) { | ||||||
|  | 				mutOpts.add(OptionTempHandler(opt)) | ||||||
|  | 			} | ||||||
|  | 			opts = mutOpts | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		override fun getRowCount() = opts.size | ||||||
|  | 		override fun getColumnCount() = 2 | ||||||
|  |  | ||||||
|  | 		private val columnNames = arrayOf("Enabled", "Mod name") | ||||||
|  | 		private val columnTypes = arrayOf(Boolean::class.javaObjectType, String::class.java) | ||||||
|  | 		private val columnEditables = booleanArrayOf(true, false) | ||||||
|  |  | ||||||
|  | 		override fun getColumnName(columnIndex: Int) = columnNames[columnIndex] | ||||||
|  | 		override fun getColumnClass(columnIndex: Int) = columnTypes[columnIndex] | ||||||
|  | 		override fun isCellEditable(rowIndex: Int, columnIndex: Int) = columnEditables[columnIndex] | ||||||
|  |  | ||||||
|  | 		override fun getValueAt(rowIndex: Int, columnIndex: Int): Any { | ||||||
|  | 			val opt = opts[rowIndex] | ||||||
|  | 			return if (columnIndex == 0) opt.optionValue else opt.name | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		override fun setValueAt(aValue: Any, rowIndex: Int, columnIndex: Int) { | ||||||
|  | 			if (columnIndex == 0) { | ||||||
|  | 				val opt = opts[rowIndex] | ||||||
|  | 				opt.optionValue = aValue as Boolean | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Noop, the table model doesn't change! | ||||||
|  | 		override fun addTableModelListener(l: TableModelListener) {} | ||||||
|  | 		override fun removeTableModelListener(l: TableModelListener) {} | ||||||
|  |  | ||||||
|  | 		fun getDescription(rowIndex: Int) = opts[rowIndex].optionDescription | ||||||
|  |  | ||||||
|  | 		fun finalise() { | ||||||
|  | 			for (opt in opts) { | ||||||
|  | 				opt.finalise() | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	override fun actionPerformed(e: ActionEvent) { | ||||||
|  | 		if (e.actionCommand == "OK") { | ||||||
|  | 			tableModel.finalise() | ||||||
|  | 			future.complete(false) | ||||||
|  | 			dispose() | ||||||
|  | 		} else if (e.actionCommand == "Cancel") { | ||||||
|  | 			future.complete(true) | ||||||
|  | 			dispose() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Create the dialog. | ||||||
|  | 	 */ | ||||||
|  | 	init { | ||||||
|  | 		tableModel = OptionTableModel(optList) | ||||||
|  | 		this.future = future | ||||||
|  |  | ||||||
|  | 		setBounds(100, 100, 450, 300) | ||||||
|  | 		setLocationRelativeTo(parentWindow) | ||||||
|  |  | ||||||
|  | 		contentPane.apply { | ||||||
|  | 			layout = BorderLayout() | ||||||
|  | 			add(JPanel().apply { | ||||||
|  | 				border = EmptyBorder(5, 5, 5, 5) | ||||||
|  | 				layout = BorderLayout(0, 0) | ||||||
|  |  | ||||||
|  | 				add(JSplitPane().apply { | ||||||
|  | 					resizeWeight = 0.5 | ||||||
|  |  | ||||||
|  | 					lblOptionDescription = JTextArea("Select an option...").apply { | ||||||
|  | 						background = UIManager.getColor("List.background") | ||||||
|  | 						isOpaque = true | ||||||
|  | 						wrapStyleWord = true | ||||||
|  | 						lineWrap = true | ||||||
|  | 						isEditable = false | ||||||
|  | 						isFocusable = false | ||||||
|  | 						font = UIManager.getFont("Label.font") | ||||||
|  | 						border = EmptyBorder(10, 10, 10, 10) | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					leftComponent = JScrollPane(JTable().apply { | ||||||
|  | 						showVerticalLines = false | ||||||
|  | 						showHorizontalLines = false | ||||||
|  | 						setSelectionMode(ListSelectionModel.SINGLE_SELECTION) | ||||||
|  | 						setShowGrid(false) | ||||||
|  | 						model = tableModel | ||||||
|  | 						columnModel.getColumn(0).resizable = false | ||||||
|  | 						columnModel.getColumn(0).preferredWidth = 15 | ||||||
|  | 						columnModel.getColumn(0).maxWidth = 15 | ||||||
|  | 						columnModel.getColumn(1).resizable = false | ||||||
|  | 						selectionModel.addListSelectionListener { | ||||||
|  | 							val i = selectedRow | ||||||
|  | 							if (i > -1) { | ||||||
|  | 								lblOptionDescription.text = tableModel.getDescription(i) | ||||||
|  | 							} else { | ||||||
|  | 								lblOptionDescription.text = "Select an option..." | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 						tableHeader = null | ||||||
|  | 					}).apply { | ||||||
|  | 						viewport.background = UIManager.getColor("List.background") | ||||||
|  | 						horizontalScrollBarPolicy = ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					rightComponent = JScrollPane(lblOptionDescription) | ||||||
|  | 				}) | ||||||
|  |  | ||||||
|  | 				add(JPanel().apply { | ||||||
|  | 					layout = FlowLayout(FlowLayout.RIGHT) | ||||||
|  |  | ||||||
|  | 					add(JButton("OK").apply { | ||||||
|  | 						actionCommand = "OK" | ||||||
|  | 						addActionListener(this@OptionsSelectWindow) | ||||||
|  |  | ||||||
|  | 						this@OptionsSelectWindow.rootPane.defaultButton = this | ||||||
|  | 					}) | ||||||
|  |  | ||||||
|  | 					add(JButton("Cancel").apply { | ||||||
|  | 						actionCommand = "Cancel" | ||||||
|  | 						addActionListener(this@OptionsSelectWindow) | ||||||
|  | 					}) | ||||||
|  | 				}, BorderLayout.SOUTH) | ||||||
|  | 			}, BorderLayout.CENTER) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		addWindowListener(object : WindowAdapter() { | ||||||
|  | 			override fun windowClosing(e: WindowEvent) { | ||||||
|  | 				future.complete(true) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			override fun windowClosed(e: WindowEvent) { | ||||||
|  | 				// Just in case closing didn't get triggered - if something else called dispose() the | ||||||
|  | 				// future will have already completed | ||||||
|  | 				future.complete(true) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										38
									
								
								src/main/kotlin/link/infra/packwiz/installer/util/Exts.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								src/main/kotlin/link/infra/packwiz/installer/util/Exts.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | package link.infra.packwiz.installer.util | ||||||
|  |  | ||||||
|  | import link.infra.packwiz.installer.ui.IUserInterface | ||||||
|  |  | ||||||
|  | inline fun <T> iflet(value: T?, whenNotNull: (T) -> Unit) { | ||||||
|  | 	if (value != null) { | ||||||
|  | 		whenNotNull(value) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline fun <T, U> IUserInterface.ifletOrErr(value: T?, message: String, whenNotNull: (T) -> U): U = | ||||||
|  | 	if (value != null) { | ||||||
|  | 		whenNotNull(value) | ||||||
|  | 	} else { | ||||||
|  | 		this.showErrorAndExit(message) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | inline fun <T, U, V> IUserInterface.ifletOrErr(value: T?, value2: U?, message: String, whenNotNull: (T, U) -> V): V = | ||||||
|  | 	if (value != null && value2 != null) { | ||||||
|  | 		whenNotNull(value, value2) | ||||||
|  | 	} else { | ||||||
|  | 		this.showErrorAndExit(message) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | inline fun <T> ifletOrWarn(value: T?, message: String, whenNotNull: (T) -> Unit) { | ||||||
|  | 	if (value != null) { | ||||||
|  | 		whenNotNull(value) | ||||||
|  | 	} else { | ||||||
|  | 		Log.warn(message) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline fun <T, U> iflet(value: T?, whenNotNull: (T) -> U, whenNull: () -> U): U = | ||||||
|  | 	if (value != null) { | ||||||
|  | 		whenNotNull(value) | ||||||
|  | 	} else { | ||||||
|  | 		whenNull() | ||||||
|  | 	} | ||||||
							
								
								
									
										16
									
								
								src/main/kotlin/link/infra/packwiz/installer/util/Log.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/main/kotlin/link/infra/packwiz/installer/util/Log.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | package link.infra.packwiz.installer.util | ||||||
|  |  | ||||||
|  | object Log { | ||||||
|  | 	fun info(message: String) = println(message) | ||||||
|  |  | ||||||
|  | 	fun warn(message: String) = println("[Warning] $message") | ||||||
|  | 	fun warn(message: String, exception: Exception) = println("[Warning] $message: $exception") | ||||||
|  |  | ||||||
|  | 	fun fatal(message: String) { | ||||||
|  | 		println("[FATAL] $message") | ||||||
|  | 	} | ||||||
|  | 	fun fatal(message: String, exception: Exception) { | ||||||
|  | 		println("[FATAL] $message: ") | ||||||
|  | 		exception.printStackTrace() | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										250
									
								
								src/main/resources/META-INF/LICENSES.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								src/main/resources/META-INF/LICENSES.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,250 @@ | |||||||
|  | # Licenses | ||||||
|  |  | ||||||
|  | packwiz-installer itself is under the MIT license, except for Murmur2Lib and bundled dependencies as follows: | ||||||
|  |  | ||||||
|  | - Murmur2Lib: Apache 2.0 ([Source](https://github.com/prasanthj/hasher/blob/master/src/main/java/hasher/Murmur2.java)) | ||||||
|  | - Google Gson 2.8.1: Apache 2.0 ([Source](https://github.com/google/gson)) | ||||||
|  | - Okio 2.9.0: Apache 2.0 ([Source](https://github.com/square/okio/)) | ||||||
|  | - Commons CLI 1.4: Apache 2.0 ([Source](http://commons.apache.org/proper/commons-cli/)) | ||||||
|  | - Jetbrains Annotations 13.0: Apache 2.0 ([Source](https://github.com/JetBrains/java-annotations)) | ||||||
|  | - Kotlin Standard Library 1.4.21: Apache 2.0 ([Source](https://github.com/JetBrains/kotlin)) | ||||||
|  | - toml4j 0.7.2: MIT ([Source](https://github.com/mwanji/toml4j)) | ||||||
|  |  | ||||||
|  | ## Associated notices | ||||||
|  |  | ||||||
|  | ### Commons CLI | ||||||
|  | Apache Commons CLI | ||||||
|  | Copyright 2001-2017 The Apache Software Foundation | ||||||
|  |  | ||||||
|  | This product includes software developed at | ||||||
|  | The Apache Software Foundation (http://www.apache.org/). | ||||||
|  |  | ||||||
|  | ## Full license texts | ||||||
|  |  | ||||||
|  | ### MIT | ||||||
|  | MIT License | ||||||
|  |  | ||||||
|  | Copyright (c) | ||||||
|  |  | ||||||
|  | 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. | ||||||
|  |  | ||||||
|  | ### Apache 2.0 | ||||||
|  |  | ||||||
|  |                                  Apache License | ||||||
|  |                            Version 2.0, January 2004 | ||||||
|  |                         http://www.apache.org/licenses/ | ||||||
|  |  | ||||||
|  | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
|  |  | ||||||
|  | 1. Definitions. | ||||||
|  |  | ||||||
|  |    "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |    and distribution as defined by Sections 1 through 9 of this document. | ||||||
|  |  | ||||||
|  |    "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|  |    the copyright owner that is granting the License. | ||||||
|  |  | ||||||
|  |    "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |    other entities that control, are controlled by, or are under common | ||||||
|  |    control with that entity. For the purposes of this definition, | ||||||
|  |    "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |    direction or management of such entity, whether by contract or | ||||||
|  |    otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |    outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
|  |  | ||||||
|  |    "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |    exercising permissions granted by this License. | ||||||
|  |  | ||||||
|  |    "Source" form shall mean the preferred form for making modifications, | ||||||
|  |    including but not limited to software source code, documentation | ||||||
|  |    source, and configuration files. | ||||||
|  |  | ||||||
|  |    "Object" form shall mean any form resulting from mechanical | ||||||
|  |    transformation or translation of a Source form, including but | ||||||
|  |    not limited to compiled object code, generated documentation, | ||||||
|  |    and conversions to other media types. | ||||||
|  |  | ||||||
|  |    "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |    Object form, made available under the License, as indicated by a | ||||||
|  |    copyright notice that is included in or attached to the work | ||||||
|  |    (an example is provided in the Appendix below). | ||||||
|  |  | ||||||
|  |    "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |    form, that is based on (or derived from) the Work and for which the | ||||||
|  |    editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |    represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |    of this License, Derivative Works shall not include works that remain | ||||||
|  |    separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |    the Work and Derivative Works thereof. | ||||||
|  |  | ||||||
|  |    "Contribution" shall mean any work of authorship, including | ||||||
|  |    the original version of the Work and any modifications or additions | ||||||
|  |    to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |    submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |    or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |    the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |    means any form of electronic, verbal, or written communication sent | ||||||
|  |    to the Licensor or its representatives, including but not limited to | ||||||
|  |    communication on electronic mailing lists, source code control systems, | ||||||
|  |    and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |    Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |    excluding communication that is conspicuously marked or otherwise | ||||||
|  |    designated in writing by the copyright owner as "Not a Contribution." | ||||||
|  |  | ||||||
|  |    "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |    on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |    subsequently incorporated within the Work. | ||||||
|  |  | ||||||
|  | 2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
|  |    this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |    worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |    copyright license to reproduce, prepare Derivative Works of, | ||||||
|  |    publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |    Work and such Derivative Works in Source or Object form. | ||||||
|  |  | ||||||
|  | 3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |    this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |    worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |    (except as stated in this section) patent license to make, have made, | ||||||
|  |    use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |    where such license applies only to those patent claims licensable | ||||||
|  |    by such Contributor that are necessarily infringed by their | ||||||
|  |    Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |    with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |    institute patent litigation against any entity (including a | ||||||
|  |    cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |    or a Contribution incorporated within the Work constitutes direct | ||||||
|  |    or contributory patent infringement, then any patent licenses | ||||||
|  |    granted to You under this License for that Work shall terminate | ||||||
|  |    as of the date such litigation is filed. | ||||||
|  |  | ||||||
|  | 4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |    Work or Derivative Works thereof in any medium, with or without | ||||||
|  |    modifications, and in Source or Object form, provided that You | ||||||
|  |    meet the following conditions: | ||||||
|  |  | ||||||
|  |    (a) You must give any other recipients of the Work or | ||||||
|  |    Derivative Works a copy of this License; and | ||||||
|  |  | ||||||
|  |    (b) You must cause any modified files to carry prominent notices | ||||||
|  |    stating that You changed the files; and | ||||||
|  |  | ||||||
|  |    (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |    that You distribute, all copyright, patent, trademark, and | ||||||
|  |    attribution notices from the Source form of the Work, | ||||||
|  |    excluding those notices that do not pertain to any part of | ||||||
|  |    the Derivative Works; and | ||||||
|  |  | ||||||
|  |    (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |    distribution, then any Derivative Works that You distribute must | ||||||
|  |    include a readable copy of the attribution notices contained | ||||||
|  |    within such NOTICE file, excluding those notices that do not | ||||||
|  |    pertain to any part of the Derivative Works, in at least one | ||||||
|  |    of the following places: within a NOTICE text file distributed | ||||||
|  |    as part of the Derivative Works; within the Source form or | ||||||
|  |    documentation, if provided along with the Derivative Works; or, | ||||||
|  |    within a display generated by the Derivative Works, if and | ||||||
|  |    wherever such third-party notices normally appear. The contents | ||||||
|  |    of the NOTICE file are for informational purposes only and | ||||||
|  |    do not modify the License. You may add Your own attribution | ||||||
|  |    notices within Derivative Works that You distribute, alongside | ||||||
|  |    or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |    that such additional attribution notices cannot be construed | ||||||
|  |    as modifying the License. | ||||||
|  |  | ||||||
|  |    You may add Your own copyright statement to Your modifications and | ||||||
|  |    may provide additional or different license terms and conditions | ||||||
|  |    for use, reproduction, or distribution of Your modifications, or | ||||||
|  |    for any such Derivative Works as a whole, provided Your use, | ||||||
|  |    reproduction, and distribution of the Work otherwise complies with | ||||||
|  |    the conditions stated in this License. | ||||||
|  |  | ||||||
|  | 5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |    any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |    by You to the Licensor shall be under the terms and conditions of | ||||||
|  |    this License, without any additional terms or conditions. | ||||||
|  |    Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |    the terms of any separate license agreement you may have executed | ||||||
|  |    with Licensor regarding such Contributions. | ||||||
|  |  | ||||||
|  | 6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |    names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |    except as required for reasonable and customary use in describing the | ||||||
|  |    origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  |  | ||||||
|  | 7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |    agreed to in writing, Licensor provides the Work (and each | ||||||
|  |    Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |    implied, including, without limitation, any warranties or conditions | ||||||
|  |    of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |    PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |    appropriateness of using or redistributing the Work and assume any | ||||||
|  |    risks associated with Your exercise of permissions under this License. | ||||||
|  |  | ||||||
|  | 8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |    whether in tort (including negligence), contract, or otherwise, | ||||||
|  |    unless required by applicable law (such as deliberate and grossly | ||||||
|  |    negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |    liable to You for damages, including any direct, indirect, special, | ||||||
|  |    incidental, or consequential damages of any character arising as a | ||||||
|  |    result of this License or out of the use or inability to use the | ||||||
|  |    Work (including but not limited to damages for loss of goodwill, | ||||||
|  |    work stoppage, computer failure or malfunction, or any and all | ||||||
|  |    other commercial damages or losses), even if such Contributor | ||||||
|  |    has been advised of the possibility of such damages. | ||||||
|  |  | ||||||
|  | 9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |    the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |    and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |    or other liability obligations and/or rights consistent with this | ||||||
|  |    License. However, in accepting such obligations, You may act only | ||||||
|  |    on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |    of any other Contributor, and only if You agree to indemnify, | ||||||
|  |    defend, and hold each Contributor harmless for any liability | ||||||
|  |    incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |    of your accepting any such warranty or additional liability. | ||||||
|  |  | ||||||
|  | END OF TERMS AND CONDITIONS | ||||||
|  |  | ||||||
|  | APPENDIX: How to apply the Apache License to your work. | ||||||
|  |  | ||||||
|  |       To apply the Apache License to your work, attach the following | ||||||
|  |       boilerplate notice, with the fields enclosed by brackets "[]" | ||||||
|  |       replaced with your own identifying information. (Don't include | ||||||
|  |       the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |       comment syntax for the file format. We also recommend that a | ||||||
|  |       file or class name and description of purpose be included on the | ||||||
|  |       same "printed page" as the copyright notice for easier | ||||||
|  |       identification within third-party archives. | ||||||
|  |  | ||||||
|  | Copyright [yyyy] [name of copyright owner] | ||||||
|  |  | ||||||
|  | Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | you may not use this file except in compliance with the License. | ||||||
|  | You may obtain a copy of the License at | ||||||
|  |  | ||||||
|  |        http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  |  | ||||||
|  | Unless required by applicable law or agreed to in writing, software | ||||||
|  | distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | See the License for the specific language governing permissions and | ||||||
|  | limitations under the License. | ||||||
|  |  | ||||||
		Reference in New Issue
	
	Block a user