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 | ||||
| # Edit at https://www.gitignore.io/?templates=java,gradle,intellij | ||||
| # Created by https://www.toptal.com/developers/gitignore/api/java,gradle,intellij+all | ||||
| # Edit at https://www.toptal.com/developers/gitignore?templates=java,gradle,intellij+all | ||||
|  | ||||
| ### Intellij ### | ||||
| # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and WebStorm | ||||
| ### Intellij+all ### | ||||
| # 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 | ||||
|  | ||||
| # User-specific stuff | ||||
| @@ -33,6 +33,9 @@ | ||||
| # 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 | ||||
| # auto-import. | ||||
| # .idea/artifacts | ||||
| # .idea/compiler.xml | ||||
| # .idea/jarRepositories.xml | ||||
| # .idea/modules.xml | ||||
| # .idea/*.iml | ||||
| # .idea/modules | ||||
| @@ -72,13 +75,18 @@ fabric.properties | ||||
| # Android studio 3.1+ serialized cache file | ||||
| .idea/caches/build_file_checksums.ser | ||||
|  | ||||
| ### Intellij Patch ### | ||||
| # Comment Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-215987721 | ||||
| ### Intellij+all Patch ### | ||||
| # 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 | ||||
| # modules.xml | ||||
| # .idea/misc.xml | ||||
| # *.ipr | ||||
| .idea/ | ||||
|  | ||||
| # Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-249601023 | ||||
|  | ||||
| *.iml | ||||
| modules.xml | ||||
| .idea/misc.xml | ||||
| *.ipr | ||||
|  | ||||
| # Sonarlint plugin | ||||
| .idea/sonarlint | ||||
| @@ -127,4 +135,4 @@ gradle-app.setting | ||||
| ### Gradle Patch ### | ||||
| **/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 | ||||
|  | ||||
| Copyright (c) 2019 | ||||
| Copyright (c) 2021 comp500 | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
|   | ||||
| @@ -1,2 +1,2 @@ | ||||
| # 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 | ||||
| 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 | ||||
| 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; | ||||
|     if (left != 0) { | ||||
|       if (left >= 3) { | ||||
|         h ^= (int) data[length - 3] << 16; | ||||
|         h ^= (int) data[length - (left - 2)] << 16; | ||||
|       } | ||||
|       if (left >= 2) { | ||||
|         h ^= (int) data[length - 2] << 8; | ||||
|         h ^= (int) data[length - (left - 1)] << 8; | ||||
|       } | ||||
|       if (left >= 1) { | ||||
|         h ^= (int) data[length - 1]; | ||||
|         h ^= data[length - left]; | ||||
|       } | ||||
|  | ||||
|       h *= M_32; | ||||
| @@ -152,7 +152,7 @@ public class Murmur2Lib { | ||||
|       case 2: | ||||
|         h ^= (long) (data[tailStart + 1] & 0xff) << 8; | ||||
|       case 1: | ||||
|         h ^= (long) (data[tailStart] & 0xff); | ||||
|         h ^= data[tailStart] & 0xff; | ||||
|         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,17 +1,16 @@ | ||||
| package link.infra.packwiz.installer.request; | ||||
| package link.infra.packwiz.installer.request | ||||
| 
 | ||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI; | ||||
| import okio.Source; | ||||
| import link.infra.packwiz.installer.metadata.SpaceSafeURI | ||||
| import okio.Source | ||||
| 
 | ||||
| /** | ||||
|  * IRequestHandler handles requests for locations specified in modpack metadata. | ||||
|  */ | ||||
| public interface IRequestHandler { | ||||
| interface IRequestHandler { | ||||
| 	fun matchesHandler(loc: SpaceSafeURI): Boolean | ||||
| 
 | ||||
| 	boolean matchesHandler(SpaceSafeURI loc); | ||||
| 	 | ||||
| 	default SpaceSafeURI getNewLoc(SpaceSafeURI loc) { | ||||
| 		return loc; | ||||
| 	fun getNewLoc(loc: SpaceSafeURI): SpaceSafeURI { | ||||
| 		return loc | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| @@ -21,6 +20,5 @@ public interface IRequestHandler { | ||||
| 	 * @return The Source containing the data of the 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