mirror of
				https://github.com/JonasunderscoreJones/YetAnotherDiscordChatLink.git
				synced 2025-10-23 00:39:17 +02:00 
			
		
		
		
	Initial Commit (x2)
This commit is contained in:
		
						commit
						38b0f89dd5
					
				
					 22 changed files with 1230 additions and 0 deletions
				
			
		
							
								
								
									
										40
									
								
								.github/workflows/build.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.github/workflows/build.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,40 @@ | ||||||
|  | # Automatically build the project and run any configured tests for every push | ||||||
|  | # and submitted pull request. This can help catch issues that only occur on | ||||||
|  | # certain platforms or Java versions, and provides a first line of defence | ||||||
|  | # against bad commits. | ||||||
|  | 
 | ||||||
|  | name: build | ||||||
|  | on: [pull_request, push] | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         # Use these Java versions | ||||||
|  |         java: [ | ||||||
|  |           17,    # Current Java LTS & minimum supported by Minecraft | ||||||
|  |         ] | ||||||
|  |         # and run on both Linux and Windows | ||||||
|  |         os: [ubuntu-22.04, windows-2022] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |       - name: checkout repository | ||||||
|  |         uses: actions/checkout@v3 | ||||||
|  |       - name: validate gradle wrapper | ||||||
|  |         uses: gradle/wrapper-validation-action@v1 | ||||||
|  |       - name: setup jdk ${{ matrix.java }} | ||||||
|  |         uses: actions/setup-java@v3 | ||||||
|  |         with: | ||||||
|  |           java-version: ${{ matrix.java }} | ||||||
|  |           distribution: 'microsoft' | ||||||
|  |       - name: make gradle wrapper executable | ||||||
|  |         if: ${{ runner.os != 'Windows' }} | ||||||
|  |         run: chmod +x ./gradlew | ||||||
|  |       - name: build | ||||||
|  |         run: ./gradlew build | ||||||
|  |       - name: capture build artifacts | ||||||
|  |         if: ${{ runner.os == 'Linux' && matrix.java == '17' }} # Only upload artifacts built from latest java on one OS | ||||||
|  |         uses: actions/upload-artifact@v3 | ||||||
|  |         with: | ||||||
|  |           name: Artifacts | ||||||
|  |           path: build/libs/ | ||||||
							
								
								
									
										40
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,40 @@ | ||||||
|  | # gradle | ||||||
|  | 
 | ||||||
|  | .gradle/ | ||||||
|  | build/ | ||||||
|  | out/ | ||||||
|  | classes/ | ||||||
|  | 
 | ||||||
|  | # eclipse | ||||||
|  | 
 | ||||||
|  | *.launch | ||||||
|  | 
 | ||||||
|  | # idea | ||||||
|  | 
 | ||||||
|  | .idea/ | ||||||
|  | *.iml | ||||||
|  | *.ipr | ||||||
|  | *.iws | ||||||
|  | 
 | ||||||
|  | # vscode | ||||||
|  | 
 | ||||||
|  | .settings/ | ||||||
|  | .vscode/ | ||||||
|  | bin/ | ||||||
|  | .classpath | ||||||
|  | .project | ||||||
|  | 
 | ||||||
|  | # macos | ||||||
|  | 
 | ||||||
|  | *.DS_Store | ||||||
|  | 
 | ||||||
|  | # fabric | ||||||
|  | 
 | ||||||
|  | run/ | ||||||
|  | 
 | ||||||
|  | # java | ||||||
|  | 
 | ||||||
|  | hs_err_*.log | ||||||
|  | replay_*.log | ||||||
|  | *.hprof | ||||||
|  | *.jfr | ||||||
							
								
								
									
										121
									
								
								LICENSE
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								LICENSE
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,121 @@ | ||||||
|  | Creative Commons Legal Code | ||||||
|  | 
 | ||||||
|  | CC0 1.0 Universal | ||||||
|  | 
 | ||||||
|  |     CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE | ||||||
|  |     LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN | ||||||
|  |     ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS | ||||||
|  |     INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES | ||||||
|  |     REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS | ||||||
|  |     PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM | ||||||
|  |     THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED | ||||||
|  |     HEREUNDER. | ||||||
|  | 
 | ||||||
|  | Statement of Purpose | ||||||
|  | 
 | ||||||
|  | The laws of most jurisdictions throughout the world automatically confer | ||||||
|  | exclusive Copyright and Related Rights (defined below) upon the creator | ||||||
|  | and subsequent owner(s) (each and all, an "owner") of an original work of | ||||||
|  | authorship and/or a database (each, a "Work"). | ||||||
|  | 
 | ||||||
|  | Certain owners wish to permanently relinquish those rights to a Work for | ||||||
|  | the purpose of contributing to a commons of creative, cultural and | ||||||
|  | scientific works ("Commons") that the public can reliably and without fear | ||||||
|  | of later claims of infringement build upon, modify, incorporate in other | ||||||
|  | works, reuse and redistribute as freely as possible in any form whatsoever | ||||||
|  | and for any purposes, including without limitation commercial purposes. | ||||||
|  | These owners may contribute to the Commons to promote the ideal of a free | ||||||
|  | culture and the further production of creative, cultural and scientific | ||||||
|  | works, or to gain reputation or greater distribution for their Work in | ||||||
|  | part through the use and efforts of others. | ||||||
|  | 
 | ||||||
|  | For these and/or other purposes and motivations, and without any | ||||||
|  | expectation of additional consideration or compensation, the person | ||||||
|  | associating CC0 with a Work (the "Affirmer"), to the extent that he or she | ||||||
|  | is an owner of Copyright and Related Rights in the Work, voluntarily | ||||||
|  | elects to apply CC0 to the Work and publicly distribute the Work under its | ||||||
|  | terms, with knowledge of his or her Copyright and Related Rights in the | ||||||
|  | Work and the meaning and intended legal effect of CC0 on those rights. | ||||||
|  | 
 | ||||||
|  | 1. Copyright and Related Rights. A Work made available under CC0 may be | ||||||
|  | protected by copyright and related or neighboring rights ("Copyright and | ||||||
|  | Related Rights"). Copyright and Related Rights include, but are not | ||||||
|  | limited to, the following: | ||||||
|  | 
 | ||||||
|  |   i. the right to reproduce, adapt, distribute, perform, display, | ||||||
|  |      communicate, and translate a Work; | ||||||
|  |  ii. moral rights retained by the original author(s) and/or performer(s); | ||||||
|  | iii. publicity and privacy rights pertaining to a person's image or | ||||||
|  |      likeness depicted in a Work; | ||||||
|  |  iv. rights protecting against unfair competition in regards to a Work, | ||||||
|  |      subject to the limitations in paragraph 4(a), below; | ||||||
|  |   v. rights protecting the extraction, dissemination, use and reuse of data | ||||||
|  |      in a Work; | ||||||
|  |  vi. database rights (such as those arising under Directive 96/9/EC of the | ||||||
|  |      European Parliament and of the Council of 11 March 1996 on the legal | ||||||
|  |      protection of databases, and under any national implementation | ||||||
|  |      thereof, including any amended or successor version of such | ||||||
|  |      directive); and | ||||||
|  | vii. other similar, equivalent or corresponding rights throughout the | ||||||
|  |      world based on applicable law or treaty, and any national | ||||||
|  |      implementations thereof. | ||||||
|  | 
 | ||||||
|  | 2. Waiver. To the greatest extent permitted by, but not in contravention | ||||||
|  | of, applicable law, Affirmer hereby overtly, fully, permanently, | ||||||
|  | irrevocably and unconditionally waives, abandons, and surrenders all of | ||||||
|  | Affirmer's Copyright and Related Rights and associated claims and causes | ||||||
|  | of action, whether now known or unknown (including existing as well as | ||||||
|  | future claims and causes of action), in the Work (i) in all territories | ||||||
|  | worldwide, (ii) for the maximum duration provided by applicable law or | ||||||
|  | treaty (including future time extensions), (iii) in any current or future | ||||||
|  | medium and for any number of copies, and (iv) for any purpose whatsoever, | ||||||
|  | including without limitation commercial, advertising or promotional | ||||||
|  | purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each | ||||||
|  | member of the public at large and to the detriment of Affirmer's heirs and | ||||||
|  | successors, fully intending that such Waiver shall not be subject to | ||||||
|  | revocation, rescission, cancellation, termination, or any other legal or | ||||||
|  | equitable action to disrupt the quiet enjoyment of the Work by the public | ||||||
|  | as contemplated by Affirmer's express Statement of Purpose. | ||||||
|  | 
 | ||||||
|  | 3. Public License Fallback. Should any part of the Waiver for any reason | ||||||
|  | be judged legally invalid or ineffective under applicable law, then the | ||||||
|  | Waiver shall be preserved to the maximum extent permitted taking into | ||||||
|  | account Affirmer's express Statement of Purpose. In addition, to the | ||||||
|  | extent the Waiver is so judged Affirmer hereby grants to each affected | ||||||
|  | person a royalty-free, non transferable, non sublicensable, non exclusive, | ||||||
|  | irrevocable and unconditional license to exercise Affirmer's Copyright and | ||||||
|  | Related Rights in the Work (i) in all territories worldwide, (ii) for the | ||||||
|  | maximum duration provided by applicable law or treaty (including future | ||||||
|  | time extensions), (iii) in any current or future medium and for any number | ||||||
|  | of copies, and (iv) for any purpose whatsoever, including without | ||||||
|  | limitation commercial, advertising or promotional purposes (the | ||||||
|  | "License"). The License shall be deemed effective as of the date CC0 was | ||||||
|  | applied by Affirmer to the Work. Should any part of the License for any | ||||||
|  | reason be judged legally invalid or ineffective under applicable law, such | ||||||
|  | partial invalidity or ineffectiveness shall not invalidate the remainder | ||||||
|  | of the License, and in such case Affirmer hereby affirms that he or she | ||||||
|  | will not (i) exercise any of his or her remaining Copyright and Related | ||||||
|  | Rights in the Work or (ii) assert any associated claims and causes of | ||||||
|  | action with respect to the Work, in either case contrary to Affirmer's | ||||||
|  | express Statement of Purpose. | ||||||
|  | 
 | ||||||
|  | 4. Limitations and Disclaimers. | ||||||
|  | 
 | ||||||
|  |  a. No trademark or patent rights held by Affirmer are waived, abandoned, | ||||||
|  |     surrendered, licensed or otherwise affected by this document. | ||||||
|  |  b. Affirmer offers the Work as-is and makes no representations or | ||||||
|  |     warranties of any kind concerning the Work, express, implied, | ||||||
|  |     statutory or otherwise, including without limitation warranties of | ||||||
|  |     title, merchantability, fitness for a particular purpose, non | ||||||
|  |     infringement, or the absence of latent or other defects, accuracy, or | ||||||
|  |     the present or absence of errors, whether or not discoverable, all to | ||||||
|  |     the greatest extent permissible under applicable law. | ||||||
|  |  c. Affirmer disclaims responsibility for clearing rights of other persons | ||||||
|  |     that may apply to the Work or any use thereof, including without | ||||||
|  |     limitation any person's Copyright and Related Rights in the Work. | ||||||
|  |     Further, Affirmer disclaims responsibility for obtaining any necessary | ||||||
|  |     consents, permissions or other rights required for any use of the | ||||||
|  |     Work. | ||||||
|  |  d. Affirmer understands and acknowledges that Creative Commons is not a | ||||||
|  |     party to this document and has no duty or obligation with respect to | ||||||
|  |     this CC0 or use of the Work. | ||||||
							
								
								
									
										2
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | # Yet Another Discord Chat Link | ||||||
|  | 
 | ||||||
							
								
								
									
										95
									
								
								build.gradle
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								build.gradle
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,95 @@ | ||||||
|  | plugins { | ||||||
|  | 	id 'fabric-loom' version '1.4-SNAPSHOT' | ||||||
|  | 	id 'maven-publish' | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | version = project.mod_version | ||||||
|  | group = project.maven_group | ||||||
|  | 
 | ||||||
|  | base { | ||||||
|  | 	archivesName = project.archives_base_name | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | repositories { | ||||||
|  | 	// Add repositories to retrieve artifacts from in here. | ||||||
|  | 	// You should only use this when depending on other mods because | ||||||
|  | 	// Loom adds the essential maven repositories to download Minecraft and libraries from automatically. | ||||||
|  | 	// See https://docs.gradle.org/current/userguide/declaring_repositories.html | ||||||
|  | 	// for more information about repositories. | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | loom { | ||||||
|  |     splitEnvironmentSourceSets() | ||||||
|  | 
 | ||||||
|  | 	mods { | ||||||
|  | 		"modid" { | ||||||
|  | 			sourceSet sourceSets.main | ||||||
|  | 			sourceSet sourceSets.client | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | dependencies { | ||||||
|  |     implementation 'org.projectlombok:lombok:1.18.26' | ||||||
|  | 	annotationProcessor 'org.projectlombok:lombok:1.18.26' | ||||||
|  | 
 | ||||||
|  |     // To change the versions see the gradle.properties file | ||||||
|  | 	minecraft "com.mojang:minecraft:${project.minecraft_version}" | ||||||
|  | 	mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2" | ||||||
|  | 	modImplementation "net.fabricmc:fabric-loader:${project.loader_version}" | ||||||
|  | 
 | ||||||
|  | 	// Fabric API. This is technically optional, but you probably want it anyway. | ||||||
|  | 	modImplementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}" | ||||||
|  | 	 | ||||||
|  | 	// Uncomment the following line to enable the deprecated Fabric API modules.  | ||||||
|  | 	// These are included in the Fabric API production distribution and allow you to update your mod to the latest modules at a later more convenient time. | ||||||
|  | 
 | ||||||
|  | 	// modImplementation "net.fabricmc.fabric-api:fabric-api-deprecated:${project.fabric_version}" | ||||||
|  | 	implementation 'org.javacord:javacord:3.8.0' | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | processResources { | ||||||
|  | 	inputs.property "version", project.version | ||||||
|  | 
 | ||||||
|  | 	filesMatching("fabric.mod.json") { | ||||||
|  | 		expand "version": project.version | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | tasks.withType(JavaCompile).configureEach { | ||||||
|  | 	it.options.release = 17 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | java { | ||||||
|  | 	// Loom will automatically attach sourcesJar to a RemapSourcesJar task and to the "build" task | ||||||
|  | 	// if it is present. | ||||||
|  | 	// If you remove this line, sources will not be generated. | ||||||
|  | 	withSourcesJar() | ||||||
|  | 
 | ||||||
|  | 	sourceCompatibility = JavaVersion.VERSION_17 | ||||||
|  | 	targetCompatibility = JavaVersion.VERSION_17 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | jar { | ||||||
|  | 	from("LICENSE") { | ||||||
|  | 		rename { "${it}_${project.base.archivesName.get()}"} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // configure the maven publication | ||||||
|  | publishing { | ||||||
|  | 	publications { | ||||||
|  | 		mavenJava(MavenPublication) { | ||||||
|  | 			from components.java | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// See https://docs.gradle.org/current/userguide/publishing_maven.html for information on how to set up publishing. | ||||||
|  | 	repositories { | ||||||
|  | 		// Add repositories to publish to here. | ||||||
|  | 		// Notice: This block does NOT have the same function as the block in the top level. | ||||||
|  | 		// The repositories here will be used for publishing your artifact, not for | ||||||
|  | 		// retrieving dependencies. | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								gradle.properties
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								gradle.properties
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | ||||||
|  | # Done to increase the memory available to gradle. | ||||||
|  | org.gradle.jvmargs=-Xmx1G | ||||||
|  | org.gradle.parallel=true | ||||||
|  | 
 | ||||||
|  | # Fabric Properties | ||||||
|  | # check these on https://fabricmc.net/develop | ||||||
|  | minecraft_version=1.20.2 | ||||||
|  | yarn_mappings=1.20.2+build.1 | ||||||
|  | loader_version=0.14.22 | ||||||
|  | 
 | ||||||
|  | # Mod Properties | ||||||
|  | mod_version=0.0.0 | ||||||
|  | maven_group=dev.jonasjones | ||||||
|  | archives_base_name=yet-another-discord-chat-link | ||||||
|  | 
 | ||||||
|  | # Dependencies | ||||||
|  | fabric_version=0.89.1+1.20.2 | ||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										7
									
								
								gradle/wrapper/gradle-wrapper.properties
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								gradle/wrapper/gradle-wrapper.properties
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | ||||||
|  | distributionBase=GRADLE_USER_HOME | ||||||
|  | distributionPath=wrapper/dists | ||||||
|  | distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-bin.zip | ||||||
|  | networkTimeout=10000 | ||||||
|  | validateDistributionUrl=true | ||||||
|  | zipStoreBase=GRADLE_USER_HOME | ||||||
|  | zipStorePath=wrapper/dists | ||||||
							
								
								
									
										248
									
								
								gradlew
									
										
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										248
									
								
								gradlew
									
										
									
									
										vendored
									
									
										Executable file
									
								
							|  | @ -0,0 +1,248 @@ | ||||||
|  | #!/bin/sh | ||||||
|  | 
 | ||||||
|  | # | ||||||
|  | # Copyright © 2015-2021 the original authors. | ||||||
|  | # | ||||||
|  | # 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 | ||||||
|  | # | ||||||
|  | #      https://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. | ||||||
|  | # | ||||||
|  | 
 | ||||||
|  | ############################################################################## | ||||||
|  | # | ||||||
|  | #   Gradle start up script for POSIX generated by Gradle. | ||||||
|  | # | ||||||
|  | #   Important for running: | ||||||
|  | # | ||||||
|  | #   (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is | ||||||
|  | #       noncompliant, but you have some other compliant shell such as ksh or | ||||||
|  | #       bash, then to run this script, type that shell name before the whole | ||||||
|  | #       command line, like: | ||||||
|  | # | ||||||
|  | #           ksh Gradle | ||||||
|  | # | ||||||
|  | #       Busybox and similar reduced shells will NOT work, because this script | ||||||
|  | #       requires all of these POSIX shell features: | ||||||
|  | #         * functions; | ||||||
|  | #         * expansions «$var», «${var}», «${var:-default}», «${var+SET}», | ||||||
|  | #           «${var#prefix}», «${var%suffix}», and «$( cmd )»; | ||||||
|  | #         * compound commands having a testable exit status, especially «case»; | ||||||
|  | #         * various built-in commands including «command», «set», and «ulimit». | ||||||
|  | # | ||||||
|  | #   Important for patching: | ||||||
|  | # | ||||||
|  | #   (2) This script targets any POSIX shell, so it avoids extensions provided | ||||||
|  | #       by Bash, Ksh, etc; in particular arrays are avoided. | ||||||
|  | # | ||||||
|  | #       The "traditional" practice of packing multiple parameters into a | ||||||
|  | #       space-separated string is a well documented source of bugs and security | ||||||
|  | #       problems, so this is (mostly) avoided, by progressively accumulating | ||||||
|  | #       options in "$@", and eventually passing that to Java. | ||||||
|  | # | ||||||
|  | #       Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, | ||||||
|  | #       and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; | ||||||
|  | #       see the in-line comments for details. | ||||||
|  | # | ||||||
|  | #       There are tweaks for specific operating systems such as AIX, CygWin, | ||||||
|  | #       Darwin, MinGW, and NonStop. | ||||||
|  | # | ||||||
|  | #   (3) This script is generated from the Groovy template | ||||||
|  | #       https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt | ||||||
|  | #       within the Gradle project. | ||||||
|  | # | ||||||
|  | #       You can find Gradle at https://github.com/gradle/gradle/. | ||||||
|  | # | ||||||
|  | ############################################################################## | ||||||
|  | 
 | ||||||
|  | # Attempt to set APP_HOME | ||||||
|  | 
 | ||||||
|  | # Resolve links: $0 may be a link | ||||||
|  | app_path=$0 | ||||||
|  | 
 | ||||||
|  | # Need this for daisy-chained symlinks. | ||||||
|  | while | ||||||
|  |     APP_HOME=${app_path%"${app_path##*/}"}  # leaves a trailing /; empty if no leading path | ||||||
|  |     [ -h "$app_path" ] | ||||||
|  | do | ||||||
|  |     ls=$( ls -ld "$app_path" ) | ||||||
|  |     link=${ls#*' -> '} | ||||||
|  |     case $link in             #( | ||||||
|  |       /*)   app_path=$link ;; #( | ||||||
|  |       *)    app_path=$APP_HOME$link ;; | ||||||
|  |     esac | ||||||
|  | done | ||||||
|  | 
 | ||||||
|  | # This is normally unused | ||||||
|  | # shellcheck disable=SC2034 | ||||||
|  | APP_BASE_NAME=${0##*/} | ||||||
|  | APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit | ||||||
|  | 
 | ||||||
|  | # Use the maximum available, or set MAX_FD != -1 to use that value. | ||||||
|  | MAX_FD=maximum | ||||||
|  | 
 | ||||||
|  | warn () { | ||||||
|  |     echo "$*" | ||||||
|  | } >&2 | ||||||
|  | 
 | ||||||
|  | die () { | ||||||
|  |     echo | ||||||
|  |     echo "$*" | ||||||
|  |     echo | ||||||
|  |     exit 1 | ||||||
|  | } >&2 | ||||||
|  | 
 | ||||||
|  | # OS specific support (must be 'true' or 'false'). | ||||||
|  | cygwin=false | ||||||
|  | msys=false | ||||||
|  | darwin=false | ||||||
|  | nonstop=false | ||||||
|  | case "$( uname )" in                #( | ||||||
|  |   CYGWIN* )         cygwin=true  ;; #( | ||||||
|  |   Darwin* )         darwin=true  ;; #( | ||||||
|  |   MSYS* | MINGW* )  msys=true    ;; #( | ||||||
|  |   NONSTOP* )        nonstop=true ;; | ||||||
|  | esac | ||||||
|  | 
 | ||||||
|  | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Determine the Java command to use to start the JVM. | ||||||
|  | if [ -n "$JAVA_HOME" ] ; then | ||||||
|  |     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then | ||||||
|  |         # IBM's JDK on AIX uses strange locations for the executables | ||||||
|  |         JAVACMD=$JAVA_HOME/jre/sh/java | ||||||
|  |     else | ||||||
|  |         JAVACMD=$JAVA_HOME/bin/java | ||||||
|  |     fi | ||||||
|  |     if [ ! -x "$JAVACMD" ] ; then | ||||||
|  |         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME | ||||||
|  | 
 | ||||||
|  | Please set the JAVA_HOME variable in your environment to match the | ||||||
|  | location of your Java installation." | ||||||
|  |     fi | ||||||
|  | else | ||||||
|  |     JAVACMD=java | ||||||
|  |     if ! command -v java >/dev/null 2>&1 | ||||||
|  |     then | ||||||
|  |         die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||||
|  | 
 | ||||||
|  | Please set the JAVA_HOME variable in your environment to match the | ||||||
|  | location of your Java installation." | ||||||
|  |     fi | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # Increase the maximum file descriptors if we can. | ||||||
|  | if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then | ||||||
|  |     case $MAX_FD in #( | ||||||
|  |       max*) | ||||||
|  |         # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. | ||||||
|  |         # shellcheck disable=SC3045 | ||||||
|  |         MAX_FD=$( ulimit -H -n ) || | ||||||
|  |             warn "Could not query maximum file descriptor limit" | ||||||
|  |     esac | ||||||
|  |     case $MAX_FD in  #( | ||||||
|  |       '' | soft) :;; #( | ||||||
|  |       *) | ||||||
|  |         # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. | ||||||
|  |         # shellcheck disable=SC3045 | ||||||
|  |         ulimit -n "$MAX_FD" || | ||||||
|  |             warn "Could not set maximum file descriptor limit to $MAX_FD" | ||||||
|  |     esac | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # Collect all arguments for the java command, stacking in reverse order: | ||||||
|  | #   * args from the command line | ||||||
|  | #   * the main class name | ||||||
|  | #   * -classpath | ||||||
|  | #   * -D...appname settings | ||||||
|  | #   * --module-path (only if needed) | ||||||
|  | #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. | ||||||
|  | 
 | ||||||
|  | # For Cygwin or MSYS, switch paths to Windows format before running java | ||||||
|  | if "$cygwin" || "$msys" ; then | ||||||
|  |     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) | ||||||
|  |     CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) | ||||||
|  | 
 | ||||||
|  |     JAVACMD=$( cygpath --unix "$JAVACMD" ) | ||||||
|  | 
 | ||||||
|  |     # Now convert the arguments - kludge to limit ourselves to /bin/sh | ||||||
|  |     for arg do | ||||||
|  |         if | ||||||
|  |             case $arg in                                #( | ||||||
|  |               -*)   false ;;                            # don't mess with options #( | ||||||
|  |               /?*)  t=${arg#/} t=/${t%%/*}              # looks like a POSIX filepath | ||||||
|  |                     [ -e "$t" ] ;;                      #( | ||||||
|  |               *)    false ;; | ||||||
|  |             esac | ||||||
|  |         then | ||||||
|  |             arg=$( cygpath --path --ignore --mixed "$arg" ) | ||||||
|  |         fi | ||||||
|  |         # Roll the args list around exactly as many times as the number of | ||||||
|  |         # args, so each arg winds up back in the position where it started, but | ||||||
|  |         # possibly modified. | ||||||
|  |         # | ||||||
|  |         # NB: a `for` loop captures its iteration list before it begins, so | ||||||
|  |         # changing the positional parameters here affects neither the number of | ||||||
|  |         # iterations, nor the values presented in `arg`. | ||||||
|  |         shift                   # remove old arg | ||||||
|  |         set -- "$@" "$arg"      # push replacement arg | ||||||
|  |     done | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||||||
|  | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | ||||||
|  | 
 | ||||||
|  | # Collect all arguments for the java command; | ||||||
|  | #   * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of | ||||||
|  | #     shell script including quotes and variable substitutions, so put them in | ||||||
|  | #     double quotes to make sure that they get re-expanded; and | ||||||
|  | #   * put everything else in single quotes, so that it's not re-expanded. | ||||||
|  | 
 | ||||||
|  | set -- \ | ||||||
|  |         "-Dorg.gradle.appname=$APP_BASE_NAME" \ | ||||||
|  |         -classpath "$CLASSPATH" \ | ||||||
|  |         org.gradle.wrapper.GradleWrapperMain \ | ||||||
|  |         "$@" | ||||||
|  | 
 | ||||||
|  | # Stop when "xargs" is not available. | ||||||
|  | if ! command -v xargs >/dev/null 2>&1 | ||||||
|  | then | ||||||
|  |     die "xargs is not available" | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # Use "xargs" to parse quoted args. | ||||||
|  | # | ||||||
|  | # With -n1 it outputs one arg per line, with the quotes and backslashes removed. | ||||||
|  | # | ||||||
|  | # In Bash we could simply go: | ||||||
|  | # | ||||||
|  | #   readarray ARGS < <( xargs -n1 <<<"$var" ) && | ||||||
|  | #   set -- "${ARGS[@]}" "$@" | ||||||
|  | # | ||||||
|  | # but POSIX shell has neither arrays nor command substitution, so instead we | ||||||
|  | # post-process each arg (as a line of input to sed) to backslash-escape any | ||||||
|  | # character that might be a shell metacharacter, then use eval to reverse | ||||||
|  | # that process (while maintaining the separation between arguments), and wrap | ||||||
|  | # the whole thing up as a single "set" statement. | ||||||
|  | # | ||||||
|  | # This will of course break if any of these variables contains a newline or | ||||||
|  | # an unmatched quote. | ||||||
|  | # | ||||||
|  | 
 | ||||||
|  | eval "set -- $( | ||||||
|  |         printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | | ||||||
|  |         xargs -n1 | | ||||||
|  |         sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | | ||||||
|  |         tr '\n' ' ' | ||||||
|  |     )" '"$@"' | ||||||
|  | 
 | ||||||
|  | exec "$JAVACMD" "$@" | ||||||
							
								
								
									
										92
									
								
								gradlew.bat
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								gradlew.bat
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,92 @@ | ||||||
|  | @rem | ||||||
|  | @rem Copyright 2015 the original author or authors. | ||||||
|  | @rem | ||||||
|  | @rem Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | @rem you may not use this file except in compliance with the License. | ||||||
|  | @rem You may obtain a copy of the License at | ||||||
|  | @rem | ||||||
|  | @rem      https://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | @rem | ||||||
|  | @rem Unless required by applicable law or agreed to in writing, software | ||||||
|  | @rem distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | @rem See the License for the specific language governing permissions and | ||||||
|  | @rem limitations under the License. | ||||||
|  | @rem | ||||||
|  | 
 | ||||||
|  | @if "%DEBUG%"=="" @echo off | ||||||
|  | @rem ########################################################################## | ||||||
|  | @rem | ||||||
|  | @rem  Gradle startup script for Windows | ||||||
|  | @rem | ||||||
|  | @rem ########################################################################## | ||||||
|  | 
 | ||||||
|  | @rem Set local scope for the variables with windows NT shell | ||||||
|  | if "%OS%"=="Windows_NT" setlocal | ||||||
|  | 
 | ||||||
|  | set DIRNAME=%~dp0 | ||||||
|  | if "%DIRNAME%"=="" set DIRNAME=. | ||||||
|  | @rem This is normally unused | ||||||
|  | set APP_BASE_NAME=%~n0 | ||||||
|  | set APP_HOME=%DIRNAME% | ||||||
|  | 
 | ||||||
|  | @rem Resolve any "." and ".." in APP_HOME to make it shorter. | ||||||
|  | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi | ||||||
|  | 
 | ||||||
|  | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||||||
|  | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" | ||||||
|  | 
 | ||||||
|  | @rem Find java.exe | ||||||
|  | if defined JAVA_HOME goto findJavaFromJavaHome | ||||||
|  | 
 | ||||||
|  | set JAVA_EXE=java.exe | ||||||
|  | %JAVA_EXE% -version >NUL 2>&1 | ||||||
|  | if %ERRORLEVEL% equ 0 goto execute | ||||||
|  | 
 | ||||||
|  | echo. | ||||||
|  | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||||
|  | echo. | ||||||
|  | echo Please set the JAVA_HOME variable in your environment to match the | ||||||
|  | echo location of your Java installation. | ||||||
|  | 
 | ||||||
|  | goto fail | ||||||
|  | 
 | ||||||
|  | :findJavaFromJavaHome | ||||||
|  | set JAVA_HOME=%JAVA_HOME:"=% | ||||||
|  | set JAVA_EXE=%JAVA_HOME%/bin/java.exe | ||||||
|  | 
 | ||||||
|  | if exist "%JAVA_EXE%" goto execute | ||||||
|  | 
 | ||||||
|  | echo. | ||||||
|  | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% | ||||||
|  | echo. | ||||||
|  | echo Please set the JAVA_HOME variable in your environment to match the | ||||||
|  | echo location of your Java installation. | ||||||
|  | 
 | ||||||
|  | goto fail | ||||||
|  | 
 | ||||||
|  | :execute | ||||||
|  | @rem Setup the command line | ||||||
|  | 
 | ||||||
|  | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @rem Execute Gradle | ||||||
|  | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* | ||||||
|  | 
 | ||||||
|  | :end | ||||||
|  | @rem End local scope for the variables with windows NT shell | ||||||
|  | if %ERRORLEVEL% equ 0 goto mainEnd | ||||||
|  | 
 | ||||||
|  | :fail | ||||||
|  | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of | ||||||
|  | rem the _cmd.exe /c_ return code! | ||||||
|  | set EXIT_CODE=%ERRORLEVEL% | ||||||
|  | if %EXIT_CODE% equ 0 set EXIT_CODE=1 | ||||||
|  | if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% | ||||||
|  | exit /b %EXIT_CODE% | ||||||
|  | 
 | ||||||
|  | :mainEnd | ||||||
|  | if "%OS%"=="Windows_NT" endlocal | ||||||
|  | 
 | ||||||
|  | :omega | ||||||
							
								
								
									
										10
									
								
								settings.gradle
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								settings.gradle
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | pluginManagement { | ||||||
|  | 	repositories { | ||||||
|  | 		maven { | ||||||
|  | 			name = 'Fabric' | ||||||
|  | 			url = 'https://maven.fabricmc.net/' | ||||||
|  | 		} | ||||||
|  | 		mavenCentral() | ||||||
|  | 		gradlePluginPortal() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -0,0 +1,34 @@ | ||||||
|  | package dev.jonasjones.yadcl; | ||||||
|  | 
 | ||||||
|  | import dev.jonasjones.yadcl.config.ModConfigs; | ||||||
|  | import dev.jonasjones.yadcl.dcbot.DiscordBot; | ||||||
|  | import net.fabricmc.api.ModInitializer; | ||||||
|  | 
 | ||||||
|  | import org.slf4j.Logger; | ||||||
|  | import org.slf4j.LoggerFactory; | ||||||
|  | import static dev.jonasjones.yadcl.dcbot.DiscordBot.sendToDiscord; | ||||||
|  | 
 | ||||||
|  | public class YetAnotherDiscordChatLink implements ModInitializer { | ||||||
|  |     // This logger is used to write text to the console and the log file. | ||||||
|  |     // It is considered best practice to use your mod id as the logger's name. | ||||||
|  |     // That way, it's clear which mod wrote info, warnings, and errors. | ||||||
|  |     public static final String MOD_ID = "yadcl"; | ||||||
|  |     public static final Logger LOGGER = LoggerFactory.getLogger(MOD_ID); | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void onInitialize() { | ||||||
|  |         // Register the config | ||||||
|  |         ModConfigs.registerConfigs(); | ||||||
|  | 
 | ||||||
|  |         // Set the token and channel id | ||||||
|  |         DiscordBot.setToken(ModConfigs.TOKEN); | ||||||
|  |         DiscordBot.setTargetChannelId(ModConfigs.CHANNEL_ID); | ||||||
|  | 
 | ||||||
|  |         // Start the bot | ||||||
|  |         DiscordBot.startBot(); | ||||||
|  |         // send starting message | ||||||
|  |         sendToDiscord("Server is starting up."); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,32 @@ | ||||||
|  | package dev.jonasjones.yadcl.config; | ||||||
|  | 
 | ||||||
|  | import com.mojang.datafixers.util.Pair; | ||||||
|  | 
 | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.List; | ||||||
|  | 
 | ||||||
|  | public class ModConfigProvider implements SimpleConfig.DefaultConfig { | ||||||
|  | 
 | ||||||
|  |     private String configContents = ""; | ||||||
|  | 
 | ||||||
|  |     public List<Pair> getConfigsList() { | ||||||
|  |         return configsList; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private final List<Pair> configsList = new ArrayList<>(); | ||||||
|  | 
 | ||||||
|  |     public void addKeyValuePair(Pair<String, ?> keyValuePair, String comment) { | ||||||
|  |         configsList.add(keyValuePair); | ||||||
|  |         configContents += keyValuePair.getFirst() + "=" + keyValuePair.getSecond() + " #" | ||||||
|  |                 + comment + " | default: " + keyValuePair.getSecond() + "\n"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void addSingleLineComment(String comment) { | ||||||
|  |         configContents += "# " + comment + "\n"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String get(String namespace) { | ||||||
|  |         return configContents; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								src/main/java/dev/jonasjones/yadcl/config/ModConfigs.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								src/main/java/dev/jonasjones/yadcl/config/ModConfigs.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,35 @@ | ||||||
|  | package dev.jonasjones.yadcl.config; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import com.mojang.datafixers.util.Pair; | ||||||
|  | import dev.jonasjones.yadcl.YetAnotherDiscordChatLink; | ||||||
|  | 
 | ||||||
|  | public class ModConfigs { | ||||||
|  |     public static SimpleConfig CONFIG; | ||||||
|  |     private static ModConfigProvider configs; | ||||||
|  | 
 | ||||||
|  |     public static String TOKEN; | ||||||
|  |     public static String CHANNEL_ID; | ||||||
|  | 
 | ||||||
|  |     public static void registerConfigs() { | ||||||
|  |         configs = new ModConfigProvider(); | ||||||
|  |         createConfigs(); | ||||||
|  | 
 | ||||||
|  |         CONFIG = SimpleConfig.of(YetAnotherDiscordChatLink.MOD_ID + "config").provider(configs).request(); | ||||||
|  | 
 | ||||||
|  |         assignConfigs(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void createConfigs() { | ||||||
|  |         configs.addKeyValuePair(new Pair<>("bot.token", "[Insert bot token here]"), "The Bot token from the discord developer dashboard"); | ||||||
|  |         configs.addKeyValuePair(new Pair<>("bot.channel", "1165690308185563216"), "The channel id of the channel the bot should listen and send to"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void assignConfigs() { | ||||||
|  |         TOKEN = CONFIG.getOrDefault("bot.token", "default value"); | ||||||
|  |         CHANNEL_ID = CONFIG.getOrDefault("bot.channel", "1165690308185563216"); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         YetAnotherDiscordChatLink.LOGGER.info("All " + configs.getConfigsList().size() + " Configs have been set properly"); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										253
									
								
								src/main/java/dev/jonasjones/yadcl/config/SimpleConfig.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										253
									
								
								src/main/java/dev/jonasjones/yadcl/config/SimpleConfig.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,253 @@ | ||||||
|  | package dev.jonasjones.yadcl.config; | ||||||
|  | /* | ||||||
|  |  * Copyright (c) 2021 magistermaks | ||||||
|  |  * Modified by Jonas_Jones 2022 | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | import net.fabricmc.loader.api.FabricLoader; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
|  | 
 | ||||||
|  | import java.io.File; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.io.PrintWriter; | ||||||
|  | import java.nio.file.Files; | ||||||
|  | import java.nio.file.Path; | ||||||
|  | import java.util.HashMap; | ||||||
|  | import java.util.Scanner; | ||||||
|  | 
 | ||||||
|  | public class SimpleConfig { | ||||||
|  | 
 | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("BetterSimpleConfig"); | ||||||
|  |     private final HashMap<String, String> config = new HashMap<>(); | ||||||
|  |     private final ConfigRequest request; | ||||||
|  |     private boolean broken = false; | ||||||
|  | 
 | ||||||
|  |     public interface DefaultConfig { | ||||||
|  |         String get( String namespace ); | ||||||
|  | 
 | ||||||
|  |         static String empty( String namespace ) { | ||||||
|  |             return ""; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static class ConfigRequest { | ||||||
|  | 
 | ||||||
|  |         private final File file; | ||||||
|  |         private final String filename; | ||||||
|  |         private DefaultConfig provider; | ||||||
|  | 
 | ||||||
|  |         private ConfigRequest(File file, String filename ) { | ||||||
|  |             this.file = file; | ||||||
|  |             this.filename = filename; | ||||||
|  |             this.provider = DefaultConfig::empty; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /** | ||||||
|  |          * Sets the default config provider, used to generate the | ||||||
|  |          * config if it's missing. | ||||||
|  |          * | ||||||
|  |          * @param provider default config provider | ||||||
|  |          * @return current config request object | ||||||
|  |          * @see DefaultConfig | ||||||
|  |          */ | ||||||
|  |         public ConfigRequest provider( DefaultConfig provider ) { | ||||||
|  |             this.provider = provider; | ||||||
|  |             return this; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /** | ||||||
|  |          * Loads the config from the filesystem. | ||||||
|  |          * | ||||||
|  |          * @return config object | ||||||
|  |          * @see SimpleConfig | ||||||
|  |          */ | ||||||
|  |         public SimpleConfig request() { | ||||||
|  |             return new SimpleConfig( this ); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private String getConfig() { | ||||||
|  |             return provider.get( filename ) + "\n"; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates new config request object, ideally `namespace` | ||||||
|  |      * should be the name of the mod id of the requesting mod | ||||||
|  |      * | ||||||
|  |      * @param filename - name of the config file | ||||||
|  |      * @return new config request object | ||||||
|  |      */ | ||||||
|  |     public static ConfigRequest of( String filename ) { | ||||||
|  |         Path path = FabricLoader.getInstance().getConfigDir(); | ||||||
|  |         return new ConfigRequest( path.resolve( filename + ".properties" ).toFile(), filename ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void createConfig() throws IOException { | ||||||
|  | 
 | ||||||
|  |         // try creating missing files | ||||||
|  |         request.file.getParentFile().mkdirs(); | ||||||
|  |         Files.createFile( request.file.toPath() ); | ||||||
|  | 
 | ||||||
|  |         // write default config data | ||||||
|  |         PrintWriter writer = new PrintWriter(request.file, "UTF-8"); | ||||||
|  |         writer.write( request.getConfig() ); | ||||||
|  |         writer.close(); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void loadConfig() throws IOException { | ||||||
|  |         Scanner reader = new Scanner( request.file ); | ||||||
|  |         for( int line = 1; reader.hasNextLine(); line ++ ) { | ||||||
|  |             parseConfigEntry( reader.nextLine(), line ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void parseConfigEntry( String entry, int line ) { | ||||||
|  |         if( !entry.isEmpty() && !entry.startsWith( "#" ) ) { | ||||||
|  |             String[] parts = entry.split("=", 2); | ||||||
|  |             if( parts.length == 2 ) { | ||||||
|  |                 //Recognise comments after a value | ||||||
|  |                 String temp = parts[1].split(" #")[0]; | ||||||
|  |                 config.put( parts[0], temp); | ||||||
|  |             }else{ | ||||||
|  |                 throw new RuntimeException("Syntax error in config file on line " + line + "!"); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private SimpleConfig( ConfigRequest request ) { | ||||||
|  |         this.request = request; | ||||||
|  |         String identifier = "Config '" + request.filename + "'"; | ||||||
|  | 
 | ||||||
|  |         if( !request.file.exists() ) { | ||||||
|  |             LOGGER.info( identifier + " is missing, generating default one..." ); | ||||||
|  | 
 | ||||||
|  |             try { | ||||||
|  |                 createConfig(); | ||||||
|  |             } catch (IOException e) { | ||||||
|  |                 LOGGER.error( identifier + " failed to generate!" ); | ||||||
|  |                 LOGGER.trace( e ); | ||||||
|  |                 broken = true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if( !broken ) { | ||||||
|  |             try { | ||||||
|  |                 loadConfig(); | ||||||
|  |             } catch (Exception e) { | ||||||
|  |                 LOGGER.error( identifier + " failed to load!" ); | ||||||
|  |                 LOGGER.trace( e ); | ||||||
|  |                 broken = true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Queries a value from config, returns `null` if the | ||||||
|  |      * key does not exist. | ||||||
|  |      * | ||||||
|  |      * @return  value corresponding to the given key | ||||||
|  |      * @see     SimpleConfig#getOrDefault | ||||||
|  |      */ | ||||||
|  |     @Deprecated | ||||||
|  |     public String get( String key ) { | ||||||
|  |         return config.get( key ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns string value from config corresponding to the given | ||||||
|  |      * key, or the default string if the key is missing. | ||||||
|  |      * | ||||||
|  |      * @return  value corresponding to the given key, or the default value | ||||||
|  |      */ | ||||||
|  |     public String getOrDefault( String key, String def ) { | ||||||
|  |         String val = get(key); | ||||||
|  |         return val == null ? def : val; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns integer value from config corresponding to the given | ||||||
|  |      * key, or the default integer if the key is missing or invalid. | ||||||
|  |      * | ||||||
|  |      * @return  value corresponding to the given key, or the default value | ||||||
|  |      */ | ||||||
|  |     public int getOrDefault( String key, int def ) { | ||||||
|  |         try { | ||||||
|  |             return Integer.parseInt( get(key) ); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             return def; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns boolean value from config corresponding to the given | ||||||
|  |      * key, or the default boolean if the key is missing. | ||||||
|  |      * | ||||||
|  |      * @return  value corresponding to the given key, or the default value | ||||||
|  |      */ | ||||||
|  |     public boolean getOrDefault( String key, boolean def ) { | ||||||
|  |         String val = get(key); | ||||||
|  |         if( val != null ) { | ||||||
|  |             return val.equalsIgnoreCase("true"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return def; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns double value from config corresponding to the given | ||||||
|  |      * key, or the default string if the key is missing or invalid. | ||||||
|  |      * | ||||||
|  |      * @return  value corresponding to the given key, or the default value | ||||||
|  |      */ | ||||||
|  |     public double getOrDefault( String key, double def ) { | ||||||
|  |         try { | ||||||
|  |             return Double.parseDouble( get(key) ); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             return def; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * If any error occurred during loading or reading from the config | ||||||
|  |      * a 'broken' flag is set, indicating that the config's state | ||||||
|  |      * is undefined and should be discarded using `delete()` | ||||||
|  |      * | ||||||
|  |      * @return the 'broken' flag of the configuration | ||||||
|  |      */ | ||||||
|  |     public boolean isBroken() { | ||||||
|  |         return broken; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * deletes the config file from the filesystem | ||||||
|  |      * | ||||||
|  |      * @return true if the operation was successful | ||||||
|  |      */ | ||||||
|  |     public boolean delete() { | ||||||
|  |         LOGGER.warn( "Config '" + request.filename + "' was removed from existence! Restart the game to regenerate it." ); | ||||||
|  |         return request.file.delete(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										79
									
								
								src/main/java/dev/jonasjones/yadcl/dcbot/DiscordBot.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								src/main/java/dev/jonasjones/yadcl/dcbot/DiscordBot.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,79 @@ | ||||||
|  | package dev.jonasjones.yadcl.dcbot; | ||||||
|  | 
 | ||||||
|  | import lombok.Setter; | ||||||
|  | import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents; | ||||||
|  | import net.minecraft.server.MinecraftServer; | ||||||
|  | import net.minecraft.text.Text; | ||||||
|  | import org.javacord.api.DiscordApi; | ||||||
|  | import org.javacord.api.DiscordApiBuilder; | ||||||
|  | import org.javacord.api.entity.channel.TextChannel; | ||||||
|  | 
 | ||||||
|  | import static dev.jonasjones.yadcl.YetAnotherDiscordChatLink.LOGGER; | ||||||
|  | 
 | ||||||
|  | public class DiscordBot { | ||||||
|  |     @Setter | ||||||
|  |     private static String token; | ||||||
|  |     @Setter | ||||||
|  |     private static String targetChannelId; | ||||||
|  |     private static DiscordApi api; | ||||||
|  |     private static MinecraftServer minecraftServer; | ||||||
|  |     private static Boolean isBotRunning = false; | ||||||
|  | 
 | ||||||
|  |     public static void startBot() { | ||||||
|  |         if (isBotRunning) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             registerServerTickEvent(); | ||||||
|  |             api = new DiscordApiBuilder().setToken(token).setAllIntents().login().join(); | ||||||
|  |             api.addMessageCreateListener(event -> { | ||||||
|  |                 // Check if the message is from the specific channel by comparing channel IDs | ||||||
|  |                 if (String.valueOf(event.getChannel().getId()).equals(targetChannelId)) { | ||||||
|  |                     //check if message author is the bot | ||||||
|  |                     if (event.getMessageAuthor().isBotUser()) { | ||||||
|  |                         return; | ||||||
|  |                     } | ||||||
|  |                     String discordMessage = "[" + event.getMessageAuthor().getDisplayName() + "] " + event.getMessageContent(); | ||||||
|  |                     // Broadcast the message to Minecraft chat | ||||||
|  |                     // You can implement a method to broadcast messages to Minecraft players | ||||||
|  |                     minecraftServer.getPlayerManager().broadcast(Text.of(discordMessage), false); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             LOGGER.error("Failed to start Discord bot. Check the provided discord token in the config file."); | ||||||
|  |         } | ||||||
|  |         isBotRunning = true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void stopBot() { | ||||||
|  |         if (!isBotRunning) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         api.disconnect(); | ||||||
|  |         isBotRunning = false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void sendToDiscord(String message) { | ||||||
|  |         if (!isBotRunning) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         // Get the text channel by its ID | ||||||
|  |         TextChannel channel = api.getTextChannelById(targetChannelId).orElse(null); | ||||||
|  | 
 | ||||||
|  |         // Check if the channel exists and send the message | ||||||
|  |         if (channel != null) { | ||||||
|  |             channel.sendMessage(message); | ||||||
|  |         } else { | ||||||
|  |             // Handle the case where the channel does not exist | ||||||
|  |             LOGGER.error("Discord Channel does not exist!"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void registerServerTickEvent() { | ||||||
|  |         ServerTickEvents.START_SERVER_TICK.register(server -> { | ||||||
|  |             // This code is executed on every server tick | ||||||
|  |             minecraftServer = server; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,24 @@ | ||||||
|  | package dev.jonasjones.yadcl.mixin; | ||||||
|  | 
 | ||||||
|  | import net.minecraft.network.packet.c2s.play.ChatMessageC2SPacket; | ||||||
|  | import net.minecraft.server.MinecraftServer; | ||||||
|  | import net.minecraft.server.network.ServerPlayNetworkHandler; | ||||||
|  | import net.minecraft.server.network.ServerPlayerEntity; | ||||||
|  | import org.spongepowered.asm.mixin.Mixin; | ||||||
|  | import org.spongepowered.asm.mixin.Shadow; | ||||||
|  | import org.spongepowered.asm.mixin.injection.At; | ||||||
|  | import org.spongepowered.asm.mixin.injection.Inject; | ||||||
|  | import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; | ||||||
|  | 
 | ||||||
|  | import static dev.jonasjones.yadcl.dcbot.DiscordBot.sendToDiscord; | ||||||
|  | 
 | ||||||
|  | @Mixin(ServerPlayNetworkHandler.class) | ||||||
|  | public abstract class MixinServerPlayNetworkHandler { | ||||||
|  | 	@Shadow public ServerPlayerEntity player; | ||||||
|  | 
 | ||||||
|  | 	@Inject(method = "onChatMessage", at = @At("HEAD"), cancellable = true) | ||||||
|  | 	private void onChatMessage(ChatMessageC2SPacket packet, CallbackInfo ci) { | ||||||
|  | 		String messageContent = packet.chatMessage(); // Get the content of the chat message | ||||||
|  | 		sendToDiscord("<" + this.player.getDisplayName().getString() + "> " + messageContent); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -0,0 +1,25 @@ | ||||||
|  | package dev.jonasjones.yadcl.mixin; | ||||||
|  | 
 | ||||||
|  | import net.minecraft.network.ClientConnection; | ||||||
|  | import net.minecraft.server.PlayerManager; | ||||||
|  | import net.minecraft.server.network.ConnectedClientData; | ||||||
|  | import net.minecraft.server.network.ServerPlayerEntity; | ||||||
|  | import org.spongepowered.asm.mixin.Mixin; | ||||||
|  | import org.spongepowered.asm.mixin.injection.At; | ||||||
|  | import org.spongepowered.asm.mixin.injection.Inject; | ||||||
|  | import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; | ||||||
|  | 
 | ||||||
|  | import static dev.jonasjones.yadcl.dcbot.DiscordBot.sendToDiscord; | ||||||
|  | 
 | ||||||
|  | @Mixin(PlayerManager.class) | ||||||
|  | public class PlayerManagerMixin { | ||||||
|  |     @Inject(at = @At("HEAD"), method = "onPlayerConnect") | ||||||
|  |     public void onPlayerConnect(ClientConnection connection, ServerPlayerEntity player, ConnectedClientData clientData, CallbackInfo ci) { | ||||||
|  |         sendToDiscord(player.getDisplayName().getString() + " joined the game."); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Inject(at = @At("HEAD"), method = "remove") | ||||||
|  |     public void remove(ServerPlayerEntity player, CallbackInfo ci) { | ||||||
|  |         sendToDiscord(player.getDisplayName().getString() + " left the game."); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,25 @@ | ||||||
|  | package dev.jonasjones.yadcl.mixin; | ||||||
|  | 
 | ||||||
|  | import net.minecraft.server.MinecraftServer; | ||||||
|  | import org.spongepowered.asm.mixin.Mixin; | ||||||
|  | import org.spongepowered.asm.mixin.injection.At; | ||||||
|  | import org.spongepowered.asm.mixin.injection.Inject; | ||||||
|  | import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; | ||||||
|  | 
 | ||||||
|  | import static dev.jonasjones.yadcl.dcbot.DiscordBot.sendToDiscord; | ||||||
|  | import static dev.jonasjones.yadcl.dcbot.DiscordBot.stopBot; | ||||||
|  | 
 | ||||||
|  | @Mixin(MinecraftServer.class) | ||||||
|  | public class ServerStopMixin { | ||||||
|  |     @Inject(at = @At("HEAD"), method = "shutdown") | ||||||
|  |     private void init(CallbackInfo info) { | ||||||
|  |         sendToDiscord("Server is shutting down."); | ||||||
|  |         //wait 2 seconds for the message to send | ||||||
|  |         try { | ||||||
|  |             Thread.sleep(500); | ||||||
|  |         } catch (InterruptedException e) { | ||||||
|  |             e.printStackTrace(); | ||||||
|  |         } | ||||||
|  |         stopBot(); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								src/main/resources/assets/yadcl/icon.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/main/resources/assets/yadcl/icon.png
									
										
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 453 B | 
							
								
								
									
										38
									
								
								src/main/resources/fabric.mod.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								src/main/resources/fabric.mod.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | ||||||
|  | { | ||||||
|  | 	"schemaVersion": 1, | ||||||
|  | 	"id": "yadcl", | ||||||
|  | 	"version": "${version}", | ||||||
|  | 	"name": "Yet Another Discord Chat Link", | ||||||
|  | 	"description": "A mod that allows you to link your Minecraft chat to a Discord channel.", | ||||||
|  | 	"authors": [ | ||||||
|  | 		"Me!" | ||||||
|  | 	], | ||||||
|  | 	"contact": { | ||||||
|  | 		"homepage": "https://jonasjones.dev/", | ||||||
|  | 		"sources": "https://github.com/J-onasJones/YetAnotherDiscordChatLink" | ||||||
|  | 	}, | ||||||
|  | 	"license": "CC0-1.0", | ||||||
|  | 	"icon": "assets/yadcl/icon.png", | ||||||
|  | 	"environment": "*", | ||||||
|  | 	"entrypoints": { | ||||||
|  | 		"main": [ | ||||||
|  | 			"dev.jonasjones.yadcl.YetAnotherDiscordChatLink" | ||||||
|  | 		] | ||||||
|  | 	}, | ||||||
|  | 	"mixins": [ | ||||||
|  |       "yadcl.mixins.json", | ||||||
|  | 		{ | ||||||
|  | 			"config": "modid.client.mixins.json", | ||||||
|  | 			"environment": "client" | ||||||
|  | 		} | ||||||
|  | 	], | ||||||
|  | 	"depends": { | ||||||
|  | 		"fabricloader": ">=0.14.22", | ||||||
|  | 		"minecraft": "~1.20.2", | ||||||
|  | 		"java": ">=17", | ||||||
|  | 		"fabric-api": "*" | ||||||
|  | 	}, | ||||||
|  | 	"suggests": { | ||||||
|  | 		"another-mod": "*" | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								src/main/resources/yadcl.mixins.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/main/resources/yadcl.mixins.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,13 @@ | ||||||
|  | { | ||||||
|  | 	"required": true, | ||||||
|  | 	"package": "dev.jonasjones.yadcl.mixin", | ||||||
|  | 	"compatibilityLevel": "JAVA_17", | ||||||
|  | 	"mixins": [ | ||||||
|  | 		"MixinServerPlayNetworkHandler", | ||||||
|  | 		"PlayerManagerMixin", | ||||||
|  | 		"ServerStopMixin" | ||||||
|  | 	], | ||||||
|  | 	"injectors": { | ||||||
|  | 		"defaultRequire": 1 | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue