Squashed 'solution/H02/' content from commit 5a755f9
git-subtree-dir: solution/H02 git-subtree-split: 5a755f9abf28a4f7f6670bd3587d1ae2cb3b531d
This commit is contained in:
		
						commit
						58e3b905b1
					
				
					 34 changed files with 3943 additions and 0 deletions
				
			
		
							
								
								
									
										12
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,12 @@ | ||||||
|  | # Editor configuration, see https://editorconfig.org | ||||||
|  | 
 | ||||||
|  | [*] | ||||||
|  | charset = utf-8 | ||||||
|  | end_of_line = lf | ||||||
|  | indent_style = space | ||||||
|  | indent_size = 4 | ||||||
|  | insert_final_newline = true | ||||||
|  | trim_trailing_whitespace = true | ||||||
|  | 
 | ||||||
|  | [{*.yml,*.json}] | ||||||
|  | indent_size = 2 | ||||||
							
								
								
									
										87
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,87 @@ | ||||||
|  | ### Intellij ### | ||||||
|  | .idea/ | ||||||
|  | *.iws | ||||||
|  | /out/ | ||||||
|  | *.iml | ||||||
|  | .idea_modules/ | ||||||
|  | atlassian-ide-plugin.xml | ||||||
|  | 
 | ||||||
|  | ### VS-Code ### | ||||||
|  | .vscode/ | ||||||
|  | .VSCodeCounter/ | ||||||
|  | 
 | ||||||
|  | ### Eclipse ### | ||||||
|  | .metadata | ||||||
|  | bin/ | ||||||
|  | tmp/ | ||||||
|  | *.tmp | ||||||
|  | *.bak | ||||||
|  | *.swp | ||||||
|  | *~.nib | ||||||
|  | local.properties | ||||||
|  | .settings/ | ||||||
|  | .loadpath | ||||||
|  | .recommenders | ||||||
|  | .externalToolBuilders/ | ||||||
|  | *.launch | ||||||
|  | .factorypath | ||||||
|  | .recommenders/ | ||||||
|  | .apt_generated/ | ||||||
|  | .project | ||||||
|  | .classpath | ||||||
|  | 
 | ||||||
|  | ### Linux ### | ||||||
|  | *~ | ||||||
|  | .fuse_hidden* | ||||||
|  | .directory | ||||||
|  | .Trash-* | ||||||
|  | .nfs* | ||||||
|  | 
 | ||||||
|  | ### macOS ### | ||||||
|  | .DS_Store | ||||||
|  | .AppleDouble | ||||||
|  | .LSOverride | ||||||
|  | Icon | ||||||
|  | ._* | ||||||
|  | .DocumentRevisions-V100 | ||||||
|  | .fseventsd | ||||||
|  | .Spotlight-V100 | ||||||
|  | .TemporaryItems | ||||||
|  | .Trashes | ||||||
|  | .VolumeIcon.icns | ||||||
|  | .com.apple.timemachine.donotpresent | ||||||
|  | .AppleDB | ||||||
|  | .AppleDesktop | ||||||
|  | Network Trash Folder | ||||||
|  | Temporary Items | ||||||
|  | .apdisk | ||||||
|  | 
 | ||||||
|  | ### NetBeans ### | ||||||
|  | nbproject/private/ | ||||||
|  | build/ | ||||||
|  | nbbuild/ | ||||||
|  | dist/ | ||||||
|  | nbdist/ | ||||||
|  | .nb-gradle/ | ||||||
|  | 
 | ||||||
|  | ### Windows ### | ||||||
|  | # Windows thumbnail cache files | ||||||
|  | Thumbs.db | ||||||
|  | ehthumbs.db | ||||||
|  | ehthumbs_vista.db | ||||||
|  | *.stackdump | ||||||
|  | [Dd]esktop.ini | ||||||
|  | $RECYCLE.BIN/ | ||||||
|  | *.lnk | ||||||
|  | 
 | ||||||
|  | ### Gradle ### | ||||||
|  | .gradle | ||||||
|  | /build/ | ||||||
|  | out/ | ||||||
|  | gradle-app.setting | ||||||
|  | !gradle-wrapper.jar | ||||||
|  | .gradletasknamecache | ||||||
|  | 
 | ||||||
|  | *.hprof | ||||||
|  | screenshots/ | ||||||
|  | jagr.conf | ||||||
							
								
								
									
										4
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | ||||||
|  | # Musterlösung zu Hausübung 02 | ||||||
|  | 
 | ||||||
|  | Beachten Sie die Hinweise zum Herunterladen, Importieren, Bearbeitern, Exportieren und Hochladen in unserem | ||||||
|  | [Studierenden-Guide](https://wiki.tudalgo.org/) | ||||||
							
								
								
									
										41
									
								
								build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,41 @@ | ||||||
|  | plugins { | ||||||
|  |     alias(libs.plugins.algomate) | ||||||
|  |     alias(libs.plugins.style) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | version = file("version").readLines().first() | ||||||
|  | 
 | ||||||
|  | exercise { | ||||||
|  |     assignmentId.set("h02") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | submission { | ||||||
|  |     // ACHTUNG! | ||||||
|  |     // Setzen Sie im folgenden Bereich Ihre TU-ID (NICHT Ihre Matrikelnummer!), Ihren Nachnamen und Ihren Vornamen | ||||||
|  |     // in Anführungszeichen (z.B. "ab12cdef" für Ihre TU-ID) ein! | ||||||
|  |     // BEISPIEL: | ||||||
|  |     // studentId = "ab12cdef" | ||||||
|  |     // firstName = "sol_first" | ||||||
|  |     // lastName = "sol_last" | ||||||
|  |     studentId = "ab12cdef" | ||||||
|  |     firstName = "sol_first" | ||||||
|  |     lastName = "sol_last" | ||||||
|  | 
 | ||||||
|  |     // Optionally require own tests for mainBuildSubmission task. Default is false | ||||||
|  |     requireTests = false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | dependencies { | ||||||
|  |     implementation(libs.fopbot) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | jagr { | ||||||
|  |     graders { | ||||||
|  |         val graderPublic by getting | ||||||
|  |         val graderPrivate by creating { | ||||||
|  |             parent(graderPublic) | ||||||
|  |             graderName.set("FOP-2425-H02-Private") | ||||||
|  |             rubricProviderName.set("h02.H02_RubricProvider") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								gradle/libs.versions.toml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								gradle/libs.versions.toml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | ||||||
|  | [plugins] | ||||||
|  | algomate = { id = "org.tudalgo.algomate", version = "0.7.1" } | ||||||
|  | style = { id = "org.sourcegrade.style", version = "3.0.0" } | ||||||
|  | 
 | ||||||
|  | [libraries] | ||||||
|  | fopbot = { module = "org.tudalgo:fopbot", version = "0.8.1" } | ||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										6
									
								
								gradle/wrapper/gradle-wrapper.properties
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								gradle/wrapper/gradle-wrapper.properties
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | ||||||
|  | distributionBase=GRADLE_USER_HOME | ||||||
|  | distributionPath=wrapper/dists | ||||||
|  | distributionUrl=https\://services.gradle.org/distributions/gradle-8.10.2-all.zip | ||||||
|  | networkTimeout=10000 | ||||||
|  | zipStoreBase=GRADLE_USER_HOME | ||||||
|  | zipStorePath=wrapper/dists | ||||||
							
								
								
									
										245
									
								
								gradlew
									
										
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										245
									
								
								gradlew
									
										
									
									
										vendored
									
									
										Executable file
									
								
							|  | @ -0,0 +1,245 @@ | ||||||
|  | #!/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 | ||||||
|  |     which java >/dev/null 2>&1 || 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 | ||||||
|  | 
 | ||||||
|  | # 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 | ||||||
							
								
								
									
										11
									
								
								settings.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								settings.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | ||||||
|  | dependencyResolutionManagement { | ||||||
|  |     repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) | ||||||
|  |     repositories { | ||||||
|  | //        mavenLocal() | ||||||
|  |         maven("https://s01.oss.sonatype.org/content/repositories/snapshots") | ||||||
|  |         maven("https://jitpack.io") | ||||||
|  |         mavenCentral() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | rootProject.name = "H02-Root" | ||||||
							
								
								
									
										748
									
								
								src/graderPrivate/java/h02/FourWinsTest.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										748
									
								
								src/graderPrivate/java/h02/FourWinsTest.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,748 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import fopbot.Direction; | ||||||
|  | import fopbot.Robot; | ||||||
|  | import fopbot.RobotFamily; | ||||||
|  | import fopbot.RobotTrace; | ||||||
|  | import fopbot.Transition; | ||||||
|  | import fopbot.World; | ||||||
|  | import h02.template.InputHandler; | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  | import org.junit.jupiter.api.Timeout; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.junit.jupiter.params.provider.ValueSource; | ||||||
|  | import org.mockito.MockedStatic; | ||||||
|  | import org.mockito.Mockito; | ||||||
|  | import org.mockito.stubbing.Answer; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.TestForSubmission; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.annotation.SkipAfterFirstFailedTest; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.Assertions2; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.Context; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.PreCommentSupplier; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.ResultOfObject; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSet; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSetTest; | ||||||
|  | import spoon.reflect.code.CtInvocation; | ||||||
|  | import spoon.reflect.code.CtLoop; | ||||||
|  | import spoon.reflect.declaration.CtElement; | ||||||
|  | import spoon.reflect.declaration.CtMethod; | ||||||
|  | import spoon.reflect.reference.CtExecutableReference; | ||||||
|  | 
 | ||||||
|  | import java.io.ByteArrayOutputStream; | ||||||
|  | import java.io.PrintStream; | ||||||
|  | import java.lang.reflect.Method; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.Iterator; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  | import java.util.function.Consumer; | ||||||
|  | 
 | ||||||
|  | import static h02.TestUtils.getCtMethod; | ||||||
|  | import static h02.TestUtils.iterateMethodStatements; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertCallEquals; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertCallFalse; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertEquals; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertSame; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertTrue; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.call; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.callObject; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.contextBuilder; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.emptyContext; | ||||||
|  | 
 | ||||||
|  | @TestForSubmission | ||||||
|  | @Timeout( | ||||||
|  |     value = TestConstants.TEST_TIMEOUT_IN_SECONDS, | ||||||
|  |     unit = TimeUnit.SECONDS, | ||||||
|  |     threadMode = Timeout.ThreadMode.SEPARATE_THREAD | ||||||
|  | ) | ||||||
|  | @SkipAfterFirstFailedTest(TestConstants.SKIP_AFTER_FIRST_FAILED_TEST) | ||||||
|  | public class FourWinsTest { | ||||||
|  | 
 | ||||||
|  |     private static final Consumer<Iterator<CtElement>> SINGLE_LOOP_VA = iterator -> { | ||||||
|  |         int loopStatements = 0; | ||||||
|  | 
 | ||||||
|  |         while (iterator.hasNext()) { | ||||||
|  |             if (iterator.next() instanceof CtLoop) { | ||||||
|  |                 loopStatements++; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         assertEquals(1, loopStatements, emptyContext(), result -> "Method does not use exactly one loop"); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     private int worldHeight; | ||||||
|  |     private int worldWidth; | ||||||
|  |     private RobotFamily[][] stones; | ||||||
|  |     private RobotFamily currentPlayer; | ||||||
|  |     private Context.Builder<?> baseContextBuilder; | ||||||
|  | 
 | ||||||
|  |     public void setup(JsonParameterSet params) { | ||||||
|  |         worldHeight = params.getInt("worldHeight"); | ||||||
|  |         worldWidth = params.getInt("worldWidth"); | ||||||
|  |         List<List<String>> paramStones = params.get("gameBoard"); | ||||||
|  |         stones = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |         for (int row = 0; row < worldHeight; row++) { | ||||||
|  |             for (int col = 0; col < worldWidth; col++) { | ||||||
|  |                 stones[row][col] = robotFamilyLookup(paramStones.get(row).get(col)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         baseContextBuilder = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth) | ||||||
|  |             .add("stones", stones); | ||||||
|  |         if (params.availableKeys().contains("currentPlayer")) { | ||||||
|  |             currentPlayer = robotFamilyLookup(params.get("currentPlayer")); | ||||||
|  |             baseContextBuilder.add("currentPlayer", currentPlayer); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         World.setDelay(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "FourWinsTestValidateInput.json") | ||||||
|  |     public void testValidateInputEdgeCases(final JsonParameterSet params) { | ||||||
|  |         testValidateInput(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "FourWinsTestValidateInputRandomCases.generated.json") | ||||||
|  |     public void testValidateInputRandomCases(final JsonParameterSet params) { | ||||||
|  |         testValidateInput(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void testValidateInput(final JsonParameterSet params) { | ||||||
|  |         // get params | ||||||
|  |         final int paramColumn = params.getInt("column"); | ||||||
|  |         final int paramWidth = params.getInt("width"); | ||||||
|  |         final int paramHeight = params.getInt("height"); | ||||||
|  |         final List<List<String>> paramStones = params.get("stones"); | ||||||
|  |         final boolean expectedResult = params.getBoolean("expected result"); | ||||||
|  | 
 | ||||||
|  |         // write params to context | ||||||
|  |         final var ParamsContext = params.toContext("expected result"); | ||||||
|  |         final var cb = contextBuilder() | ||||||
|  |             .add(ParamsContext) | ||||||
|  |             .add("Method", "validateInput"); | ||||||
|  | 
 | ||||||
|  |         // init the world size | ||||||
|  |         World.setSize(paramWidth, paramHeight); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // parse array and calculate result | ||||||
|  |         RobotFamily[][] paramStonesArray = paramStones.stream() | ||||||
|  |             .map(innerList -> innerList.stream() | ||||||
|  |                 .map(str -> "EMPTY".equals(str) ? null : "SQUARE_RED".equals(str) ? RobotFamily.SQUARE_RED : | ||||||
|  |                     RobotFamily.SQUARE_BLUE) | ||||||
|  |                 .toArray(RobotFamily[]::new)) | ||||||
|  |             .toArray(RobotFamily[][]::new); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         final boolean actualResult = Assertions2.callObject( | ||||||
|  |             () -> FourWins.validateInput( | ||||||
|  |                 paramColumn, | ||||||
|  |                 paramStonesArray | ||||||
|  |             ), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "An error occurred during execution." | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         // validate result | ||||||
|  |         Assertions2.assertEquals( | ||||||
|  |             expectedResult, | ||||||
|  |             actualResult, | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "Invalid result." | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoard.generated.json") | ||||||
|  |     public void testGetDestinationRowFreeSlot(JsonParameterSet params) { | ||||||
|  |         testGetDestinationRow(params, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoard.generated.json") | ||||||
|  |     public void testGetDestinationRowBlockedSlot(JsonParameterSet params) { | ||||||
|  |         testGetDestinationRow(params, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testGetDestinationRowVAnforderung() { | ||||||
|  |         iterateMethodStatements(FourWins.class, | ||||||
|  |             "getDestinationRow", | ||||||
|  |             new Class[] {int.class, RobotFamily[][].class}, | ||||||
|  |             SINGLE_LOOP_VA); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoard.generated.json") | ||||||
|  |     public void testDropStoneRobotCorrect(JsonParameterSet params) { | ||||||
|  |         setup(params); | ||||||
|  |         List<Integer> firstFreeIndex = params.get("firstFreeIndex"); | ||||||
|  | 
 | ||||||
|  |         for (int col = 0; col < worldWidth; col++) { | ||||||
|  |             if (firstFreeIndex.get(col) >= worldHeight) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             World.getGlobalWorld().reset();  // clear entities | ||||||
|  |             RobotFamily currentPlayer = col % 2 == 0 ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |             Context context = baseContextBuilder | ||||||
|  |                 .add("column", col) | ||||||
|  |                 .add("currentPlayer", currentPlayer) | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             try { | ||||||
|  |                 final int finalCol = col; | ||||||
|  |                 call(() -> FourWins.dropStone(finalCol, stones, currentPlayer), context, result -> | ||||||
|  |                     "An exception occurred while invoking method dropStone. Result may be salvageable, continuing..."); | ||||||
|  |             } catch (Throwable t) { | ||||||
|  |                 t.printStackTrace(System.err); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             List<Robot> robots = World.getGlobalWorld() | ||||||
|  |                 .getAllFieldEntities() | ||||||
|  |                 .stream() | ||||||
|  |                 .filter(fieldEntity -> fieldEntity instanceof Robot) | ||||||
|  |                 .map(fieldEntity -> (Robot) fieldEntity) | ||||||
|  |                 .toList(); | ||||||
|  |             assertEquals(1, robots.size(), context, result -> | ||||||
|  |                 "Unexpected number of robots in world"); | ||||||
|  | 
 | ||||||
|  |             RobotTrace trace = World.getGlobalWorld().getTrace(robots.get(0)); | ||||||
|  |             Robot robot = trace.getTransitions().get(0).robot; | ||||||
|  |             assertEquals(col, robot.getX(), context, result -> | ||||||
|  |                 "Robot was initialized with incorrect x coordinate"); | ||||||
|  |             assertEquals(worldHeight - 1, robot.getY(), context, result -> | ||||||
|  |                 "Robot was initialized with incorrect y coordinate"); | ||||||
|  |             assertEquals(Direction.DOWN, robot.getDirection(), context, result -> | ||||||
|  |                 "Robot was initialized with incorrect direction"); | ||||||
|  |             assertEquals(0, robot.getNumberOfCoins(), context, result -> | ||||||
|  |                 "Robot was initialized with incorrect number of coins"); | ||||||
|  |             assertEquals(currentPlayer, robot.getRobotFamily(), context, result -> | ||||||
|  |                 "Robot was initialized with incorrect robot family"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testDropStoneCallsGetDestinationRow() { | ||||||
|  |         CtMethod<?> dropStoneCtMethod = getCtMethod(FourWins.class, "dropStone", int.class, RobotFamily[][].class, RobotFamily.class); | ||||||
|  |         CtExecutableReference<?> getDestinationRowCtExecRef = getCtMethod(FourWins.class, "getDestinationRow", int.class, RobotFamily[][].class) | ||||||
|  |             .getReference(); | ||||||
|  |         Iterator<CtElement> iterator = dropStoneCtMethod.descendantIterator(); | ||||||
|  | 
 | ||||||
|  |         boolean getDestinationRowCalled = false; | ||||||
|  |         while (!getDestinationRowCalled && iterator.hasNext()) { | ||||||
|  |             CtElement ctElement = iterator.next(); | ||||||
|  |             if (ctElement instanceof CtInvocation<?> ctInvocation) { | ||||||
|  |                 getDestinationRowCalled = ctInvocation.getExecutable().equals(getDestinationRowCtExecRef); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         assertTrue(getDestinationRowCalled, emptyContext(), result -> | ||||||
|  |             "Method dropStone does not call method getDestinationRow"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoard.generated.json") | ||||||
|  |     public void testDropStoneMovementCorrect(JsonParameterSet params) { | ||||||
|  |         setup(params); | ||||||
|  |         List<Integer> firstFreeIndex = params.get("firstFreeIndex"); | ||||||
|  | 
 | ||||||
|  |         for (int col = 0; col < worldWidth; col++) { | ||||||
|  |             if (firstFreeIndex.get(col) >= worldHeight) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             World.getGlobalWorld().reset();  // clear entities | ||||||
|  |             RobotFamily currentPlayer = RobotFamily.SQUARE_RED; | ||||||
|  |             Context context = baseContextBuilder | ||||||
|  |                 .add("column", col) | ||||||
|  |                 .add("currentPlayer", currentPlayer) | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             try { | ||||||
|  |                 final int finalCol = col; | ||||||
|  |                 call(() -> FourWins.dropStone(finalCol, stones, currentPlayer), context, result -> | ||||||
|  |                     "An exception occurred while invoking method dropStone. Result may be salvageable, continuing..."); | ||||||
|  |             } catch (Throwable t) { | ||||||
|  |                 t.printStackTrace(System.err); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             List<Robot> robots = World.getGlobalWorld() | ||||||
|  |                 .getAllFieldEntities() | ||||||
|  |                 .stream() | ||||||
|  |                 .filter(Robot.class::isInstance) | ||||||
|  |                 .map(Robot.class::cast) | ||||||
|  |                 .toList(); | ||||||
|  |             assertEquals(1, robots.size(), context, result -> | ||||||
|  |                 "Unexpected number of robots in world"); | ||||||
|  | 
 | ||||||
|  |             List<Transition> transitions = World.getGlobalWorld().getTrace(robots.get(0)).getTransitions(); | ||||||
|  |             int expectedTransitionsSize = worldHeight - 1 - firstFreeIndex.get(col) + 3; | ||||||
|  |             for (int i = 0; i < expectedTransitionsSize; i++) { | ||||||
|  |                 Transition transition = transitions.get(i); | ||||||
|  |                 final int finalI = i; | ||||||
|  |                 PreCommentSupplier<ResultOfObject<Transition.RobotAction>> preCommentSupplier = result -> | ||||||
|  |                     "Robot did not perform the expected action (action number %d)".formatted(finalI); | ||||||
|  |                 if (i < expectedTransitionsSize - 3) {  // moving | ||||||
|  |                     assertEquals(Transition.RobotAction.MOVE, transition.action, context, preCommentSupplier); | ||||||
|  |                 } else if (i < expectedTransitionsSize - 1) {  // left turns | ||||||
|  |                     assertEquals(Transition.RobotAction.TURN_LEFT, transition.action, context, preCommentSupplier); | ||||||
|  |                 } else {  // last action (none) | ||||||
|  |                     assertEquals(Transition.RobotAction.NONE, transition.action, context, preCommentSupplier); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testDropStoneVAnforderung() { | ||||||
|  |         iterateMethodStatements(FourWins.class, | ||||||
|  |             "dropStone", | ||||||
|  |             new Class[] {int.class, RobotFamily[][].class, RobotFamily.class}, | ||||||
|  |             SINGLE_LOOP_VA); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoardHorizontalWin.generated.json") | ||||||
|  |     public void testTestWinHorizontal(JsonParameterSet params) { | ||||||
|  |         setup(params); | ||||||
|  |         List<Map<String, Integer>> winningRowCoordinates = params.get("winningRowCoordinates"); | ||||||
|  |         Context context = baseContextBuilder.build(); | ||||||
|  |         boolean expected = !winningRowCoordinates.isEmpty(); | ||||||
|  |         boolean actual = callObject(() -> FourWins.testWinHorizontal(stones, currentPlayer), context, result -> | ||||||
|  |             "An exception occurred while invoking method testWinHorizontal"); | ||||||
|  |         assertEquals(expected, actual, context, result -> | ||||||
|  |             "Method testWinHorizontal did not return the correct value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testTestWinHorizontalVAnforderung1() { | ||||||
|  |         testWinVAnforderung("testWinHorizontal"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testTestWinHorizontalVAnforderung2() { | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         RobotFamily[][] stones = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |         for (int row = 0; row < 4; row++) { | ||||||
|  |             stones[row][0] = RobotFamily.SQUARE_RED; | ||||||
|  |         } | ||||||
|  |         RobotFamily currentPlayer = RobotFamily.SQUARE_RED; | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth) | ||||||
|  |             .add("stones", stones) | ||||||
|  |             .add("currentPlayer", currentPlayer) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         assertCallFalse(() -> FourWins.testWinHorizontal(stones, currentPlayer), context, result -> | ||||||
|  |             "Method testWinHorizontal returned an incorrect value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("FourWinsTestGameBoardVerticalWin.generated.json") | ||||||
|  |     public void testTestWinVertical(JsonParameterSet params) { | ||||||
|  |         setup(params); | ||||||
|  |         List<Map<String, Integer>> winningColCoordinates = params.get("winningColCoordinates"); | ||||||
|  |         Context context = baseContextBuilder.build(); | ||||||
|  | 
 | ||||||
|  |         boolean expected = !winningColCoordinates.isEmpty(); | ||||||
|  |         boolean actual = callObject(() -> FourWins.testWinVertical(stones, currentPlayer), context, result -> | ||||||
|  |             "An exception occurred while invoking method testWinVertical"); | ||||||
|  |         assertEquals(expected, actual, context, result -> | ||||||
|  |             "Method testWinVertical did not return the correct value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testTestWinVerticalVAnforderung1() { | ||||||
|  |         testWinVAnforderung("testWinVertical"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testTestWinVerticalVAnforderung2() { | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         RobotFamily[][] stones = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |         for (int col = 0; col < 4; col++) { | ||||||
|  |             stones[0][col] = RobotFamily.SQUARE_RED; | ||||||
|  |         } | ||||||
|  |         RobotFamily currentPlayer = RobotFamily.SQUARE_RED; | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth) | ||||||
|  |             .add("stones", stones) | ||||||
|  |             .add("currentPlayer", currentPlayer) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         assertCallFalse(() -> FourWins.testWinVertical(stones, currentPlayer), context, result -> | ||||||
|  |             "Method testWinVertical returned an incorrect value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Tests {@link FourWins#testWinConditions(RobotFamily[][], RobotFamily)}. | ||||||
|  |      * The parameter {@code flags} is used to determine the return value of the win condition methods, | ||||||
|  |      * where a set bit is interpreted as {@code true} and an unset bit as {@code false}. | ||||||
|  |      * Only the first three bits are evaluated and correspond to the methods as follows: | ||||||
|  |      * <ul> | ||||||
|  |      *     <li>Bit 0: {@link FourWins#testWinHorizontal(RobotFamily[][], RobotFamily)}</li> | ||||||
|  |      *     <li>Bit 1: {@link FourWins#testWinVertical(RobotFamily[][], RobotFamily)}</li> | ||||||
|  |      *     <li>Bit 2: {@link FourWins#testWinDiagonal(RobotFamily[][], RobotFamily)}</li> | ||||||
|  |      * </ul> | ||||||
|  |      * | ||||||
|  |      * @param flags the flags (2:0, 31:3 are unused) | ||||||
|  |      * @throws ReflectiveOperationException if methods testWinHorizontal, testWinVertical or testWinDiagonal are not found | ||||||
|  |      */ | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(ints = {0, 1, 2, 3, 4, 5, 6, 7}) | ||||||
|  |     public void testTestWinConditions(int flags) throws ReflectiveOperationException { | ||||||
|  |         Method testWinHorizontalMethod = FourWins.class.getDeclaredMethod("testWinHorizontal", RobotFamily[][].class, RobotFamily.class); | ||||||
|  |         Method testWinVerticalMethod = FourWins.class.getDeclaredMethod("testWinVertical", RobotFamily[][].class, RobotFamily.class); | ||||||
|  |         Method testWinDiagonalMethod = FourWins.class.getDeclaredMethod("testWinDiagonal", RobotFamily[][].class, RobotFamily.class); | ||||||
|  |         Answer<?> answer = invocation -> { | ||||||
|  |             if (invocation.getMethod().equals(testWinHorizontalMethod)) { | ||||||
|  |                 return (flags & 1) == 1; | ||||||
|  |             } else if (invocation.getMethod().equals(testWinVerticalMethod)) { | ||||||
|  |                 return (flags >> 1 & 1) == 1; | ||||||
|  |             } else if (invocation.getMethod().equals(testWinDiagonalMethod)) { | ||||||
|  |                 return (flags >> 2 & 1) == 1; | ||||||
|  |             } else { | ||||||
|  |                 return invocation.callRealMethod(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         RobotFamily[][] stones = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |         RobotFamily currentPlayer = RobotFamily.SQUARE_RED; | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth) | ||||||
|  |             .add("stones (ignored)", stones) | ||||||
|  |             .add("currentPlayer (ignored)", currentPlayer.getName()) | ||||||
|  |             .add("testWinHorizontal (mocked) return value", (flags & 1) == 1) | ||||||
|  |             .add("testWinVertical (mocked) return value", (flags >> 1 & 1) == 1) | ||||||
|  |             .add("testWinDiagonal (mocked) return value", (flags >> 2 & 1) == 1) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         try (MockedStatic<FourWins> mock = Mockito.mockStatic(FourWins.class, answer)) { | ||||||
|  |             assertCallEquals(flags != 0, () -> FourWins.testWinConditions(stones, currentPlayer), context, result -> | ||||||
|  |                 "Method testWinConditions did not return the correct value"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testTestWinConditionsVAnforderung() { | ||||||
|  |         iterateMethodStatements(FourWins.class, "testWinConditions", new Class[] {RobotFamily[][].class, RobotFamily.class}, iterator -> { | ||||||
|  |             boolean callsTestWinHorizontal = false; | ||||||
|  |             boolean callsTestWinVertical = false; | ||||||
|  |             boolean callsTestWinDiagonal = false; | ||||||
|  | 
 | ||||||
|  |             while (iterator.hasNext()) { | ||||||
|  |                 if (iterator.next() instanceof CtInvocation<?> ctInvocation) { | ||||||
|  |                     if (ctInvocation.getExecutable().getSimpleName().equals("testWinHorizontal")) { | ||||||
|  |                         callsTestWinHorizontal = true; | ||||||
|  |                     } else if (ctInvocation.getExecutable().getSimpleName().equals("testWinVertical")) { | ||||||
|  |                         callsTestWinVertical = true; | ||||||
|  |                     } else if (ctInvocation.getExecutable().getSimpleName().equals("testWinDiagonal")) { | ||||||
|  |                         callsTestWinDiagonal = true; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             assertTrue(callsTestWinHorizontal, emptyContext(), result -> "Method testWinConditions did not call testWinHorizontal"); | ||||||
|  |             assertTrue(callsTestWinVertical, emptyContext(), result -> "Method testWinConditions did not call testWinVertical"); | ||||||
|  |             assertTrue(callsTestWinDiagonal, emptyContext(), result -> "Method testWinConditions did not call testWinDiagonal"); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(booleans = {true, false}) | ||||||
|  |     public void testNextPlayer(boolean isRedPlayer) { | ||||||
|  |         RobotFamily currentPlayer = isRedPlayer ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("currentPlayer", currentPlayer) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         RobotFamily expected = isRedPlayer ? RobotFamily.SQUARE_BLUE : RobotFamily.SQUARE_RED; | ||||||
|  |         assertCallEquals(expected, () -> FourWins.nextPlayer(currentPlayer), context, result -> | ||||||
|  |             "Method nextPlayer did not return the correct value"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(booleans = {true, false}) | ||||||
|  |     public void testColorFieldBackground(boolean isRedPlayer) { | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         RobotFamily winner = isRedPlayer ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth) | ||||||
|  |             .add("winner", winner) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         World.setDelay(0); | ||||||
|  |         call(() -> FourWins.colorFieldBackground(winner), context, result -> | ||||||
|  |             "An exception occurred while invoking colorFieldBackground"); | ||||||
|  |         for (int row = 0; row < worldHeight; row++) { | ||||||
|  |             for (int col = 0; col < worldWidth; col++) { | ||||||
|  |                 int finalRow = row; | ||||||
|  |                 int finalCol = col; | ||||||
|  |                 assertEquals(winner.getColor(), World.getGlobalWorld().getFieldColor(col, row), context, result -> | ||||||
|  |                     "Color of field at row %d, column %d is incorrect".formatted(finalRow, finalCol)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testWriteMessages() { | ||||||
|  |         PrintStream originalOut = System.out; | ||||||
|  |         ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); | ||||||
|  |         PrintStream printStream = new PrintStream(outputStream, true); | ||||||
|  |         System.setOut(printStream); | ||||||
|  | 
 | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         FourWins fourWinsInstance = new FourWins(worldWidth, worldHeight); | ||||||
|  |         Context.Builder<?> contextBuilder = contextBuilder() | ||||||
|  |             .add("world height", worldHeight) | ||||||
|  |             .add("world width", worldWidth); | ||||||
|  | 
 | ||||||
|  |         World.setSize(worldWidth, worldHeight); | ||||||
|  |         try { | ||||||
|  |             contextBuilder.add("method", "writeDrawMessage"); | ||||||
|  |             fourWinsInstance.writeDrawMessage(); | ||||||
|  |             assertEquals( | ||||||
|  |                 "No valid columns found. Hence, game ends with a draw.", | ||||||
|  |                 outputStream.toString().strip(), | ||||||
|  |                 contextBuilder.build(), | ||||||
|  |                 result -> "Method did not print the correct string" | ||||||
|  |             ); | ||||||
|  | 
 | ||||||
|  |             contextBuilder.add("method", "writeWinnerMessage"); | ||||||
|  |             for (RobotFamily winner : new RobotFamily[] {RobotFamily.SQUARE_RED, RobotFamily.SQUARE_BLUE}) { | ||||||
|  |                 contextBuilder.add("winner", winner); | ||||||
|  |                 outputStream.reset(); | ||||||
|  |                 fourWinsInstance.writeWinnerMessage(winner); | ||||||
|  |                 assertEquals( | ||||||
|  |                     "Player %s wins the game!".formatted(winner), | ||||||
|  |                     outputStream.toString().strip(), | ||||||
|  |                     contextBuilder.build(), | ||||||
|  |                     result -> "Method did not print the correct string" | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |         } finally { | ||||||
|  |             System.setOut(originalOut); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testGameLoopCallsNextPlayer() { | ||||||
|  |         List<RobotFamily> nextPlayerArgs = new ArrayList<>(); | ||||||
|  |         Answer<?> answer = invocation -> { | ||||||
|  |             Method method = invocation.getMethod(); | ||||||
|  |             if (method.getName().equals("nextPlayer") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily.class})) { | ||||||
|  |                 RobotFamily currentPlayer = invocation.getArgument(0); | ||||||
|  |                 nextPlayerArgs.add(currentPlayer); | ||||||
|  |                 return currentPlayer == RobotFamily.SQUARE_RED ? RobotFamily.SQUARE_BLUE : RobotFamily.SQUARE_RED; | ||||||
|  |             } else if (method.getName().equals("dropStone") && Arrays.equals(method.getParameterTypes(), new Class[] {int.class, RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 return null; | ||||||
|  |             } else if (method.getName().equals("testWinConditions") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 return nextPlayerArgs.size() >= 5; | ||||||
|  |             } else { | ||||||
|  |                 return invocation.callRealMethod(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         try (MockedStatic<FourWins> mockedStatic = Mockito.mockStatic(FourWins.class, answer)) { | ||||||
|  |             int worldHeight = 5; | ||||||
|  |             int worldWidth = 5; | ||||||
|  |             FourWins fourWins = new FourWins(worldWidth, worldHeight); | ||||||
|  |             InputHandler inputHandler = fourWins.getInputHandler(); | ||||||
|  |             for (int i = 0; i < 10; i++) { | ||||||
|  |                 inputHandler.addInput(0); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             fourWins.startGame(); | ||||||
|  |             for (int i = 0; i < nextPlayerArgs.size(); i++) { | ||||||
|  |                 Context context = contextBuilder() | ||||||
|  |                     .add("world height", worldHeight) | ||||||
|  |                     .add("world width", worldWidth) | ||||||
|  |                     .add("game loop iteration", i + 1) | ||||||
|  |                     .build(); | ||||||
|  | 
 | ||||||
|  |                 if (i % 2 == 0) { | ||||||
|  |                     assertEquals(RobotFamily.SQUARE_BLUE, nextPlayerArgs.get(i), context, result -> | ||||||
|  |                         "Method nextPlayer was not called with correct parameters"); | ||||||
|  |                 } else { | ||||||
|  |                     assertEquals(RobotFamily.SQUARE_RED, nextPlayerArgs.get(i), context, result -> | ||||||
|  |                         "Method nextPlayer was not called with correct parameters"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testGameLoopCallsDropStone() { | ||||||
|  |         List<Integer> columnArgs = new ArrayList<>(); | ||||||
|  |         List<RobotFamily[][]> stonesArgs = new ArrayList<>(); | ||||||
|  |         List<RobotFamily> currentPlayerArgs = new ArrayList<>(); | ||||||
|  |         Answer<?> answer = invocation -> { | ||||||
|  |             Method method = invocation.getMethod(); | ||||||
|  |             if (method.getName().equals("nextPlayer") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily.class})) { | ||||||
|  |                 RobotFamily currentPlayer = invocation.getArgument(0); | ||||||
|  |                 return currentPlayer == RobotFamily.SQUARE_RED ? RobotFamily.SQUARE_BLUE : RobotFamily.SQUARE_RED; | ||||||
|  |             } else if (method.getName().equals("dropStone") && Arrays.equals(method.getParameterTypes(), new Class[] {int.class, RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 columnArgs.add(invocation.getArgument(0)); | ||||||
|  |                 stonesArgs.add(invocation.getArgument(1)); | ||||||
|  |                 currentPlayerArgs.add(invocation.getArgument(2)); | ||||||
|  |                 return null; | ||||||
|  |             } else if (method.getName().equals("testWinConditions") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 return currentPlayerArgs.size() >= 5; | ||||||
|  |             } else { | ||||||
|  |                 return invocation.callRealMethod(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         try (MockedStatic<FourWins> mockedStatic = Mockito.mockStatic(FourWins.class, answer)) { | ||||||
|  |             FourWins fourWins = new FourWins(worldWidth, worldHeight); | ||||||
|  |             InputHandler inputHandler = fourWins.getInputHandler(); | ||||||
|  |             for (int i = 0; i < 10; i++) { | ||||||
|  |                 inputHandler.addInput(i % worldWidth); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             fourWins.startGame(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         RobotFamily[][] stones = null; | ||||||
|  |         for (int i = 0; i < currentPlayerArgs.size(); i++) { | ||||||
|  |             if (stones == null) { | ||||||
|  |                 stones = stonesArgs.get(i); | ||||||
|  |             } | ||||||
|  |             Context context = contextBuilder() | ||||||
|  |                 .add("world height", worldHeight) | ||||||
|  |                 .add("world width", worldWidth) | ||||||
|  |                 .add("game loop iteration", i + 1) | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             assertEquals(i % worldWidth, columnArgs.get(i), context, result -> | ||||||
|  |                 "Method dropStone was not called with correct parameter column"); | ||||||
|  |             assertSame(stones, stonesArgs.get(i), context, result -> | ||||||
|  |                 "Method dropStone was not called with correct parameter stones"); | ||||||
|  |             if (i % 2 == 0) { | ||||||
|  |                 assertEquals(RobotFamily.SQUARE_RED, currentPlayerArgs.get(i), context, result -> | ||||||
|  |                     "Method dropStone was not called with correct parameter currentPlayer"); | ||||||
|  |             } else { | ||||||
|  |                 assertEquals(RobotFamily.SQUARE_BLUE, currentPlayerArgs.get(i), context, result -> | ||||||
|  |                     "Method dropStone was not called with correct parameter currentPlayer"); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testGameLoopCallsGetWinConditions() { | ||||||
|  |         List<RobotFamily[][]> stonesArgs = new ArrayList<>(); | ||||||
|  |         List<RobotFamily> currentPlayerArgs = new ArrayList<>(); | ||||||
|  |         Answer<?> answer = invocation -> { | ||||||
|  |             Method method = invocation.getMethod(); | ||||||
|  |             if (method.getName().equals("nextPlayer") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily.class})) { | ||||||
|  |                 RobotFamily currentPlayer = invocation.getArgument(0); | ||||||
|  |                 return currentPlayer == RobotFamily.SQUARE_RED ? RobotFamily.SQUARE_BLUE : RobotFamily.SQUARE_RED; | ||||||
|  |             } else if (method.getName().equals("dropStone") && Arrays.equals(method.getParameterTypes(), new Class[] {int.class, RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 return null; | ||||||
|  |             } else if (method.getName().equals("testWinConditions") && Arrays.equals(method.getParameterTypes(), new Class[] {RobotFamily[][].class, RobotFamily.class})) { | ||||||
|  |                 stonesArgs.add(invocation.getArgument(0)); | ||||||
|  |                 currentPlayerArgs.add(invocation.getArgument(1)); | ||||||
|  |                 return currentPlayerArgs.size() >= 5; | ||||||
|  |             } else { | ||||||
|  |                 return invocation.callRealMethod(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         int worldHeight = 5; | ||||||
|  |         int worldWidth = 5; | ||||||
|  |         try (MockedStatic<FourWins> mockedStatic = Mockito.mockStatic(FourWins.class, answer)) { | ||||||
|  |             FourWins fourWins = new FourWins(worldWidth, worldHeight); | ||||||
|  |             InputHandler inputHandler = fourWins.getInputHandler(); | ||||||
|  |             for (int i = 0; i < 10; i++) { | ||||||
|  |                 inputHandler.addInput(i % worldWidth); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             fourWins.startGame(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         RobotFamily[][] stones = null; | ||||||
|  |         for (int i = 0; i < currentPlayerArgs.size(); i++) { | ||||||
|  |             if (stones == null) { | ||||||
|  |                 stones = stonesArgs.get(i); | ||||||
|  |             } | ||||||
|  |             Context context = contextBuilder() | ||||||
|  |                 .add("world height", worldHeight) | ||||||
|  |                 .add("world width", worldWidth) | ||||||
|  |                 .add("game loop iteration", i + 1) | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             assertSame(stones, stonesArgs.get(i), context, result -> | ||||||
|  |                 "Method getWinConditions was not called with correct parameter stones"); | ||||||
|  |             if (i % 2 == 0) { | ||||||
|  |                 assertEquals(RobotFamily.SQUARE_RED, currentPlayerArgs.get(i), context, result -> | ||||||
|  |                     "Method getWinConditions was not called with correct parameter currentPlayer"); | ||||||
|  |             } else { | ||||||
|  |                 assertEquals(RobotFamily.SQUARE_BLUE, currentPlayerArgs.get(i), context, result -> | ||||||
|  |                     "Method getWinConditions was not called with correct parameter currentPlayer"); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void testGetDestinationRow(JsonParameterSet params, boolean testFreeSlots) { | ||||||
|  |         setup(params); | ||||||
|  |         List<Integer> firstFreeIndex = params.get("firstFreeIndex"); | ||||||
|  | 
 | ||||||
|  |         for (int i = 0; i < firstFreeIndex.size(); i++) { | ||||||
|  |             int index = firstFreeIndex.get(i); | ||||||
|  |             if ((testFreeSlots && index >= worldHeight) || (!testFreeSlots && index < worldHeight)) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             final int column = i; | ||||||
|  |             int expected = testFreeSlots ? index : -1; | ||||||
|  |             Context context = baseContextBuilder | ||||||
|  |                 .add("column", column) | ||||||
|  |                 .build(); | ||||||
|  |             int actual = callObject(() -> FourWins.getDestinationRow(column, stones), context, result -> | ||||||
|  |                 "An exception occurred while invoking method getDestinationRow"); | ||||||
|  |             assertEquals(expected, actual, context, result -> | ||||||
|  |                 "Method getDestinationRow returned an incorrect value"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void testWinVAnforderung(String methodName) { | ||||||
|  |         List<CtLoop> loops = getCtMethod(FourWins.class, methodName, RobotFamily[][].class, RobotFamily.class) | ||||||
|  |             .filterChildren(CtLoop.class::isInstance) | ||||||
|  |             .list(); | ||||||
|  | 
 | ||||||
|  |         assertEquals(2, loops.size(), emptyContext(), result -> | ||||||
|  |             "Method %s does not use exactly two loops".formatted(methodName)); | ||||||
|  |         assertTrue(loops.get(0).getBody().equals(loops.get(1).getParent()), emptyContext(), result -> | ||||||
|  |             "Method %s does not use exactly two nested loops".formatted(methodName)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static RobotFamily robotFamilyLookup(String robotFamilyName) { | ||||||
|  |         if (robotFamilyName == null) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return switch (robotFamilyName) { | ||||||
|  |             case "SQUARE_RED" -> RobotFamily.SQUARE_RED; | ||||||
|  |             case "SQUARE_BLUE" -> RobotFamily.SQUARE_BLUE; | ||||||
|  |             default -> null; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										367
									
								
								src/graderPrivate/java/h02/H02_RubricProvider.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										367
									
								
								src/graderPrivate/java/h02/H02_RubricProvider.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,367 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.sourcegrade.jagr.api.rubric.Criterion; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.JUnitTestRef; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.Rubric; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.RubricProvider; | ||||||
|  | import org.sourcegrade.jagr.api.testing.RubricConfiguration; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSet; | ||||||
|  | 
 | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.jagr.RubricUtils.criterion; | ||||||
|  | 
 | ||||||
|  | public class H02_RubricProvider implements RubricProvider { | ||||||
|  | 
 | ||||||
|  |     public static final Rubric RUBRIC = Rubric.builder() | ||||||
|  |         .title("H02 | Vier Gewinnt") | ||||||
|  |         .addChildCriteria( | ||||||
|  |             Criterion.builder() | ||||||
|  |                 .shortDescription("H2.1 | Grundlagen-Training") | ||||||
|  |                 .minPoints(0) | ||||||
|  |                 .addChildCriteria( | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.1.1 | Fibonacci mit 1D Array") | ||||||
|  |                         .minPoints(0) | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode push: Das letzte Element des Ergebnis-Arrays ist das übergebene Element.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testPushLastElementCorrect", | ||||||
|  |                                         JsonParameterSet.class | ||||||
|  |                                     ) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode push: Die Elemente des Ergebnis-Arrays sind korrekt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testPushAllElementsCorrect", | ||||||
|  |                                         JsonParameterSet.class | ||||||
|  |                                     ) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode push: Das Eingabe-Array wird nicht verändert.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testPushOriginalArrayUnchanged", | ||||||
|  |                                         JsonParameterSet.class | ||||||
|  |                                     ) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode calculateNextFibonacci: Das Ergebnis ist korrekt mit zwei positiven Zahlen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testCalculateNextFibonacciPositiveOnly", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode calculateNextFibonacci: Das Ergebnis ist korrekt mit beliebigen Eingaben.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testCalculateNextFibonacciAllNumbers", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode calculateNextFibonacci: Eine verbindliche Anforderung wurde verletzt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testCalculateNextFibonacciVanforderungen", JsonParameterSet.class) | ||||||
|  |                                 ), | ||||||
|  |                                 -1 | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode fibonacci: Das Ergebnis ist korrekt für n < 2.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testFibonacciSmallerThanTwo", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode fibonacci: Das Ergebnis ist korrekt für n >= 2.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testFibonacciBigNumbers", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode fibonacci: Eine Verbindliche Anforderung wurde verletzt.", | ||||||
|  |                                 JUnitTestRef.and( | ||||||
|  |                                     JUnitTestRef.ofMethod( | ||||||
|  |                                         () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                             "testFibonacciVanforderungen", JsonParameterSet.class) | ||||||
|  |                                     ), | ||||||
|  |                                     JUnitTestRef.ofMethod( | ||||||
|  |                                         () -> OneDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                             "testFibonacciNonIterativeVanforderungen") | ||||||
|  |                                     ) | ||||||
|  |                                 ), | ||||||
|  |                                 -1 | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         .build(), | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.1.2 | Textsuche mit 2D Arrays") | ||||||
|  |                         .minPoints(0) | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode occurrences: Die Methode funktioniert korrekt mit einem leeren Array.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> TwoDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testOccurrencesEmptyArray", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode occurrences: Die Methode funktioniert mit einem Satz.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> TwoDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testOccurrencesSingleSentence", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode occurrences: Die Methode funktioniert mit mehreren Sätzen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> TwoDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testOccurrencesMultipleSentences", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode mean: Methode funktioniert mit ganzzahligen Rechenwerten.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> TwoDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testMeanInteger", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode mean: Die Methode funktioniert auch dann korrekt, wenn das Ergebnis eine fließkommazahl ist.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> TwoDimensionalArrayStuffTest.class.getDeclaredMethod( | ||||||
|  |                                         "testMeanFloat", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         .build() | ||||||
|  |                 ) | ||||||
|  |                 .build() | ||||||
|  |         ) | ||||||
|  |         .addChildCriteria( | ||||||
|  |             Criterion.builder() | ||||||
|  |                 .shortDescription("H2.2 | Vier Gewinnt") | ||||||
|  |                 .minPoints(0) | ||||||
|  |                 .addChildCriteria( | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.2.1 | Slot Prüfen") | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode validateInput: Methode ist vollständig korrekt implementiert.", | ||||||
|  |                                 JUnitTestRef.and( | ||||||
|  |                                     JUnitTestRef.ofMethod( | ||||||
|  |                                         () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                             "testValidateInputEdgeCases", JsonParameterSet.class) | ||||||
|  |                                     ), | ||||||
|  |                                     JUnitTestRef.ofMethod( | ||||||
|  |                                         () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                             "testValidateInputRandomCases", JsonParameterSet.class) | ||||||
|  |                                     ) | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         .build(), | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.2.2 | Münzen fallen lassen") | ||||||
|  |                         .minPoints(0) | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode getDestinationRow: Die Rückgabe ist korrekt, wenn ein freier Slot existiert.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGetDestinationRowFreeSlot", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode getDestinationRow: Die Rückgabe ist korrekt, wenn KEIN freier Slot existiert.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGetDestinationRowBlockedSlot", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode getDestinationRow: Verbindliche Anforderung 'genau eine Schleife' wurde verletzt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGetDestinationRowVAnforderung") | ||||||
|  |                                 ), | ||||||
|  |                                 -1 | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode dropStone: Robot wird mit korrekten Parametern erstellt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testDropStoneRobotCorrect", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode dropStone: getDestinationRow wird korrekt aufgerufen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testDropStoneCallsGetDestinationRow") | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode dropStone: Robot führt die korrekte Bewegung aus.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testDropStoneMovementCorrect", JsonParameterSet.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode dropStone: Verbindliche Anforderung 'genau eine Schleife' wurde verletzt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod("testDropStoneVAnforderung") | ||||||
|  |                                 ), | ||||||
|  |                                 -1 | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         .build(), | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.2.3 | Gewinnbedingung prüfen") | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             Criterion.builder() | ||||||
|  |                                 .shortDescription("Methode testWinHorizontal: ") | ||||||
|  |                                 .addChildCriteria( | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode erkennt richtige horizontale Steinfolgen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinHorizontal", JsonParameterSet.class) | ||||||
|  |                                         ), | ||||||
|  |                                         2 | ||||||
|  |                                     ), | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode nutzt genau zwei verschachtelte Schleifen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinHorizontalVAnforderung1") | ||||||
|  |                                         ), | ||||||
|  |                                         1 | ||||||
|  |                                     ), | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode erkennt keine falschen Steinfolgen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod("testTestWinHorizontalVAnforderung2") | ||||||
|  |                                         ), | ||||||
|  |                                         -2 | ||||||
|  |                                     ) | ||||||
|  |                                 ).minPoints(0).build(), | ||||||
|  |                             Criterion.builder() | ||||||
|  |                                 .shortDescription("Methode testWinVertical: ") | ||||||
|  |                                 .addChildCriteria( | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode erkennt richtige vertikale Steinfolgen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinVertical", JsonParameterSet.class) | ||||||
|  |                                         ), | ||||||
|  |                                         2 | ||||||
|  |                                     ), | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode nutzt genau zwei verschachtelte Schleifen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinVerticalVAnforderung1") | ||||||
|  |                                         ), | ||||||
|  |                                         1 | ||||||
|  |                                     ), | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Methode erkennt keine falschen Steinfolgen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinVerticalVAnforderung2") | ||||||
|  |                                         ), | ||||||
|  |                                         -2 | ||||||
|  |                                     ) | ||||||
|  |                                 ).minPoints(0).build(), | ||||||
|  |                             Criterion.builder() | ||||||
|  |                                 .shortDescription("Methode testWinConditions: ") | ||||||
|  |                                 .addChildCriteria( | ||||||
|  |                                     criterion( | ||||||
|  |                                         "Die Rückgabe ist in allen Fällen korrekt.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinConditions", int.class) | ||||||
|  |                                         ) | ||||||
|  |                                     ), | ||||||
|  |                                     criterion( | ||||||
|  |                                         "testWinHorizontal, testWinVertical und testWinDiagonal werden korrekt aufgerufen.", | ||||||
|  |                                         JUnitTestRef.ofMethod( | ||||||
|  |                                             () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                                 "testTestWinConditionsVAnforderung") | ||||||
|  |                                         ) | ||||||
|  |                                     ) | ||||||
|  |                                 ).build() | ||||||
|  |                         ) | ||||||
|  |                         .build(), | ||||||
|  |                     Criterion.builder() | ||||||
|  |                         .shortDescription("H2.2.4 | Game Loop") | ||||||
|  |                         .minPoints(0) | ||||||
|  |                         .addChildCriteria( | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode nextPlayer: Die Rückgabe für beide RobotFamily.SQUARE_BLUE und SQUARE_RED korrekt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testNextPlayer", boolean.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode colorFieldBackground: Das Spielfeld wird korrekt eingefärbt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testColorFieldBackground", boolean.class) | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methoden writeDrawMessage, writeWinnerMessage: Die Ausgabe in die Konsole ist korrekt.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testWriteMessages") | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode gameLoop: nextPlayer wird mit korrektem Parameter aufgerufen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGameLoopCallsNextPlayer") | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode gameLoop: dropStone wird mit korrekten Parametern aufgerufen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGameLoopCallsDropStone") | ||||||
|  |                                 ) | ||||||
|  |                             ), | ||||||
|  |                             criterion( | ||||||
|  |                                 "Methode gameLoop: testWinConditions wird mit korrekten Parametern aufgerufen.", | ||||||
|  |                                 JUnitTestRef.ofMethod( | ||||||
|  |                                     () -> FourWinsTest.class.getDeclaredMethod( | ||||||
|  |                                         "testGameLoopCallsGetWinConditions") | ||||||
|  |                                 ) | ||||||
|  |                             ) | ||||||
|  |                         ) | ||||||
|  |                         .build() | ||||||
|  |                 ) | ||||||
|  |                 .build() | ||||||
|  |         ) | ||||||
|  |         .build(); | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public Rubric getRubric() { | ||||||
|  |         return RUBRIC; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void configure(RubricConfiguration configuration) { | ||||||
|  |         configuration.addTransformer(new SubmissionTransformer()); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										327
									
								
								src/graderPrivate/java/h02/OneDimensionalArrayStuffTest.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										327
									
								
								src/graderPrivate/java/h02/OneDimensionalArrayStuffTest.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,327 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  | import org.junit.jupiter.api.Timeout; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.mockito.Answers; | ||||||
|  | import org.mockito.Mockito; | ||||||
|  | import org.mockito.exceptions.base.MockitoException; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.TestForSubmission; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.annotation.SkipAfterFirstFailedTest; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.Assertions2; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.Assertions4; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSet; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSetTest; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.reflections.BasicMethodLink; | ||||||
|  | import spoon.reflect.code.CtInvocation; | ||||||
|  | import spoon.reflect.code.CtLocalVariable; | ||||||
|  | 
 | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  | 
 | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.contextBuilder; | ||||||
|  | 
 | ||||||
|  | @TestForSubmission | ||||||
|  | @Timeout( | ||||||
|  |     value = TestConstants.TEST_TIMEOUT_IN_SECONDS, | ||||||
|  |     unit = TimeUnit.SECONDS, | ||||||
|  |     threadMode = Timeout.ThreadMode.SEPARATE_THREAD | ||||||
|  | ) | ||||||
|  | @SkipAfterFirstFailedTest(TestConstants.SKIP_AFTER_FIRST_FAILED_TEST) | ||||||
|  | public class OneDimensionalArrayStuffTest { | ||||||
|  |     /** | ||||||
|  |      * Tests the {@link OneDimensionalArrayStuff#push(int[], int)} method. | ||||||
|  |      * | ||||||
|  |      * @param params        The {@link JsonParameterSet} to use for the test. | ||||||
|  |      * @param lastOnly      Whether to only test the last element. | ||||||
|  |      * @param unchangedOnly Whether to only test that the input array is unchanged. Will not test the result. | ||||||
|  |      */ | ||||||
|  |     private static void testPush(final JsonParameterSet params, final boolean lastOnly, final boolean unchangedOnly) { | ||||||
|  |         final List<Integer> input = params.get("array"); | ||||||
|  |         final int value = params.getInt("value"); | ||||||
|  |         final List<Integer> expectedArray = params.get("expected result"); | ||||||
|  |         final var ParamsContext = params.toContext(); | ||||||
|  |         final var cb = contextBuilder() | ||||||
|  |             .add(ParamsContext) | ||||||
|  |             .add("Method", "push"); | ||||||
|  |         final int[] inputArray = input.stream().mapToInt(i -> i).toArray(); | ||||||
|  |         final int[] result = Assertions2.callObject( | ||||||
|  |             () -> OneDimensionalArrayStuff.push( | ||||||
|  |                 inputArray, | ||||||
|  |                 value | ||||||
|  |             ), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "An error occurred during execution." | ||||||
|  |         ); | ||||||
|  |         final var actualArray = Arrays.stream(result).boxed().toList(); | ||||||
|  |         cb.add("actual result", actualArray); | ||||||
|  |         if (unchangedOnly) { | ||||||
|  |             Assertions2.assertIterableEquals( | ||||||
|  |                 input, | ||||||
|  |                 Arrays.stream(inputArray).boxed().toList(), | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "The input array was changed." | ||||||
|  |             ); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (lastOnly) { | ||||||
|  |             Assertions2.assertEquals( | ||||||
|  |                 expectedArray.get(expectedArray.size() - 1), | ||||||
|  |                 actualArray.get(actualArray.size() - 1), | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "Invalid result." | ||||||
|  |             ); | ||||||
|  |         } else { | ||||||
|  |             Assertions2.assertIterableEquals( | ||||||
|  |                 expectedArray, | ||||||
|  |                 actualArray, | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "Invalid result." | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void testCalculateNextFibonacci(final JsonParameterSet params, final boolean vanforderung) { | ||||||
|  |         try (final var odasMock = Mockito.mockStatic(OneDimensionalArrayStuff.class, Answers.CALLS_REAL_METHODS)) { | ||||||
|  |             final List<Integer> input = params.get("array"); | ||||||
|  |             final List<Integer> expectedArray = params.get("expected result"); | ||||||
|  |             final var ParamsContext = params.toContext(); | ||||||
|  |             final var cb = contextBuilder() | ||||||
|  |                 .add(ParamsContext) | ||||||
|  |                 .add("Method", "calculateNextFibonacci"); | ||||||
|  | 
 | ||||||
|  |             odasMock.when(() -> OneDimensionalArrayStuff.push(Mockito.any(), Mockito.anyInt())).thenAnswer(invocation -> { | ||||||
|  |                 final int[] array = invocation.getArgument(0); | ||||||
|  |                 final int value = invocation.getArgument(1); | ||||||
|  |                 final int[] newArray = Arrays.copyOf(array, array.length + 1); | ||||||
|  |                 newArray[array.length] = value; | ||||||
|  |                 return newArray; | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             final int[] inputArray = input.stream().mapToInt(i -> i).toArray(); | ||||||
|  |             final int[] result = Assertions2.callObject( | ||||||
|  |                 () -> OneDimensionalArrayStuff.calculateNextFibonacci( | ||||||
|  |                     inputArray | ||||||
|  |                 ), | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "An error occurred during execution." | ||||||
|  |             ); | ||||||
|  |             final var actualArray = Arrays.stream(result).boxed().toList(); | ||||||
|  |             cb.add("actual result", actualArray); | ||||||
|  |             if (vanforderung) { | ||||||
|  |                 Assertions2.assertIterableEquals( | ||||||
|  |                     input, | ||||||
|  |                     Arrays.stream(inputArray).boxed().toList(), | ||||||
|  |                     cb.build(), | ||||||
|  |                     r -> "The input array was changed." | ||||||
|  |                 ); | ||||||
|  |                 odasMock.verify( | ||||||
|  |                     () -> OneDimensionalArrayStuff.push( | ||||||
|  |                         inputArray, | ||||||
|  |                         expectedArray.get(expectedArray.size() - 1) | ||||||
|  |                     ), | ||||||
|  |                     Mockito.atLeastOnce() | ||||||
|  |                 ); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             Assertions2.assertIterableEquals( | ||||||
|  |                 expectedArray, | ||||||
|  |                 actualArray, | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "Invalid result." | ||||||
|  |             ); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void testFibonacci(final JsonParameterSet params, final boolean vanforderung) { | ||||||
|  |         try (final var odasMock = Mockito.mockStatic(OneDimensionalArrayStuff.class, Answers.CALLS_REAL_METHODS)) { | ||||||
|  |             final Integer input = params.get("n"); | ||||||
|  |             final int expectedresult = params.get("expected result"); | ||||||
|  |             final List<Integer> referenceArray = params.get("reference array"); | ||||||
|  |             final var ParamsContext = params.toContext("reference array"); | ||||||
|  |             final var cb = contextBuilder() | ||||||
|  |                 .add(ParamsContext) | ||||||
|  |                 .add("Method", "fibonacci"); | ||||||
|  | 
 | ||||||
|  |             odasMock.when(() -> OneDimensionalArrayStuff.push(Mockito.any(), Mockito.anyInt())).thenAnswer(invocation -> { | ||||||
|  |                 final int[] array = invocation.getArgument(0); | ||||||
|  |                 final int value = invocation.getArgument(1); | ||||||
|  |                 final int[] newArray = Arrays.copyOf(array, array.length + 1); | ||||||
|  |                 newArray[array.length] = value; | ||||||
|  |                 return newArray; | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             odasMock.when(() -> OneDimensionalArrayStuff.calculateNextFibonacci(Mockito.any())).thenAnswer(invocation -> { | ||||||
|  |                 final int[] array = invocation.getArgument(0); | ||||||
|  |                 return OneDimensionalArrayStuff.push(array, array[array.length - 1] + array[array.length - 2]); | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             final int result = Assertions2.callObject( | ||||||
|  |                 () -> OneDimensionalArrayStuff.fibonacci( | ||||||
|  |                     input | ||||||
|  |                 ), | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "An error occurred during execution." | ||||||
|  |             ); | ||||||
|  | 
 | ||||||
|  |             cb.add("actual result", result); | ||||||
|  |             if (vanforderung) { | ||||||
|  |                 // test calculateFib was used | ||||||
|  |                 for (int i = 2; i < referenceArray.size(); i++) { | ||||||
|  |                     final int finalI = i; | ||||||
|  |                     try { | ||||||
|  |                         // test for entire array | ||||||
|  |                         odasMock.verify( | ||||||
|  |                             () -> OneDimensionalArrayStuff.calculateNextFibonacci( | ||||||
|  |                                 referenceArray.subList(0, finalI).stream().mapToInt(j -> j).toArray() | ||||||
|  |                             ), | ||||||
|  |                             Mockito.atLeastOnce() | ||||||
|  |                         ); | ||||||
|  |                     } catch (final MockitoException e) { | ||||||
|  |                         // test for last two elements | ||||||
|  |                         odasMock.verify( | ||||||
|  |                             () -> OneDimensionalArrayStuff.calculateNextFibonacci( | ||||||
|  |                                 referenceArray.subList(finalI - 2, finalI).stream().mapToInt(j -> j).toArray() | ||||||
|  |                             ), | ||||||
|  |                             Mockito.atLeastOnce() | ||||||
|  |                         ); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             Assertions2.assertEquals( | ||||||
|  |                 expectedresult, | ||||||
|  |                 result, | ||||||
|  |                 cb.build(), | ||||||
|  |                 r -> "Invalid result." | ||||||
|  |             ); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestPushRandomNumbers.generated.json") | ||||||
|  |     public void testPushLastElementCorrect(final JsonParameterSet params) { | ||||||
|  |         testPush(params, true, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestPushRandomNumbers.generated.json") | ||||||
|  |     public void testPushAllElementsCorrect(final JsonParameterSet params) { | ||||||
|  |         testPush(params, false, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestPushRandomNumbers.generated.json") | ||||||
|  |     public void testPushOriginalArrayUnchanged(final JsonParameterSet params) { | ||||||
|  |         testPush(params, false, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestCalculateNextFibonacciRandomNumbersTwoPositiveNumbersOnly.generated.json") | ||||||
|  |     public void testCalculateNextFibonacciPositiveOnly(final JsonParameterSet params) { | ||||||
|  |         testCalculateNextFibonacci(params, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestCalculateNextFibonacciRandomNumbers.generated.json") | ||||||
|  |     public void testCalculateNextFibonacciAllNumbers(final JsonParameterSet params) { | ||||||
|  |         testCalculateNextFibonacci(params, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestCalculateNextFibonacciRandomNumbers.generated.json") | ||||||
|  |     public void testCalculateNextFibonacciVanforderungen(final JsonParameterSet params) { | ||||||
|  |         testCalculateNextFibonacci(params, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestFibonacciRandomNumbers.generated.json") | ||||||
|  |     public void testFibonacciVanforderungen(final JsonParameterSet params) { | ||||||
|  |         testFibonacci(params, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testFibonacciNonIterativeVanforderungen() throws NoSuchMethodException { | ||||||
|  |         // test exactly one for loop | ||||||
|  |         final var ml = BasicMethodLink.of(OneDimensionalArrayStuff.class.getDeclaredMethod("fibonacci", int.class)); | ||||||
|  |         final var ctElement = ml.getCtElement(); | ||||||
|  |         final var cb = contextBuilder() | ||||||
|  |             .add("Method", "fibonacci"); | ||||||
|  |         Assertions4.assertIsNotRecursively( | ||||||
|  |             ctElement, | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should not have any recursive calls." | ||||||
|  |         ); | ||||||
|  |         final var forLoops = ctElement.filterChildren( | ||||||
|  |             c -> c instanceof spoon.reflect.code.CtFor | ||||||
|  |                 || c instanceof spoon.reflect.code.CtForEach | ||||||
|  |                 || c instanceof spoon.reflect.code.CtWhile | ||||||
|  |                 || c instanceof spoon.reflect.code.CtDo | ||||||
|  |         ).list(); | ||||||
|  |         Assertions2.assertEquals( | ||||||
|  |             1, | ||||||
|  |             forLoops.size(), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should contain exactly one for loop." | ||||||
|  |         ); | ||||||
|  |         // test exactly one variable declaration | ||||||
|  |         final var variableDeclarations = ctElement.filterChildren( | ||||||
|  |             c -> c instanceof CtLocalVariable<?> | ||||||
|  |         ).list(CtLocalVariable.class); | ||||||
|  |         cb.add("variable declarations", variableDeclarations); | ||||||
|  |         Assertions2.assertEquals( | ||||||
|  |             2, | ||||||
|  |             variableDeclarations.size(), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should contain exactly two variable declarations." | ||||||
|  |         ); | ||||||
|  |         // one of type int[] and one of type int | ||||||
|  |         Assertions2.assertTrue( | ||||||
|  |             variableDeclarations.stream().anyMatch( | ||||||
|  |                 c -> c.getType().toString().equals("int[]") | ||||||
|  |             ), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should declare exactly one variable of type int[]." | ||||||
|  |         ); | ||||||
|  |         Assertions2.assertTrue( | ||||||
|  |             variableDeclarations.stream().anyMatch( | ||||||
|  |                 c -> c.getType().toString().equals("int") | ||||||
|  |             ), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should contain exactly one variable of type int." | ||||||
|  |         ); | ||||||
|  |         // test no Method calls except for calculateNextFibonacci | ||||||
|  |         final var methodCalls = ctElement.filterChildren( | ||||||
|  |             c -> c instanceof spoon.reflect.code.CtInvocation<?> | ||||||
|  |         ).list(CtInvocation.class); | ||||||
|  |                cb.add("method calls", methodCalls); | ||||||
|  |         Assertions2.assertEquals( | ||||||
|  |             1, | ||||||
|  |             methodCalls.size(), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should contain exactly one method call." | ||||||
|  |         ); | ||||||
|  |         final var methodCall = methodCalls.get(0); | ||||||
|  |         Assertions2.assertEquals( | ||||||
|  |             "calculateNextFibonacci", | ||||||
|  |             methodCall.getExecutable().getSimpleName(), | ||||||
|  |             cb.build(), | ||||||
|  |             r -> "The method should only call calculateNextFibonacci." | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestFibonacciRandomNumbersSmallerThanTwo.generated.json") | ||||||
|  |     public void testFibonacciSmallerThanTwo(final JsonParameterSet params) { | ||||||
|  |         testFibonacci(params, false); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest(value = "OneDimensionalArrayStuffTestFibonacciRandomNumbers.generated.json") | ||||||
|  |     public void testFibonacciBigNumbers(final JsonParameterSet params) { | ||||||
|  |         testFibonacci(params, false); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										64
									
								
								src/graderPrivate/java/h02/SubmissionTransformer.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								src/graderPrivate/java/h02/SubmissionTransformer.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.objectweb.asm.*; | ||||||
|  | import org.sourcegrade.jagr.api.testing.ClassTransformer; | ||||||
|  | 
 | ||||||
|  | import static org.objectweb.asm.Opcodes.*; | ||||||
|  | 
 | ||||||
|  | public class SubmissionTransformer implements ClassTransformer { | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public String getName() { | ||||||
|  |         return "MethodInjectingTransformer"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void transform(ClassReader reader, ClassWriter writer) { | ||||||
|  |         if (reader.getClassName().equals("h02/FourWins")) { | ||||||
|  |             reader.accept(new FourWinsClassVisitor(writer), 0); | ||||||
|  |         } else { | ||||||
|  |             reader.accept(writer, 0); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static class FourWinsClassVisitor extends ClassVisitor { | ||||||
|  | 
 | ||||||
|  |         private FourWinsClassVisitor(ClassWriter classWriter) { | ||||||
|  |             super(Opcodes.ASM9, classWriter); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         @Override | ||||||
|  |         public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) { | ||||||
|  |             return new MethodVisitor(api, super.visitMethod(access, name, descriptor, signature, exceptions)) { | ||||||
|  |                 @Override | ||||||
|  |                 public void visitMethodInsn(int opcode, String owner, String name, String descriptor, boolean isInterface) { | ||||||
|  |                     if (owner.equals("h02/template/InputHandler") && name.equals("install") && descriptor.equals("()V")) { | ||||||
|  |                         super.visitInsn(POP); | ||||||
|  |                     } else { | ||||||
|  |                         super.visitMethodInsn(opcode, owner, name, descriptor, isInterface); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         @Override | ||||||
|  |         public void visitEnd() { | ||||||
|  |             Label startLabel = new Label(); | ||||||
|  |             Label endLabel = new Label(); | ||||||
|  |             MethodVisitor mv = super.visitMethod(ACC_PUBLIC, | ||||||
|  |                 "getInputHandler", | ||||||
|  |                 "()Lh02/template/InputHandler;", | ||||||
|  |                 null, | ||||||
|  |                 null); | ||||||
|  |             mv.visitLabel(startLabel); | ||||||
|  |             mv.visitVarInsn(ALOAD, 0); | ||||||
|  |             mv.visitFieldInsn(GETFIELD, "h02/FourWins", "inputHandler", "Lh02/template/InputHandler;"); | ||||||
|  |             mv.visitInsn(ARETURN); | ||||||
|  |             mv.visitLabel(endLabel); | ||||||
|  |             mv.visitLocalVariable("this", "Lh02/FourWins;", null, startLabel, endLabel, 0); | ||||||
|  |             mv.visitMaxs(1, 1); | ||||||
|  | 
 | ||||||
|  |             super.visitEnd(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										154
									
								
								src/graderPrivate/java/h02/TemplateAnnotationProcessor.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								src/graderPrivate/java/h02/TemplateAnnotationProcessor.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,154 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.SolutionOnly; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentCreationRequired; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentImplementationRequired; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.reflections.BasicPackageLink; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.reflections.BasicTypeLink; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.reflections.TypeLink; | ||||||
|  | import spoon.Launcher; | ||||||
|  | import spoon.reflect.CtModel; | ||||||
|  | import spoon.reflect.code.CtCodeSnippetStatement; | ||||||
|  | import spoon.reflect.declaration.CtAnnotation; | ||||||
|  | import spoon.reflect.declaration.CtElement; | ||||||
|  | import spoon.reflect.declaration.CtMethod; | ||||||
|  | import spoon.reflect.declaration.CtRecord; | ||||||
|  | import spoon.reflect.factory.Factory; | ||||||
|  | import spoon.support.sniper.SniperJavaPrettyPrinter; | ||||||
|  | 
 | ||||||
|  | import java.io.File; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.lang.annotation.Annotation; | ||||||
|  | import java.nio.file.Files; | ||||||
|  | import java.nio.file.Path; | ||||||
|  | import java.util.Collection; | ||||||
|  | import java.util.Comparator; | ||||||
|  | import java.util.HashSet; | ||||||
|  | import java.util.Set; | ||||||
|  | import java.util.concurrent.atomic.AtomicBoolean; | ||||||
|  | import java.util.function.Consumer; | ||||||
|  | 
 | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.ResourceUtils.toPathString; | ||||||
|  | 
 | ||||||
|  | public class TemplateAnnotationProcessor { | ||||||
|  |     private static Launcher createSpoonLauncher(final boolean sniper) { | ||||||
|  |         final Launcher spoon = new Launcher(); | ||||||
|  |         spoon.getEnvironment().setAutoImports(true); | ||||||
|  |         spoon.getEnvironment().setComplianceLevel(17); | ||||||
|  |         spoon.getEnvironment().setNoClasspath(true); | ||||||
|  |         spoon.getEnvironment().setCommentEnabled(true); | ||||||
|  |         spoon.getEnvironment().setIgnoreSyntaxErrors(true); | ||||||
|  |         if (sniper) { | ||||||
|  |             spoon.getEnvironment().setPrettyPrinterCreator( | ||||||
|  |                 () -> new SniperJavaPrettyPrinter(spoon.getEnvironment()) | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         return spoon; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void execute() { | ||||||
|  |         // final Collection<BasicTypeLink> types = List.of(BasicTypeLink.of(TestClass.class)); | ||||||
|  |         final Collection<BasicTypeLink> types = BasicPackageLink.of("h02", true).getTypes(); | ||||||
|  |         Path tmpDir = null; | ||||||
|  |         try { | ||||||
|  |             tmpDir = Files.createTempDirectory("spoon_output"); | ||||||
|  |             System.out.println("tmpdir: " + tmpDir.toString()); | ||||||
|  |             for (final BasicTypeLink type : types) { | ||||||
|  |                 if (type.reflection().getName().contains("TemplateAnnotationProcessor") || type.reflection().getName().contains("TestClass")) { | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                 System.out.println("processing " + type.name()); | ||||||
|  |                 final AtomicBoolean modified = new AtomicBoolean(false); | ||||||
|  |                 Launcher spoon = createSpoonLauncher(true); | ||||||
|  |                 spoon.addInputResource(getFilePath(type).toString()); | ||||||
|  |                 CtModel model = spoon.buildModel(); | ||||||
|  |                 var spoonClass = model.getAllTypes().stream().filter(it -> it.getQualifiedName().equals(type.reflection().getName())).findFirst().orElse(null); | ||||||
|  |                 if (spoonClass == null || spoonClass instanceof CtRecord) { | ||||||
|  |                     // workaround for records because spoon hates them | ||||||
|  |                     spoon = createSpoonLauncher(false); | ||||||
|  |                     spoon.addInputResource(getFilePath(type).toString()); | ||||||
|  |                     model = spoon.buildModel(); | ||||||
|  |                     spoonClass = model.getAllTypes().stream().filter(it -> it.getQualifiedName().equals(type.reflection().getName())).findFirst().orElseThrow(); | ||||||
|  |                 } | ||||||
|  |                 final Factory factory = spoon.getFactory(); | ||||||
|  |                 processAnnotation(spoonClass, StudentImplementationRequired.class, element -> { | ||||||
|  |                     // get annotation text | ||||||
|  |                     final var exercise = element.getAnnotation(StudentImplementationRequired.class).value(); | ||||||
|  |                     if (element instanceof final CtMethod<?> method) { | ||||||
|  |                         modified.set(true); | ||||||
|  |                         // Create a new code snippet statement with the desired code | ||||||
|  |                         String optionalReturn = ""; | ||||||
|  |                         if (!method.getType().getSimpleName().equals("void")) { | ||||||
|  |                             optionalReturn = "return "; | ||||||
|  |                         } | ||||||
|  |                         final CtCodeSnippetStatement snippet = method.getFactory().Code().createCodeSnippetStatement("// TODO: " + exercise + "\n" + optionalReturn + "org.tudalgo.algoutils.student.Student.crash(\"" + exercise + " - Remove if implemented\")"); | ||||||
|  |                         // Set the code snippet as the body of the method | ||||||
|  |                         method.setBody(snippet); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |                 processAnnotation(spoonClass, Set.of(SolutionOnly.class, StudentCreationRequired.class), element -> { | ||||||
|  |                     if (element instanceof final CtMethod<?> method) { | ||||||
|  |                         modified.set(true); | ||||||
|  |                         // remove annotations | ||||||
|  |                         method.getAnnotations().forEach(CtAnnotation::delete); | ||||||
|  |                         // remove method | ||||||
|  |                         method.delete(); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |                 if (!modified.get()) { | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                 // Save the modified class | ||||||
|  |                 spoon.getEnvironment().setSourceOutputDirectory(new File(tmpDir.toString())); | ||||||
|  |                 // overwrite the file if it already exists | ||||||
|  | //                spoon.getEnvironment().setAutoImports(true); | ||||||
|  | 
 | ||||||
|  |                 spoon.prettyprint(); | ||||||
|  |                 final var path = getFilePath(type); | ||||||
|  |                 final var outputPath = Path.of(tmpDir.toString(), toPathString(type.reflection().getName())); | ||||||
|  |                 Files.deleteIfExists(path); | ||||||
|  |                 Files.move(outputPath, path); | ||||||
|  |             } | ||||||
|  |         } catch (final IOException e) { | ||||||
|  |             throw new RuntimeException(e); | ||||||
|  |         } finally { | ||||||
|  |             try (final var walk = Files.walk(tmpDir)) { | ||||||
|  |                 walk.sorted(Comparator.reverseOrder()).map(Path::toFile).forEach(File::delete); | ||||||
|  |             } catch (final IOException e) { | ||||||
|  |                 throw new RuntimeException("Could not delete tmp dir", e); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static Path getFilePath(final TypeLink type) { | ||||||
|  |         final String pathString = toPathString(type.reflection().getName()); | ||||||
|  |         // find the file | ||||||
|  |         try (final var walk = Files.walk(Path.of("."))) { | ||||||
|  |             return walk.filter(p -> p.endsWith(pathString)).findFirst().orElseThrow(); | ||||||
|  |         } catch (final IOException e) { | ||||||
|  |             throw new RuntimeException("an error occurred while reading a source files ", e); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void processAnnotation( | ||||||
|  |         final CtElement root, final Set<Class<? extends Annotation>> annotations, | ||||||
|  |         final Consumer<CtElement> consumer | ||||||
|  |     ) { | ||||||
|  |         CtElement element = root.filterChildren(c -> annotations.stream().anyMatch(c::hasAnnotation)).first(); | ||||||
|  |         final Set<CtElement> processed = new HashSet<>(); | ||||||
|  |         while (element != null && !processed.contains(element)) { | ||||||
|  |             consumer.accept(element); | ||||||
|  |             processed.add(element); | ||||||
|  |             element = root.filterChildren(c -> !processed.contains(c) && annotations.stream().anyMatch(c::hasAnnotation)).first(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void processAnnotation( | ||||||
|  |         final CtElement root, final Class<? extends Annotation> annotation, | ||||||
|  |         final Consumer<CtElement> consumer | ||||||
|  |     ) { | ||||||
|  |         processAnnotation(root, Set.of(annotation), consumer); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								src/graderPrivate/java/h02/TestConstants.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/graderPrivate/java/h02/TestConstants.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,19 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import java.util.concurrent.ThreadLocalRandom; | ||||||
|  | 
 | ||||||
|  | public class TestConstants { | ||||||
|  |     public static long RANDOM_SEED = ThreadLocalRandom.current().nextLong(); | ||||||
|  |     public static final boolean SHOW_WORLD = java.lang.management.ManagementFactory | ||||||
|  |         .getRuntimeMXBean() | ||||||
|  |         .getInputArguments() | ||||||
|  |         .toString() | ||||||
|  |         .contains("-agentlib:jdwp"); // true if debugger is attached | ||||||
|  |     public static final int WORLD_DELAY = 500; | ||||||
|  | 
 | ||||||
|  |     public static final int TEST_TIMEOUT_IN_SECONDS = 10; | ||||||
|  | 
 | ||||||
|  |     public static final int TEST_ITERATIONS = 30; | ||||||
|  | 
 | ||||||
|  |     public static final boolean SKIP_AFTER_FIRST_FAILED_TEST = true; | ||||||
|  | } | ||||||
							
								
								
									
										267
									
								
								src/graderPrivate/java/h02/TestJsonGenerators.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										267
									
								
								src/graderPrivate/java/h02/TestJsonGenerators.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,267 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.databind.JsonNode; | ||||||
|  | import com.fasterxml.jackson.databind.node.ArrayNode; | ||||||
|  | import com.fasterxml.jackson.databind.node.ObjectNode; | ||||||
|  | import fopbot.RobotFamily; | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  | import org.junit.jupiter.api.condition.DisabledIf; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.junit.jupiter.params.provider.ValueSource; | ||||||
|  | 
 | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.List; | ||||||
|  | 
 | ||||||
|  | import static h02.TestConstants.TEST_ITERATIONS; | ||||||
|  | 
 | ||||||
|  | @DisabledIf("org.tudalgo.algoutils.tutor.general.Utils#isJagrRun()") | ||||||
|  | public class TestJsonGenerators { | ||||||
|  |     @Test | ||||||
|  |     public void generateOneDimensionalArrayStuffTestPushRandomNumbers() throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 final ObjectNode objectNode = mapper.createObjectNode(); | ||||||
|  |                 objectNode.put("value", rnd.nextInt((int) -2e5, (int) 2e5)); | ||||||
|  |                 final List<Integer> input = new ArrayList<>(); | ||||||
|  |                 if (index < 99) { | ||||||
|  |                     for (int i = 0; i < rnd.nextInt(5, 10); i++) { | ||||||
|  |                         input.add(rnd.nextInt((int) -2e5, (int) 2e5)); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 final ArrayNode inputArrayNode = mapper.createArrayNode(); | ||||||
|  |                 input.forEach(inputArrayNode::add); | ||||||
|  |                 objectNode.set("array", inputArrayNode); | ||||||
|  |                 input.add(objectNode.get("value").asInt()); | ||||||
|  |                 final ArrayNode expectedArrayNode = mapper.createArrayNode(); | ||||||
|  |                 input.forEach(expectedArrayNode::add); | ||||||
|  |                 objectNode.set("expected result", expectedArrayNode); | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             TEST_ITERATIONS, | ||||||
|  |             "OneDimensionalArrayStuffTestPushRandomNumbers.generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(booleans = {true, false}) | ||||||
|  |     public void generateOneDimensionalArrayStuffTestCalculateNextFibonacciRandomNumbers( | ||||||
|  |         final boolean twoPositivesOnly | ||||||
|  |     ) throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 final ObjectNode objectNode = mapper.createObjectNode(); | ||||||
|  |                 final List<Integer> input = new ArrayList<>(); | ||||||
|  |                 for (int i = 0; i < (twoPositivesOnly ? 2 : rnd.nextInt(5, 10)); i++) { | ||||||
|  |                     input.add(rnd.nextInt( | ||||||
|  |                         twoPositivesOnly ? 0 : (int) -2e5, | ||||||
|  |                         (int) 2e5 | ||||||
|  |                     )); | ||||||
|  |                 } | ||||||
|  |                 final ArrayNode inputArrayNode = mapper.createArrayNode(); | ||||||
|  |                 input.forEach(inputArrayNode::add); | ||||||
|  |                 objectNode.set("array", inputArrayNode); | ||||||
|  |                 System.out.println(input.size()); | ||||||
|  |                 final int nextFibonacci = input.get(input.size() - 1) + input.get(input.size() - 2); | ||||||
|  |                 input.add(nextFibonacci); | ||||||
|  |                 final ArrayNode expectedArrayNode = mapper.createArrayNode(); | ||||||
|  |                 input.forEach(expectedArrayNode::add); | ||||||
|  |                 objectNode.set("expected result", expectedArrayNode); | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             TEST_ITERATIONS, | ||||||
|  |             "OneDimensionalArrayStuffTestCalculateNextFibonacciRandomNumbers" + (twoPositivesOnly ? "TwoPositiveNumbersOnly" : "") + ".generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Reference Fibonacci implementation using the closed-form formula. | ||||||
|  |      * | ||||||
|  |      * @param n The number to calculate the Fibonacci number for. | ||||||
|  |      * @return The Fibonacci number. | ||||||
|  |      * @see <a href="https://en.wikipedia.org/wiki/Fibonacci_sequence#Closed-form_expression">Fibonacci Closed-form expression on Wikipedia</a> | ||||||
|  |      */ | ||||||
|  |     public static long fib(final int n) { | ||||||
|  |         final double sqrt5 = Math.sqrt(5); | ||||||
|  |         final double phi = (1 + sqrt5) / 2; | ||||||
|  |         final double psi = (1 - sqrt5) / 2; | ||||||
|  | 
 | ||||||
|  |         return Math.round((Math.pow(phi, n) - Math.pow(psi, n)) / sqrt5); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(booleans = {true, false}) | ||||||
|  |     public void generateOneDimensionalArrayStuffTestFibonacciRandomNumbers( | ||||||
|  |         final boolean smallerThanTwo | ||||||
|  |     ) throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 final int startIdx = smallerThanTwo ? index : index + 2; | ||||||
|  |                 final ObjectNode objectNode = mapper.createObjectNode(); | ||||||
|  |                 objectNode.put("n", startIdx); | ||||||
|  |                 objectNode.put("expected result", fib(startIdx)); | ||||||
|  |                 final ArrayNode refArrayNode = mapper.createArrayNode(); | ||||||
|  |                 for (int i = 0; i <= startIdx; i++) { | ||||||
|  |                     refArrayNode.add(fib(i)); | ||||||
|  |                 } | ||||||
|  |                 objectNode.set("reference array", refArrayNode); | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             smallerThanTwo ? 2 : TEST_ITERATIONS, | ||||||
|  |             "OneDimensionalArrayStuffTestFibonacciRandomNumbers" + (smallerThanTwo ? "SmallerThanTwo" : "") + ".generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void generateValidateInputRandomCases() throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 final ObjectNode objectNode = mapper.createObjectNode(); | ||||||
|  | 
 | ||||||
|  |                 // Spielfeldgröße zufällig wählen | ||||||
|  |                 final int width = rnd.nextInt(3, 10); // Breite zwischen 3 und 10 | ||||||
|  |                 final int height = rnd.nextInt(3, 10); // Höhe zwischen 3 und 10 | ||||||
|  |                 objectNode.put("width", width); | ||||||
|  |                 objectNode.put("height", height); | ||||||
|  | 
 | ||||||
|  |                 // Zufälliges Spielfeld generieren | ||||||
|  |                 RobotFamily[][] stones = new RobotFamily[height][width]; | ||||||
|  |                 for (int row = 0; row < height; row++) { | ||||||
|  |                     for (int col = 0; col < width; col++) { | ||||||
|  |                         if (rnd.nextBoolean()) { | ||||||
|  |                             stones[row][col] = rnd.nextBoolean() ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // Spaltenindex zufällig wählen (auch ungültige Indizes) | ||||||
|  |                 final int column = rnd.nextInt(-1, width + 1); | ||||||
|  |                 objectNode.put("column", column); | ||||||
|  | 
 | ||||||
|  |                 // Erwartetes Ergebnis berechnen | ||||||
|  |                 boolean expectedResult = column >= 0 && column < width && stones[height - 1][column] == null; | ||||||
|  |                 objectNode.put("expected result", expectedResult); | ||||||
|  | 
 | ||||||
|  |                 // Spielfeld in JSON-Format umwandeln | ||||||
|  |                 ArrayNode stonesArray = mapper.createArrayNode(); | ||||||
|  |                 for (int row = 0; row < height; row++) { | ||||||
|  |                     ArrayNode rowArray = mapper.createArrayNode(); | ||||||
|  |                     for (int col = 0; col < width; col++) { | ||||||
|  |                         if (stones[row][col] == null) { | ||||||
|  |                             rowArray.add("EMPTY"); | ||||||
|  |                         } else if (stones[row][col] == RobotFamily.SQUARE_RED) { | ||||||
|  |                             rowArray.add("SQUARE_RED"); | ||||||
|  |                         } else { | ||||||
|  |                             rowArray.add("SQUARE_BLUE"); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     stonesArray.add(rowArray); | ||||||
|  |                 } | ||||||
|  |                 objectNode.set("stones", stonesArray); | ||||||
|  | 
 | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             TEST_ITERATIONS, | ||||||
|  |             "FourWinsTestValidateInputRandomCases.generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void generateFourWinsTestGameBoard() throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 int worldHeight = rnd.nextInt(5, 10); | ||||||
|  |                 int worldWidth = rnd.nextInt(5, 10); | ||||||
|  |                 // SQUARE_RED <-> true, SQUARE_BLUE <-> false, null <-> null | ||||||
|  |                 RobotFamily[][] gameBoard = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |                 List<Integer> firstFreeIndex = new ArrayList<>(worldWidth);  // values may exceed array index range | ||||||
|  |                 for (int col = 0; col < worldWidth; col++) { | ||||||
|  |                     int rowsToFill = rnd.nextInt(worldHeight); | ||||||
|  |                     for (int row = 0; row <= rowsToFill; row++) { | ||||||
|  |                         gameBoard[row][col] = RobotFamily.SQUARE_RED; | ||||||
|  |                     } | ||||||
|  |                     firstFreeIndex.add(rowsToFill + 1); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 ArrayNode firstFreeIndexNode = mapper.createArrayNode(); | ||||||
|  |                 firstFreeIndex.stream() | ||||||
|  |                     .map(mapper.getNodeFactory()::numberNode) | ||||||
|  |                     .forEach(firstFreeIndexNode::add); | ||||||
|  | 
 | ||||||
|  |                 ArrayNode gameBoardNode = mapper.createArrayNode(); | ||||||
|  |                 Arrays.stream(gameBoard) | ||||||
|  |                     .map(gameBoardRow -> Arrays.stream(gameBoardRow) | ||||||
|  |                         .map(rf -> mapper.getNodeFactory().textNode(rf != null ? rf.getName() : null)) | ||||||
|  |                         .toList()) | ||||||
|  |                     .forEach(gameBoardRow -> gameBoardNode.add(mapper.createArrayNode().addAll(gameBoardRow))); | ||||||
|  | 
 | ||||||
|  |                 ObjectNode objectNode = mapper.createObjectNode() | ||||||
|  |                     .put("worldHeight", worldHeight) | ||||||
|  |                     .put("worldWidth", worldWidth); | ||||||
|  |                 objectNode.set("firstFreeIndex", firstFreeIndexNode); | ||||||
|  |                 objectNode.set("gameBoard", gameBoardNode); | ||||||
|  | 
 | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             TEST_ITERATIONS, | ||||||
|  |             "FourWinsTestGameBoard.generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(booleans = {true, false}) | ||||||
|  |     public void generateFourWinsTestGameBoardWin(boolean horizontal) throws IOException { | ||||||
|  |         TestUtils.generateJsonTestData( | ||||||
|  |             (mapper, index, rnd) -> { | ||||||
|  |                 int worldHeight = rnd.nextInt(5, 10); | ||||||
|  |                 int worldWidth = rnd.nextInt(5, 10); | ||||||
|  |                 RobotFamily currentPlayer = rnd.nextBoolean() ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |                 ObjectNode objectNode = mapper.createObjectNode() | ||||||
|  |                     .put("worldHeight", worldHeight) | ||||||
|  |                     .put("worldWidth", worldWidth) | ||||||
|  |                     .put("currentPlayer", currentPlayer.getName()); | ||||||
|  | 
 | ||||||
|  |                 ArrayNode winningCoordinates = mapper.createArrayNode(); | ||||||
|  |                 rnd.ints(rnd.nextInt(3), 0, horizontal ? worldHeight : worldWidth) | ||||||
|  |                     .distinct() | ||||||
|  |                     .forEach(i -> { | ||||||
|  |                         if (horizontal) { | ||||||
|  |                             winningCoordinates.add(mapper.createObjectNode() | ||||||
|  |                                 .put("x", rnd.nextInt(worldWidth - 4 + 1)) | ||||||
|  |                                 .put("y", i)); | ||||||
|  |                         } else { | ||||||
|  |                             winningCoordinates.add(mapper.createObjectNode() | ||||||
|  |                                 .put("x", i) | ||||||
|  |                                 .put("y", rnd.nextInt(worldHeight - 4 + 1))); | ||||||
|  |                         } | ||||||
|  |                     }); | ||||||
|  |                 objectNode.set(horizontal ? "winningRowCoordinates" : "winningColCoordinates", winningCoordinates); | ||||||
|  | 
 | ||||||
|  |                 RobotFamily[][] gameBoard = new RobotFamily[worldHeight][worldWidth]; | ||||||
|  |                 for (JsonNode node : winningCoordinates) { | ||||||
|  |                     for (int offset = 0; offset < 4; offset++) { | ||||||
|  |                         if (horizontal) { | ||||||
|  |                             gameBoard[node.get("y").intValue()][node.get("x").intValue() + offset] = currentPlayer; | ||||||
|  |                         } else { | ||||||
|  |                             gameBoard[node.get("y").intValue() + offset][node.get("x").intValue()] = currentPlayer; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 ArrayNode gameBoardNode = mapper.createArrayNode(); | ||||||
|  |                 Arrays.stream(gameBoard) | ||||||
|  |                     .map(gameBoardRow -> Arrays.stream(gameBoardRow) | ||||||
|  |                         .map(rf -> mapper.getNodeFactory().textNode(rf != null ? rf.getName() : null)) | ||||||
|  |                         .toList()) | ||||||
|  |                     .forEach(gameBoardRow -> gameBoardNode.add(mapper.createArrayNode().addAll(gameBoardRow))); | ||||||
|  |                 objectNode.set("gameBoard", gameBoardNode); | ||||||
|  | 
 | ||||||
|  |                 return objectNode; | ||||||
|  |             }, | ||||||
|  |             TEST_ITERATIONS, | ||||||
|  |             "FourWinsTestGameBoard" + (horizontal ? "Horizontal" : "Vertical") + "Win.generated.json" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										110
									
								
								src/graderPrivate/java/h02/TestUtils.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								src/graderPrivate/java/h02/TestUtils.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,110 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper; | ||||||
|  | import com.fasterxml.jackson.databind.node.ArrayNode; | ||||||
|  | import com.fasterxml.jackson.databind.node.ObjectNode; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.SpoonUtils; | ||||||
|  | import spoon.reflect.declaration.CtElement; | ||||||
|  | import spoon.reflect.declaration.CtMethod; | ||||||
|  | import spoon.reflect.declaration.CtParameter; | ||||||
|  | 
 | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.nio.file.Paths; | ||||||
|  | import java.util.Iterator; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Random; | ||||||
|  | import java.util.function.Consumer; | ||||||
|  | 
 | ||||||
|  | import static h02.TestConstants.RANDOM_SEED; | ||||||
|  | 
 | ||||||
|  | public abstract class TestUtils { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * A generator for JSON test data. | ||||||
|  |      */ | ||||||
|  |     public interface JsonGenerator { | ||||||
|  |         /** | ||||||
|  |          * Generates a JSON object node. | ||||||
|  |          * | ||||||
|  |          * @param mapper The object mapper to use. | ||||||
|  |          * @param index  The index of the object node. | ||||||
|  |          * @param rnd    The random number generator to use. | ||||||
|  |          * @return The generated JSON object node. | ||||||
|  |          */ | ||||||
|  |         ObjectNode generateJson(ObjectMapper mapper, int index, Random rnd); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Generates and saves JSON test data. | ||||||
|  |      * | ||||||
|  |      * @param generator The generator to use. | ||||||
|  |      * @param amount    The amount of test data to generate. | ||||||
|  |      * @param fileName  The file name to save the test data to. | ||||||
|  |      * @throws IOException If an I/O error occurs. | ||||||
|  |      */ | ||||||
|  |     public static void generateJsonTestData(final JsonGenerator generator, final int amount, final String fileName) throws IOException { | ||||||
|  |         final var seed = RANDOM_SEED; | ||||||
|  |         final var random = new java.util.Random(seed); | ||||||
|  |         final ObjectMapper mapper = new ObjectMapper(); | ||||||
|  |         final ArrayNode arrayNode = mapper.createArrayNode(); | ||||||
|  |         System.out.println("Generating test data with seed: " + seed); | ||||||
|  |         for (int i = 0; i < amount; i++) { | ||||||
|  |             arrayNode.add(generator.generateJson(mapper, i, random)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // convert `ObjectNode` to pretty-print JSON | ||||||
|  | //        System.out.println(mapper.writerWithDefaultPrettyPrinter().writeValueAsString(arrayNode)); | ||||||
|  | 
 | ||||||
|  |         final var path = Paths.get( | ||||||
|  |             "src", | ||||||
|  |             "graderPrivate", | ||||||
|  |             "resources", | ||||||
|  |             "h02", | ||||||
|  |             fileName | ||||||
|  |         ).toAbsolutePath(); | ||||||
|  |         System.out.printf("Saving to file: %s%n", path); | ||||||
|  |         final var file = path.toFile(); | ||||||
|  |         file.createNewFile(); | ||||||
|  |         mapper.writerWithDefaultPrettyPrinter().writeValue(file, arrayNode); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the Spoon representation of the given method. | ||||||
|  |      * | ||||||
|  |      * @param clazz      the method's owner | ||||||
|  |      * @param methodName the method name | ||||||
|  |      * @param paramTypes the method's formal parameter types, if any | ||||||
|  |      * @return the Spoon representation of the given method | ||||||
|  |      */ | ||||||
|  |     public static CtMethod<?> getCtMethod(Class<?> clazz, String methodName, Class<?>... paramTypes) { | ||||||
|  |         return SpoonUtils.getType(clazz.getName()) | ||||||
|  |             .getMethodsByName(methodName) | ||||||
|  |             .stream() | ||||||
|  |             .filter(ctMethod -> { | ||||||
|  |                 List<CtParameter<?>> parameters = ctMethod.getParameters(); | ||||||
|  |                 boolean result = parameters.size() == paramTypes.length; | ||||||
|  |                 for (int i = 0; result && i < parameters.size(); i++) { | ||||||
|  |                     result = parameters.get(i).getType().getQualifiedName().equals(paramTypes[i].getTypeName()); | ||||||
|  |                 } | ||||||
|  |                 return result; | ||||||
|  |             }) | ||||||
|  |             .findAny() | ||||||
|  |             .orElseThrow(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Applies the given consumer to the body and its descendants of the given method. | ||||||
|  |      * See also: {@link #getCtMethod(Class, String, Class[])}. | ||||||
|  |      * | ||||||
|  |      * @param clazz      the method's owner | ||||||
|  |      * @param methodName the method name | ||||||
|  |      * @param paramTypes the method's formal parameter types, if any | ||||||
|  |      * @param consumer   the consumer to apply | ||||||
|  |      */ | ||||||
|  |     public static void iterateMethodStatements(Class<?> clazz, String methodName, Class<?>[] paramTypes, Consumer<Iterator<CtElement>> consumer) { | ||||||
|  |         Iterator<CtElement> iterator = getCtMethod(clazz, methodName, paramTypes) | ||||||
|  |             .getBody() | ||||||
|  |             .descendantIterator(); | ||||||
|  |         consumer.accept(iterator); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										104
									
								
								src/graderPrivate/java/h02/TwoDimensionalArrayStuffTest.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/graderPrivate/java/h02/TwoDimensionalArrayStuffTest.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,104 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.junit.jupiter.api.Timeout; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.sourcegrade.jagr.api.rubric.TestForSubmission; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.annotation.SkipAfterFirstFailedTest; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.assertions.Context; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSet; | ||||||
|  | import org.tudalgo.algoutils.tutor.general.json.JsonParameterSetTest; | ||||||
|  | 
 | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  | import java.util.concurrent.atomic.AtomicInteger; | ||||||
|  | 
 | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertEquals; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.assertNotNull; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.callObject; | ||||||
|  | import static org.tudalgo.algoutils.tutor.general.assertions.Assertions2.contextBuilder; | ||||||
|  | 
 | ||||||
|  | @TestForSubmission | ||||||
|  | @Timeout( | ||||||
|  |     value = TestConstants.TEST_TIMEOUT_IN_SECONDS, | ||||||
|  |     unit = TimeUnit.SECONDS, | ||||||
|  |     threadMode = Timeout.ThreadMode.SEPARATE_THREAD | ||||||
|  | ) | ||||||
|  | @SkipAfterFirstFailedTest(TestConstants.SKIP_AFTER_FIRST_FAILED_TEST) | ||||||
|  | public class TwoDimensionalArrayStuffTest { | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("TwoDimensionalArrayStuffTestEmptySentence.json") | ||||||
|  |     public void testOccurrencesEmptyArray(JsonParameterSet params) { | ||||||
|  |         testOccurrences(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("TwoDimensionalArrayStuffTestSingleSentence.json") | ||||||
|  |     public void testOccurrencesSingleSentence(JsonParameterSet params) { | ||||||
|  |         testOccurrences(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("TwoDimensionalArrayStuffTestMultipleSentences.json") | ||||||
|  |     public void testOccurrencesMultipleSentences(JsonParameterSet params) { | ||||||
|  |         testOccurrences(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("TwoDimensionalArrayStuffTestIntegerMean.json") | ||||||
|  |     public void testMeanInteger(JsonParameterSet params) { | ||||||
|  |         testMean(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @JsonParameterSetTest("TwoDimensionalArrayStuffTestFloatMean.json") | ||||||
|  |     public void testMeanFloat(JsonParameterSet params) { | ||||||
|  |         testMean(params); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void testOccurrences(JsonParameterSet params) { | ||||||
|  |         List<String> sentences = params.get("sentences"); | ||||||
|  |         String[][] input = sentences.stream() | ||||||
|  |             .map(s -> s.split(" ")) | ||||||
|  |             .toArray(String[][]::new); | ||||||
|  |         String query = params.getString("query"); | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("input", input) | ||||||
|  |             .add("query", query) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         List<Integer> expectedList = params.get("expectedResult"); | ||||||
|  |         AtomicInteger counter = new AtomicInteger(0); | ||||||
|  |         int[] expected = new int[expectedList.size()]; | ||||||
|  |         expectedList.forEach(i -> expected[counter.getAndIncrement()] = i); | ||||||
|  |         int[] actual = callObject(() -> TwoDimensionalArrayStuff.occurrences(input, query), context, result -> | ||||||
|  |             "An exception occurred while invoking method occurrences"); | ||||||
|  | 
 | ||||||
|  |         assertNotNull(actual, context, result -> | ||||||
|  |             "Array returned by method occurrences is null"); | ||||||
|  |         assertEquals(input.length, actual.length, context, result -> | ||||||
|  |             "Array returned by method occurrences does not have correct length"); | ||||||
|  |         for (int i = 0; i < sentences.size(); i++) { | ||||||
|  |             final int finalI = i; | ||||||
|  |             assertEquals(expected[i], actual[i], context, result -> | ||||||
|  |                 "Array returned by method occurrences does not have correct value at index " + finalI); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static void testMean(JsonParameterSet params) { | ||||||
|  |         List<Integer> inputList = params.get("input"); | ||||||
|  |         AtomicInteger counter = new AtomicInteger(0); | ||||||
|  |         int[] input = new int[inputList.size()]; | ||||||
|  |         inputList.forEach(i -> input[counter.getAndIncrement()] = i); | ||||||
|  |         Context context = contextBuilder() | ||||||
|  |             .add("input", input) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         float expected = params.getFloat("expected"); | ||||||
|  |         float actual = callObject(() -> TwoDimensionalArrayStuff.mean(input), context, result -> | ||||||
|  |             "An exception occurred while invoking method mean"); | ||||||
|  | 
 | ||||||
|  |         // susceptible to rounding errors? | ||||||
|  |         assertEquals(expected, actual, context, result -> "Value returned by method mean is incorrect"); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								src/graderPrivate/resources/h02/.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/graderPrivate/resources/h02/.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | *.generated.json | ||||||
							
								
								
									
										175
									
								
								src/graderPrivate/resources/h02/FourWinsTestValidateInput.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								src/graderPrivate/resources/h02/FourWinsTestValidateInput.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,175 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "column": -1, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 7, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 3, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 5, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 2, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 6, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 4, | ||||||
|  |     "width": 7, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 2, | ||||||
|  |     "width": 4, | ||||||
|  |     "height": 4, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["SQUARE_BLUE", "SQUARE_RED", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["SQUARE_RED", "SQUARE_BLUE", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 3, | ||||||
|  |     "width": 4, | ||||||
|  |     "height": 5, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_RED", "SQUARE_BLUE"], | ||||||
|  |       ["SQUARE_RED", "SQUARE_BLUE", "SQUARE_RED", "SQUARE_RED"], | ||||||
|  |       ["SQUARE_BLUE", "SQUARE_RED", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 1, | ||||||
|  |     "width": 5, | ||||||
|  |     "height": 4, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_RED", "SQUARE_RED", "SQUARE_RED", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 0, | ||||||
|  |     "width": 3, | ||||||
|  |     "height": 3, | ||||||
|  |     "stones": [ | ||||||
|  |       ["SQUARE_RED", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_BLUE", "EMPTY", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": true | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 6, | ||||||
|  |     "width": 8, | ||||||
|  |     "height": 5, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_RED", "EMPTY"], | ||||||
|  |       ["EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "SQUARE_BLUE", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "column": 2, | ||||||
|  |     "width": 8, | ||||||
|  |     "height": 6, | ||||||
|  |     "stones": [ | ||||||
|  |       ["EMPTY", "EMPTY", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_BLUE", "SQUARE_RED", "SQUARE_BLUE", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_RED", "SQUARE_BLUE", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_BLUE", "SQUARE_RED", "SQUARE_BLUE", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_RED", "SQUARE_BLUE", "SQUARE_RED", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"], | ||||||
|  |       ["SQUARE_BLUE", "SQUARE_RED", "SQUARE_BLUE", "EMPTY", "EMPTY", "EMPTY", "EMPTY", "EMPTY"] | ||||||
|  |     ], | ||||||
|  |     "expected result": false | ||||||
|  |   } | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,7 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "sentences": [], | ||||||
|  |     "query": "", | ||||||
|  |     "expectedResult": [] | ||||||
|  |   } | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "input": [1, 1, 2, 3, 5, 8], | ||||||
|  |     "expected": 3.3333333 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], | ||||||
|  |     "expected": 5.5 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [1, 10, 100, 1000, 10000], | ||||||
|  |     "expected": 2222.2 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [11, 22, 33, 44], | ||||||
|  |     "expected": 27.5 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [5, 10, 15, 20, 25, 30], | ||||||
|  |     "expected": 17.5 | ||||||
|  |   } | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,26 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "input": [5, 5], | ||||||
|  |     "expected": 5 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [1, 2, 3, 4, 5], | ||||||
|  |     "expected": 3 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [5, 4, 3, 2, 1], | ||||||
|  |     "expected": 3 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [1, 2, 1, 3, 1, 4], | ||||||
|  |     "expected": 2 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [1, 11, 11, 1], | ||||||
|  |     "expected": 6 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "input": [2, 0, 2, 4], | ||||||
|  |     "expected": 2 | ||||||
|  |   } | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem ipsum dolor sit amet", | ||||||
|  |       "consectetur adipisici elit", | ||||||
|  |       "sed eiusmod tempor incidunt ut labore et dolore magna aliqua" | ||||||
|  |     ], | ||||||
|  |     "query": "lorem", | ||||||
|  |     "expectedResult": [1, 0, 0] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem ipsum dolor sit amet", | ||||||
|  |       "amet sit dolor ipsum lorem" | ||||||
|  |     ], | ||||||
|  |     "query": "amet", | ||||||
|  |     "expectedResult": [1, 1] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem lorem lorem", | ||||||
|  |       "lorem lorem", | ||||||
|  |       "lorem", | ||||||
|  |       "lorem lorem", | ||||||
|  |       "lorem lorem lorem" | ||||||
|  |     ], | ||||||
|  |     "query": "lorem", | ||||||
|  |     "expectedResult": [3, 2, 1, 2, 3] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "Wenn hinter fliegenden Fliegen Fliegen fliegen dann fliegen Fliegen Fliegen nach", | ||||||
|  |       "Fliegers Fritze fängt fliegende Fliegen fliegende Fliegen fängt Flieger Fritze", | ||||||
|  |       "Blaukraut bleibt Blaukraut und Brautkleid bleibt Brautkleid" | ||||||
|  |     ], | ||||||
|  |     "query": "Fliegen", | ||||||
|  |     "expectedResult": [4, 2, 0] | ||||||
|  |   } | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,30 @@ | ||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem ipsum dolor sit amet" | ||||||
|  |     ], | ||||||
|  |     "query": "lorem", | ||||||
|  |     "expectedResult": [1] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem ipsum dolor sit amet" | ||||||
|  |     ], | ||||||
|  |     "query": "amet", | ||||||
|  |     "expectedResult": [1] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "lorem lorem lorem" | ||||||
|  |     ], | ||||||
|  |     "query": "lorem", | ||||||
|  |     "expectedResult": [3] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "sentences": [ | ||||||
|  |       "ipsum ipsum ipsum" | ||||||
|  |     ], | ||||||
|  |     "query": "lorem", | ||||||
|  |     "expectedResult": [0] | ||||||
|  |   } | ||||||
|  | ] | ||||||
							
								
								
									
										388
									
								
								src/main/java/h02/FourWins.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										388
									
								
								src/main/java/h02/FourWins.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,388 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import fopbot.Direction; | ||||||
|  | import fopbot.Robot; | ||||||
|  | import fopbot.RobotFamily; | ||||||
|  | import fopbot.World; | ||||||
|  | import h02.template.InputHandler; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.DoNotTouch; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentImplementationRequired; | ||||||
|  | 
 | ||||||
|  | import java.util.Optional; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The {@link FourWins} class represents the main class of the FourWins game. | ||||||
|  |  */ | ||||||
|  | public class FourWins { | ||||||
|  |     private final InputHandler inputHandler = new InputHandler(this); | ||||||
|  |     /** | ||||||
|  |      * The width of the game board. | ||||||
|  |      */ | ||||||
|  |     private final int width; | ||||||
|  |     /** | ||||||
|  |      * The height of the game board. | ||||||
|  |      */ | ||||||
|  |     private final int height; | ||||||
|  |     /** | ||||||
|  |      * Indicates whether the game has finished. | ||||||
|  |      */ | ||||||
|  |     @SuppressWarnings({"FieldMayBeFinal", "FieldCanBeLocal"}) | ||||||
|  |     private boolean finished = false; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates a new {@link FourWins} instance with the given width and height. | ||||||
|  |      * | ||||||
|  |      * @param width  the width of the game board | ||||||
|  |      * @param height the height of the game board | ||||||
|  |      */ | ||||||
|  |     FourWins(final int width, final int height) { | ||||||
|  |         this.width = width; | ||||||
|  |         this.height = height; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Starts the game by setting up the world and executing the game loop. | ||||||
|  |      */ | ||||||
|  |     void startGame() { | ||||||
|  |         setupWorld(); | ||||||
|  |         gameLoop(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets up the world and installs the {@link InputHandler}. | ||||||
|  |      */ | ||||||
|  |     void setupWorld() { | ||||||
|  |         World.setSize(width, height); | ||||||
|  |         World.setDelay(10); | ||||||
|  |         World.setVisible(true); | ||||||
|  |         inputHandler.install(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Validates if a given column index is within the bounds of the game board and not fully occupied. | ||||||
|  |      * | ||||||
|  |      * @param column The column index to validate. | ||||||
|  |      * @param stones 2D array representing the game board, where each cell contains a RobotFamily color indicating the | ||||||
|  |      *               player that has placed a stone in that position. | ||||||
|  |      * @return true if the column is within bounds and has at least one unoccupied cell; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.1") | ||||||
|  |     public static boolean validateInput(final int column, final RobotFamily[][] stones) { | ||||||
|  |         return column >= 0 && column < World.getWidth() && stones[World.getHeight() - 1][column] == null; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Calculates the next unoccupied row index in the specified column. This row index is the next destination for a | ||||||
|  |      * falling stone. | ||||||
|  |      * | ||||||
|  |      * @param column The column index where the stone is to be dropped. | ||||||
|  |      * @param stones 2D array representing the game board, where each cell contains a RobotFamily object indicating the | ||||||
|  |      *               player that has placed a stone in that position. | ||||||
|  |      * @return Index of the next unoccupied row index in the specified column. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.2") | ||||||
|  |     public static int getDestinationRow(final int column, final RobotFamily[][] stones) { | ||||||
|  |         for (int row = 0; row < stones.length; row++) { | ||||||
|  |             if (stones[row][column] == null) { | ||||||
|  |                 return row; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Drops a stone into the specified column of the game board, simulating a falling animation. This method gets the | ||||||
|  |      * destination row for the stone in the specified column with the `getDestinationRow` method. It creates a new Robot | ||||||
|  |      * instance to represent the stone with the currentPlayer's RobotFamily in the given column and the destination row. | ||||||
|  |      * After that it simulates the stone's fall by decrementing its position until it reaches the destination row. Once | ||||||
|  |      * the  stone reaches its destination, the method updates the stones array (a 2D array of RobotFamily colors) to | ||||||
|  |      * mark the slot as occupied by the currentPlayer. | ||||||
|  |      * | ||||||
|  |      * @param column        The column index where the stone is to be dropped. | ||||||
|  |      * @param stones        2D array representing the game board, where each cell contains a RobotFamily object | ||||||
|  |      *                      indicating the player that has placed a stone in that position. | ||||||
|  |      * @param currentPlayer The RobotFamily object representing the current player dropping the stone. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.2") | ||||||
|  |     public static void dropStone(final int column, final RobotFamily[][] stones, final RobotFamily currentPlayer) { | ||||||
|  |         // spawn stone | ||||||
|  |         final Robot stone = new Robot(column, World.getHeight() - 1, Direction.DOWN, 0, currentPlayer); | ||||||
|  | 
 | ||||||
|  |         // let stone fall | ||||||
|  |         final int row = getDestinationRow(column, stones); | ||||||
|  |         for (int currentRow = World.getHeight() - 1; currentRow > row; currentRow--) { | ||||||
|  |             stone.move(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // turn stone up | ||||||
|  |         stone.turnLeft(); | ||||||
|  |         stone.turnLeft(); | ||||||
|  | 
 | ||||||
|  |         // set slot as occupied | ||||||
|  |         stones[row][column] = currentPlayer; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if the current player has won by any condition. The conditions can be a horizontal, vertical, diagonal, or | ||||||
|  |      * anti-diagonal line of at least four stones. | ||||||
|  |      * | ||||||
|  |      * @param stones        2D array representing the game board, where each cell contains a RobotFamily color | ||||||
|  |      *                      indicating the player that has placed a stone in that position. | ||||||
|  |      * @param currentPlayer The RobotFamily color representing the current player to check for a win. | ||||||
|  |      * @return true if the current player has formed a horizontal line of at least four stones; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.3") | ||||||
|  |     public static boolean testWinConditions(final RobotFamily[][] stones, final RobotFamily currentPlayer) { | ||||||
|  |         return testWinVertical(stones, currentPlayer) | ||||||
|  |             || testWinHorizontal(stones, currentPlayer) | ||||||
|  |             || testWinDiagonal(stones, currentPlayer); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if the current player has won by forming a horizontal line of at least consecutive four stones. | ||||||
|  |      * | ||||||
|  |      * @param stones        2D array representing the game board, where each cell contains a RobotFamily color | ||||||
|  |      *                      indicating the player that has placed a stone in that position. | ||||||
|  |      * @param currentPlayer The RobotFamily color representing the current player to check for a win. | ||||||
|  |      * @return true if the current player has formed a horizontal line of at least four stones; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.3") | ||||||
|  |     public static boolean testWinHorizontal(final RobotFamily[][] stones, final RobotFamily currentPlayer) { | ||||||
|  |         for (int row = 0; row < World.getHeight(); row++) { | ||||||
|  |             int stoneCount = 0; | ||||||
|  |             for (int column = 0; column < World.getWidth(); column++) { | ||||||
|  |                 stoneCount = stones[row][column] == currentPlayer ? stoneCount + 1 : 0; | ||||||
|  |                 if (stoneCount >= 4) { | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if the current player has won by forming a vertical line of at least consecutive four stones. | ||||||
|  |      * | ||||||
|  |      * @param stones        2D array representing the game board, where each cell contains a RobotFamily color | ||||||
|  |      *                      indicating the player that has placed a stone in that position. | ||||||
|  |      * @param currentPlayer The RobotFamily color representing the current player to check for a win. | ||||||
|  |      * @return true if the current player has formed a vertical line of at least four stones; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.3") | ||||||
|  |     public static boolean testWinVertical(final RobotFamily[][] stones, final RobotFamily currentPlayer) { | ||||||
|  |         for (int column = 0; column < World.getWidth(); column++) { | ||||||
|  |             int stoneCount = 0; | ||||||
|  |             for (int row = 0; row < World.getHeight(); row++) { | ||||||
|  |                 stoneCount = stones[row][column] == currentPlayer ? stoneCount + 1 : 0; | ||||||
|  |                 if (stoneCount >= 4) { | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if the current player has won by forming a diagonal line of at least consecutive four stones. | ||||||
|  |      * | ||||||
|  |      * @param stones        2D array representing the game board, where each cell contains a RobotFamily color | ||||||
|  |      *                      indicating the player that has placed a stone in that position. | ||||||
|  |      * @param currentPlayer The RobotFamily color representing the current player to check for a win. | ||||||
|  |      * @return true if the current player has formed a diagonal line of at least four stones; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public static boolean testWinDiagonal(final RobotFamily[][] stones, final RobotFamily currentPlayer) { | ||||||
|  |         @SuppressWarnings("CheckStyle") final int MAX_STONES = 4; | ||||||
|  | 
 | ||||||
|  |         @SuppressWarnings("CheckStyle") final int WIDTH = World.getWidth(); | ||||||
|  |         @SuppressWarnings("CheckStyle") final int HEIGHT = World.getHeight(); | ||||||
|  |         int[] direction = new int[]{1, 1}; | ||||||
|  | 
 | ||||||
|  |         // for every field | ||||||
|  |         for (int y = 0; y < HEIGHT; y++) { | ||||||
|  |             for (int x = 0; x < WIDTH; x++) { | ||||||
|  | 
 | ||||||
|  |                 // for every direction | ||||||
|  |                 for (int nthDirection = 0; nthDirection < 4; nthDirection++) { | ||||||
|  |                     final int[] pos = {x, y}; | ||||||
|  | 
 | ||||||
|  |                     // test for consecutive coins | ||||||
|  |                     int coinCount = 0; // start counting at 0 | ||||||
|  |                     while (pos[0] >= 0 && pos[0] < WIDTH | ||||||
|  |                         && pos[1] >= 0 && pos[1] < HEIGHT | ||||||
|  |                         && stones[pos[1]][pos[0]] == currentPlayer) { | ||||||
|  |                         coinCount++; // count every stone that has currentPlayer's color | ||||||
|  |                         if (coinCount >= MAX_STONES) { | ||||||
|  |                             return true; | ||||||
|  |                         } | ||||||
|  |                         pos[0] += direction[0]; | ||||||
|  |                         pos[1] += direction[1]; | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     direction = new int[]{direction[1], -direction[0]}; // next direction (rotate by 90 deg) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Switches the player for each turn. If the current player is SQUARE_BLUE, SQUARE_RED is returned as the next | ||||||
|  |      * player. If the current player is SQUARE_RED, SQUARE_BLUE is returned as the next player. | ||||||
|  |      * | ||||||
|  |      * @param currentPlayer The player color of the current player. | ||||||
|  |      * @return The player color of the next player. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.4") | ||||||
|  |     public static RobotFamily nextPlayer(final RobotFamily currentPlayer) { | ||||||
|  |         return currentPlayer == RobotFamily.SQUARE_BLUE ? RobotFamily.SQUARE_RED : RobotFamily.SQUARE_BLUE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Displays a Message in the console and on the game board indicating the game is drawn. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.4") | ||||||
|  |     public void writeDrawMessage() { | ||||||
|  |         inputHandler.displayDrawStatus(); | ||||||
|  | 
 | ||||||
|  |         // student implementation here: | ||||||
|  |         System.out.println("No valid columns found. Hence, game ends with a draw."); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Displays a Message in the console and on the game board indicating the game is won by a player. | ||||||
|  |      * | ||||||
|  |      * @param winner {@link RobotFamily} of the winning player | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.4") | ||||||
|  |     public void writeWinnerMessage(final RobotFamily winner) { | ||||||
|  |         inputHandler.displayWinnerStatus(winner); | ||||||
|  | 
 | ||||||
|  |         // student implementation here: | ||||||
|  |         System.out.println("Player " + winner + " wins the game!"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Displays the winner of the game by printing the winning color in the console and filling the whole field with | ||||||
|  |      * Robots of the winning color. | ||||||
|  |      * | ||||||
|  |      * @param winner The RobotFamily color of the winner. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.4") | ||||||
|  |     public static void colorFieldBackground(final RobotFamily winner) { | ||||||
|  |         for (int x = 0; x < World.getWidth(); x++) { | ||||||
|  |             for (int y = 0; y < World.getHeight(); y++) { | ||||||
|  |                 setFieldColor(x, y, winner); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Executes the main game loop, handling player turns, stone drops, and win condition checks. This method | ||||||
|  |      * initializes the game board as a 2D array of RobotFamily colors, representing the slots that can be filled with | ||||||
|  |      * players' stones. It starts with a predefined currentPlayer and continues in a loop until a win condition is met. | ||||||
|  |      * Each iteration of the loop waits for player input to select a column to drop a stone into, switches the current | ||||||
|  |      * player, drops the stone in the selected column, and checks for win conditions. If a win condition is met, the | ||||||
|  |      * loop ends, and the winner is displayed. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.2.4") | ||||||
|  |     void gameLoop() { | ||||||
|  |         final RobotFamily[][] stones = new RobotFamily[World.getHeight()][World.getWidth()]; | ||||||
|  |         RobotFamily currentPlayer = RobotFamily.SQUARE_BLUE; | ||||||
|  | 
 | ||||||
|  |         boolean draw = false; | ||||||
|  |         finished = false; | ||||||
|  | 
 | ||||||
|  |         while (!finished) { | ||||||
|  |             // student implementation here: | ||||||
|  |             currentPlayer = nextPlayer(currentPlayer); | ||||||
|  | 
 | ||||||
|  |             // wait for click in column (DO NOT TOUCH) | ||||||
|  |             finished = draw = isGameBoardFull(stones); | ||||||
|  |             if (draw) { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             final int column = inputHandler.getNextInput(currentPlayer, stones); | ||||||
|  | 
 | ||||||
|  |             // student implementation here: | ||||||
|  |             dropStone(column, stones, currentPlayer); | ||||||
|  |             finished = testWinConditions(stones, currentPlayer); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // displaying either draw or winner (DO NOT TOUCH) | ||||||
|  |         if (draw) { | ||||||
|  |             writeDrawMessage(); | ||||||
|  |             colorFieldBackground(getDrawnRobotFamily()); | ||||||
|  |         } else { | ||||||
|  |             writeWinnerMessage(currentPlayer); | ||||||
|  |             colorFieldBackground(currentPlayer); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Executes the main game loop, handling player turns, stone drops, and win condition checks. Sets the background | ||||||
|  |      * color of a field at the specified coordinates. The color is derived from the {@link RobotFamily} SQUARE_BLUE or | ||||||
|  |      * SQUARE_RED. | ||||||
|  |      * | ||||||
|  |      * @param x     the x coordinate of the field | ||||||
|  |      * @param y     the y coordinate of the field | ||||||
|  |      * @param color the {@link RobotFamily} corresponding to the field color to set | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public static void setFieldColor(final int x, final int y, final RobotFamily color) { | ||||||
|  |         World.getGlobalWorld().setFieldColor(x, y, color.getColor()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the {@link RobotFamily} which represents a drawn game. | ||||||
|  |      * | ||||||
|  |      * @return the {@link RobotFamily} which represents a drawn game. | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     @SuppressWarnings("UnstableApiUsage") | ||||||
|  |     protected static RobotFamily getDrawnRobotFamily() { | ||||||
|  |         return Optional.ofNullable(World.getGlobalWorld().getGuiPanel()).filter(guiPanel -> !guiPanel.isDarkMode()).map(guiPanel -> RobotFamily.SQUARE_ORANGE).orElse(RobotFamily.SQUARE_YELLOW); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if all columns of the game board are fully occupied. | ||||||
|  |      * | ||||||
|  |      * @param stones 2D array representing the game board, where each cell contains a RobotFamily | ||||||
|  |      * @return true if all columns of the game board are fully occupied; false otherwise. | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public static boolean isGameBoardFull(final RobotFamily[][] stones) { | ||||||
|  |         for (int x = 0; x < World.getWidth(); x++) { | ||||||
|  |             if (FourWins.validateInput(x, stones)) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns this instance's {@link InputHandler}. | ||||||
|  |      * | ||||||
|  |      * @return the input handler | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public InputHandler getInputHandler() { | ||||||
|  |         return inputHandler; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns {@code true} when the game is finished, {@code false} otherwise. | ||||||
|  |      * | ||||||
|  |      * @return whether the game is finished. | ||||||
|  |      */ | ||||||
|  |     public boolean isFinished() { | ||||||
|  |         return finished; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										218
									
								
								src/main/java/h02/Main.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										218
									
								
								src/main/java/h02/Main.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,218 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import fopbot.RobotFamily; | ||||||
|  | import fopbot.World; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.SolutionOnly; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentImplementationRequired; | ||||||
|  | 
 | ||||||
|  | import static org.tudalgo.algoutils.student.io.PropertyUtils.getIntProperty; | ||||||
|  | import static org.tudalgo.algoutils.student.test.StudentTestUtils.printTestResults; | ||||||
|  | import static org.tudalgo.algoutils.student.test.StudentTestUtils.testEquals; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Main entry point in executing the program. | ||||||
|  |  */ | ||||||
|  | public class Main { | ||||||
|  |     /** | ||||||
|  |      * Main entry point in executing the program. | ||||||
|  |      * | ||||||
|  |      * @param args program arguments, currently ignored | ||||||
|  |      */ | ||||||
|  |     public static void main(final String[] args) { | ||||||
|  |         // H1 | ||||||
|  |         sanityChecksH211(); | ||||||
|  |         sanityChecksH212(); | ||||||
|  |         printTestResults(); | ||||||
|  | 
 | ||||||
|  |         // H2 | ||||||
|  |         sanityChecksH22(); | ||||||
|  |         printTestResults(); | ||||||
|  | 
 | ||||||
|  |         // starting game (comment out if you just want to run the tests) | ||||||
|  |         final var propFile = "h02.properties"; | ||||||
|  |         new FourWins( | ||||||
|  |             getIntProperty(propFile, "FW_WORLD_WIDTH"), | ||||||
|  |             getIntProperty(propFile, "FW_WORLD_HEIGHT") | ||||||
|  |         ).startGame(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Perform sanity checks for exercise H2.1.1. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.3") | ||||||
|  |     public static void sanityChecksH211() { | ||||||
|  |         // push test | ||||||
|  |         final int[] newArray = OneDimensionalArrayStuff.push(new int[]{0, 1}, 2); | ||||||
|  |         final int[] expectedArray = {0, 1, 2}; | ||||||
|  |         testEquals(expectedArray.length, newArray.length); | ||||||
|  |         for (int i = 0; i < newArray.length; i++) { | ||||||
|  |             testEquals(expectedArray[i], newArray[i]); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // calculateNextFibonacci test | ||||||
|  |         int[] fibonacciArray = {0, 1}; | ||||||
|  |         for (int i = 0; i < 20; i++) { | ||||||
|  |             fibonacciArray = OneDimensionalArrayStuff.calculateNextFibonacci(fibonacciArray); | ||||||
|  |         } | ||||||
|  |         testEquals(22, fibonacciArray.length); | ||||||
|  |         testEquals(0, fibonacciArray[0]); | ||||||
|  |         testEquals(1, fibonacciArray[1]); | ||||||
|  |         for (int i = 2; i < fibonacciArray.length; i++) { | ||||||
|  |             testEquals(fibonacciArray[i - 1] + fibonacciArray[i - 2], fibonacciArray[i]); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // fibonacci test | ||||||
|  |         final int[] reference = {0, 1, 1, 2, 3, 5, 8, 13, 21, 34}; | ||||||
|  |         for (int i = 0; i < 10; i++) { | ||||||
|  |             testEquals(reference[i], OneDimensionalArrayStuff.fibonacci(i)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Perform sanity checks for exercise H2.1.2. | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.3") | ||||||
|  |     public static void sanityChecksH212() { | ||||||
|  |         // predefined simple test | ||||||
|  |         final String[][] simpleTest = new String[][]{ | ||||||
|  |             "a b c d e f".split(" "), | ||||||
|  |             "a b c d e f".split(" "), | ||||||
|  |             "a b c d e f".split(" "), | ||||||
|  |         }; | ||||||
|  |         // predefined complex test | ||||||
|  |         final String[][] complexTest = new String[][]{ | ||||||
|  |             "a a b b c c".split(" "), | ||||||
|  |             "a b c d e f".split(" "), | ||||||
|  |             "a a a b b b c c c".split(" "), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // student implementation here: | ||||||
|  | 
 | ||||||
|  |         sanityChecksH212Helper( | ||||||
|  |             simpleTest, | ||||||
|  |             "b", | ||||||
|  |             new int[]{1, 1, 1}, | ||||||
|  |             1 | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         sanityChecksH212Helper( | ||||||
|  |             complexTest, | ||||||
|  |             "b", | ||||||
|  |             new int[]{2, 1, 3}, | ||||||
|  |             2 | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Helper method for sanity checks for exercise H2.1.2. | ||||||
|  |      * | ||||||
|  |      * @param input   the input array | ||||||
|  |      * @param query   the query string | ||||||
|  |      * @param refOcc  the reference occurrences | ||||||
|  |      * @param refMean the reference mean | ||||||
|  |      */ | ||||||
|  |     @SolutionOnly | ||||||
|  |     public static void sanityChecksH212Helper( | ||||||
|  |         final String[][] input, | ||||||
|  |         final String query, | ||||||
|  |         final int[] refOcc, | ||||||
|  |         final float refMean | ||||||
|  |     ) { | ||||||
|  |         final int[] occ = TwoDimensionalArrayStuff.occurrences(input, query); | ||||||
|  |         testEquals(refOcc.length, occ.length); | ||||||
|  |         for (int i = 0; i < occ.length; i++) { | ||||||
|  |             testEquals(refOcc[i], occ[i]); | ||||||
|  |         } | ||||||
|  |         testEquals(refMean, TwoDimensionalArrayStuff.meanOccurrencesPerLine(input, query)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Perform sanity checks for exercise H2.2 | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.4") | ||||||
|  |     public static void sanityChecksH22() { | ||||||
|  |         // setting world size | ||||||
|  |         World.setSize(4, 5); | ||||||
|  | 
 | ||||||
|  |         // predefined stones1 array | ||||||
|  |         final RobotFamily[][] stones1 = { | ||||||
|  |             {null, RobotFamily.SQUARE_BLUE, null, RobotFamily.SQUARE_RED}, | ||||||
|  |             {null, null, null, RobotFamily.SQUARE_BLUE}, | ||||||
|  |             {null, null, null, RobotFamily.SQUARE_RED}, | ||||||
|  |             {null, null, null, RobotFamily.SQUARE_BLUE}, | ||||||
|  |             {null, null, null, RobotFamily.SQUARE_RED}, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         // predefined stones2 array | ||||||
|  |         final RobotFamily[][] stones2 = { | ||||||
|  |             {RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE}, | ||||||
|  |             {RobotFamily.SQUARE_RED, RobotFamily.SQUARE_RED, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_RED}, | ||||||
|  |             {RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_RED, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE}, | ||||||
|  |             {RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_RED, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_RED}, | ||||||
|  |             {RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_BLUE, RobotFamily.SQUARE_RED}, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // student implementation here: | ||||||
|  | 
 | ||||||
|  |         // H2.2.1 validateInput | ||||||
|  |         final boolean isInCol1 = FourWins.validateInput(1, stones1); | ||||||
|  |         final boolean isInCol3 = FourWins.validateInput(3, stones1); | ||||||
|  | 
 | ||||||
|  |         testEquals(true, isInCol1); | ||||||
|  |         testEquals(false, isInCol3); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.2 getDestinationRow | ||||||
|  |         final int rowCol1 = FourWins.getDestinationRow(1, stones1); | ||||||
|  |         final int rowCol3 = FourWins.getDestinationRow(3, stones1); | ||||||
|  | 
 | ||||||
|  |         testEquals(1, rowCol1); | ||||||
|  |         testEquals(-1, rowCol3); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.2 dropStone | ||||||
|  |         FourWins.dropStone(1, stones1, RobotFamily.SQUARE_RED); | ||||||
|  |         // System.out.println(Arrays.deepToString(stones1)); | ||||||
|  |         // System.out.println(stones1); | ||||||
|  |         testEquals(RobotFamily.SQUARE_RED, stones1[1][1]); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.3 testWinHorizontal | ||||||
|  |         final boolean winRowBlue = FourWins.testWinHorizontal(stones2, RobotFamily.SQUARE_BLUE); | ||||||
|  |         final boolean winRowRed = FourWins.testWinHorizontal(stones2, RobotFamily.SQUARE_RED); | ||||||
|  | 
 | ||||||
|  |         testEquals(true, winRowBlue); | ||||||
|  |         testEquals(false, winRowRed); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.3 testWinVertical | ||||||
|  |         final boolean winColStones2 = FourWins.testWinVertical(stones2, RobotFamily.SQUARE_BLUE); | ||||||
|  |         final boolean winColStones1 = FourWins.testWinVertical(stones1, RobotFamily.SQUARE_BLUE); | ||||||
|  | 
 | ||||||
|  |         testEquals(true, winColStones2); | ||||||
|  |         testEquals(false, winColStones1); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.3 testWinConditions | ||||||
|  |         final boolean winStones2 = FourWins.testWinConditions(stones2, RobotFamily.SQUARE_BLUE); | ||||||
|  |         final boolean winStones1 = FourWins.testWinConditions(stones1, RobotFamily.SQUARE_BLUE); | ||||||
|  | 
 | ||||||
|  |         testEquals(true, winStones2); | ||||||
|  |         testEquals(false, winStones1); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.4 switchPlayer | ||||||
|  |         final RobotFamily nextPlayer1 = FourWins.nextPlayer(RobotFamily.SQUARE_BLUE); | ||||||
|  |         final RobotFamily nextPlayer2 = FourWins.nextPlayer(RobotFamily.SQUARE_RED); | ||||||
|  | 
 | ||||||
|  |         testEquals(RobotFamily.SQUARE_RED, nextPlayer1); | ||||||
|  |         testEquals(RobotFamily.SQUARE_BLUE, nextPlayer2); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         // H2.2.4 colorFieldBackground, writeDrawMessage, writeWinnerMessage, gameLoop | ||||||
|  |         // Test by playing | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										64
									
								
								src/main/java/h02/OneDimensionalArrayStuff.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								src/main/java/h02/OneDimensionalArrayStuff.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentImplementationRequired; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This class serves as a container for the methods that are to be implemented by the students for exercise H2.1.1. | ||||||
|  |  */ | ||||||
|  | public class OneDimensionalArrayStuff { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Prevent instantiation of this utility class. | ||||||
|  |      */ | ||||||
|  |     private OneDimensionalArrayStuff() { | ||||||
|  |         throw new IllegalStateException("This class is not meant to be instantiated."); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns a new array that is a copy of the input array with the given value appended at the end. | ||||||
|  |      * | ||||||
|  |      * @param array the input array | ||||||
|  |      * @param value the value to append | ||||||
|  |      * @return a new array that is a copy of the input array with the given value appended at the end | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.1.1") | ||||||
|  |     public static int[] push(final int[] array, final int value) { | ||||||
|  |         final int[] newArray = new int[array.length + 1]; | ||||||
|  |         //noinspection ManualArrayCopy | ||||||
|  |         for (int i = 0; i < array.length; i++) { | ||||||
|  |             newArray[i] = array[i]; | ||||||
|  |         } | ||||||
|  |         newArray[array.length] = value; | ||||||
|  |         return newArray; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Calculates the next Fibonacci number based on the given array and returns a new array with the next Fibonacci | ||||||
|  |      * number appended at the end. | ||||||
|  |      * | ||||||
|  |      * @param array the input array containing the last two Fibonacci numbers up to the current point | ||||||
|  |      * @return a new array with the next Fibonacci number appended at the end | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.1.1") | ||||||
|  |     public static int[] calculateNextFibonacci(final int[] array) { | ||||||
|  |         return push(array, array[array.length - 1] + array[array.length - 2]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the n-th Fibonacci number. | ||||||
|  |      * | ||||||
|  |      * @param n the index of the Fibonacci number to return | ||||||
|  |      * @return the n-th Fibonacci number | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.1.1") | ||||||
|  |     public static int fibonacci(final int n) { | ||||||
|  |         if (n < 2) { | ||||||
|  |             return n; // base case (n=0 or n=1) | ||||||
|  |         } | ||||||
|  |         int[] array = {0, 1}; | ||||||
|  |         for (int i = 2; i <= n; i++) { | ||||||
|  |             array = calculateNextFibonacci(array); | ||||||
|  |         } | ||||||
|  |         return array[array.length - 1]; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										89
									
								
								src/main/java/h02/TwoDimensionalArrayStuff.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								src/main/java/h02/TwoDimensionalArrayStuff.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,89 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.tudalgo.algoutils.student.annotation.DoNotTouch; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.StudentImplementationRequired; | ||||||
|  | 
 | ||||||
|  | import java.util.Arrays; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This class serves as a container for the methods that are to be implemented by the students for exercise H2.1.2. | ||||||
|  |  */ | ||||||
|  | public class TwoDimensionalArrayStuff { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Prevent instantiation of this utility class. | ||||||
|  |      */ | ||||||
|  |     private TwoDimensionalArrayStuff() { | ||||||
|  |         throw new IllegalStateException("This class is not meant to be instantiated."); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns an array containing the number of occurrences of the query {@link String} in each line of the input array. | ||||||
|  |      * | ||||||
|  |      * @param input the input array | ||||||
|  |      * @param query the query {@link String} | ||||||
|  |      * @return an array containing the number of occurrences of the query {@link String} in each line of the input array | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.1.2") | ||||||
|  |     public static int[] occurrences(final String[][] input, final String query) { | ||||||
|  |         final int[] result = new int[input.length]; | ||||||
|  |         for (int row = 0; row < input.length; row++) { | ||||||
|  |             for (int col = 0; col < input[row].length; col++) { | ||||||
|  |                 if (input[row][col].equals(query)) { | ||||||
|  |                     result[row]++; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return result; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the mean of the input array. | ||||||
|  |      * | ||||||
|  |      * @param input the input array | ||||||
|  |      * @return the mean of the input array | ||||||
|  |      */ | ||||||
|  |     @StudentImplementationRequired("H2.1.2") | ||||||
|  |     public static float mean(final int[] input) { | ||||||
|  |         int sum = 0; | ||||||
|  |         for (final int j : input) { | ||||||
|  |             sum += j; | ||||||
|  |         } | ||||||
|  |         return (float) sum / input.length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the mean number of occurrences of the query {@link String} in each line of the input array. | ||||||
|  |      * | ||||||
|  |      * @param input the input array | ||||||
|  |      * @param query the query {@link String} | ||||||
|  |      * @return the mean number of occurrences of the query {@link String} in each line of the input array | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public static float meanOccurrencesPerLine(final String[][] input, final String query) { | ||||||
|  |         return mean(occurrences(input, query)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Overload that splits the input string by lines and spaces, then calls regular meanOccurrencesPerLine. | ||||||
|  |      * | ||||||
|  |      * @param input the input string to split by lines and spaces | ||||||
|  |      * @param query the query {@link String} | ||||||
|  |      * @return the mean number of occurrences of the query {@link String} in each line of the input array | ||||||
|  |      */ | ||||||
|  |     @DoNotTouch | ||||||
|  |     public static float meanOccurrencesPerLine(final String input, final String query) { | ||||||
|  |         // filter out unwanted symbols | ||||||
|  |         final String filteredInput = input.replaceAll("[^\\w\\s]", ""); | ||||||
|  |         // split by lines | ||||||
|  |         final var processedInput = Arrays.stream(filteredInput.split("(\\r\\n|\\r|\\n)")) | ||||||
|  |             // split by spaces | ||||||
|  |             .map(line -> line.split("\\s")) | ||||||
|  |             // collect to 2D array | ||||||
|  |             .toArray(String[][]::new); | ||||||
|  |         /// uncomment the following line to log processed input | ||||||
|  |         // System.out.printf("Processed input: %s%n", Arrays.deepToString(processedInput)); | ||||||
|  |         // call regular meanOccurrencesPerLine | ||||||
|  |         return meanOccurrencesPerLine(processedInput, query); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										201
									
								
								src/main/java/h02/template/InputHandler.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								src/main/java/h02/template/InputHandler.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,201 @@ | ||||||
|  | package h02.template; | ||||||
|  | 
 | ||||||
|  | import fopbot.RobotFamily; | ||||||
|  | import fopbot.World; | ||||||
|  | import h02.FourWins; | ||||||
|  | import org.tudalgo.algoutils.student.annotation.DoNotTouch; | ||||||
|  | 
 | ||||||
|  | import javax.swing.*; | ||||||
|  | import javax.swing.border.EmptyBorder; | ||||||
|  | import java.awt.*; | ||||||
|  | import java.awt.event.ComponentAdapter; | ||||||
|  | import java.awt.event.ComponentEvent; | ||||||
|  | import java.beans.PropertyChangeEvent; | ||||||
|  | import java.util.concurrent.BlockingDeque; | ||||||
|  | import java.util.concurrent.LinkedBlockingDeque; | ||||||
|  | import java.util.concurrent.atomic.AtomicBoolean; | ||||||
|  | import java.util.function.Supplier; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The {@link InputHandler} handles the input of the users. | ||||||
|  |  */ | ||||||
|  | @DoNotTouch | ||||||
|  | public class InputHandler { | ||||||
|  |     /** | ||||||
|  |      * The input queue. | ||||||
|  |      */ | ||||||
|  |     private final BlockingDeque<Integer> inputQueue = new LinkedBlockingDeque<>(); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * The {@link FourWins} instance. | ||||||
|  |      */ | ||||||
|  |     private final FourWins fourWins; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Whether the row select mode is active. | ||||||
|  |      */ | ||||||
|  |     private final AtomicBoolean rowSelectMode = new AtomicBoolean(false); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * The status label. | ||||||
|  |      */ | ||||||
|  |     private final JLabel statusLabel = new JLabel("", SwingConstants.CENTER); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates a new {@link InputHandler} instance. | ||||||
|  |      * | ||||||
|  |      * @param fourWins the {@link FourWins} instance | ||||||
|  |      */ | ||||||
|  |     public InputHandler(final FourWins fourWins) { | ||||||
|  |         this.fourWins = fourWins; | ||||||
|  |         final int padding = 4; // Padding in pixels | ||||||
|  |         statusLabel.setBorder(new EmptyBorder(padding, padding, padding, padding)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the color of the given column to the given color. | ||||||
|  |      * | ||||||
|  |      * @param column        the column to set the color of | ||||||
|  |      * @param colorSupplier the color to set | ||||||
|  |      */ | ||||||
|  |     private void setColumnColor(final int column, final Supplier<Color> colorSupplier) { | ||||||
|  |         for (int i = 0; i < World.getHeight(); i++) { | ||||||
|  |             final int finalI = i; | ||||||
|  |             SwingUtilities.invokeLater(() -> World.getGlobalWorld().getField(column, finalI).setFieldColor(colorSupplier)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Executes the given action only if the game is running. | ||||||
|  |      * | ||||||
|  |      * @param action the action to execute | ||||||
|  |      */ | ||||||
|  |     private void whenGameIsRunning(final Runnable action) { | ||||||
|  |         if (!fourWins.isFinished()) { | ||||||
|  |             action.run(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Installs the input handler to the fopbot world. | ||||||
|  |      */ | ||||||
|  |     @SuppressWarnings("UnstableApiUsage") | ||||||
|  |     public void install() { | ||||||
|  |         final var guiPanel = World.getGlobalWorld().getGuiPanel(); | ||||||
|  |         final var guiFrame = World.getGlobalWorld().getGuiFrame(); | ||||||
|  |         World.getGlobalWorld().getInputHandler().addFieldClickListener( | ||||||
|  |             e -> whenGameIsRunning(() -> addInput(e.getField().getX())) | ||||||
|  |         ); | ||||||
|  |         World.getGlobalWorld().getInputHandler().addFieldHoverListener(e -> whenGameIsRunning(() -> { | ||||||
|  |             // deselect last hovered field, if any | ||||||
|  |             if (e.getPreviousField() != null) { | ||||||
|  |                 setColumnColor(e.getPreviousField().getX(), () -> null); | ||||||
|  |             } | ||||||
|  |             if (rowSelectMode.get()) { | ||||||
|  |                 // select current hovered field | ||||||
|  |                 if (e.getField() != null) { | ||||||
|  |                     setColumnColor( | ||||||
|  |                         e.getField().getX(), | ||||||
|  |                         () -> guiPanel.isDarkMode() | ||||||
|  |                             ? Color.yellow | ||||||
|  |                             : Color.orange | ||||||
|  |                     ); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         })); | ||||||
|  |         statusLabel.setFont(statusLabel.getFont().deriveFont(guiPanel.scale(20.0f))); | ||||||
|  |         guiFrame.add(statusLabel, BorderLayout.NORTH); | ||||||
|  |         guiFrame.pack(); | ||||||
|  |         guiPanel.addDarkModeChangeListener(this::onDarkModeChange); | ||||||
|  |         guiPanel.addComponentListener(new ComponentAdapter() { | ||||||
|  |             @Override | ||||||
|  |             public void componentResized(final ComponentEvent e) { | ||||||
|  |                 statusLabel.setFont( | ||||||
|  |                     statusLabel.getFont().deriveFont( | ||||||
|  |                         Math.max(20f, 0.04f * Math.min(guiPanel.getWidth(), guiPanel.getHeight())) | ||||||
|  |                     ) | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         // trigger dark mode change to set the correct color | ||||||
|  |         guiPanel.setDarkMode(World.getGlobalWorld().getGuiPanel().isDarkMode()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Called when the dark mode changes. | ||||||
|  |      * | ||||||
|  |      * @param e the property change event | ||||||
|  |      */ | ||||||
|  |     @SuppressWarnings("UnstableApiUsage") | ||||||
|  |     public void onDarkModeChange(final PropertyChangeEvent e) { | ||||||
|  |         final var darkMode = (boolean) e.getNewValue(); | ||||||
|  |         statusLabel.setForeground(darkMode ? Color.white : Color.black); | ||||||
|  |         World.getGlobalWorld().getGuiFrame().getContentPane().setBackground(darkMode ? Color.black : Color.white); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Adds an input to the input queue. When {@link #getNextInput(RobotFamily, RobotFamily[][])} is called, the program | ||||||
|  |      * will wait until this method is called. | ||||||
|  |      * | ||||||
|  |      * @param input the input to add | ||||||
|  |      */ | ||||||
|  |     public void addInput(final int input) { | ||||||
|  |         inputQueue.add(input); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the next input from the input queue. If the input is invalid, the user will be prompted to enter a new | ||||||
|  |      * input. The program will halt until a valid input is entered. | ||||||
|  |      * | ||||||
|  |      * @param currentPlayer the current player | ||||||
|  |      * @param stones        the current state of the game board | ||||||
|  |      * @return the next input from the input queue | ||||||
|  |      */ | ||||||
|  |     public int getNextInput(final RobotFamily currentPlayer, final RobotFamily[][] stones) { | ||||||
|  |         rowSelectMode.set(true); | ||||||
|  |         statusLabel.setText( | ||||||
|  |             "<html>Click on a column to insert a disc.<br>Current Player: %s</html>".formatted(currentPlayer.getName()) | ||||||
|  |         ); | ||||||
|  |         try { | ||||||
|  |             final int input = inputQueue.take(); | ||||||
|  |             System.out.println("Received column input: " + input); | ||||||
|  |             if (!FourWins.validateInput(input, stones)) { | ||||||
|  |                 System.out.println("Invalid column input, please try again."); | ||||||
|  |                 return getNextInput(currentPlayer, stones); | ||||||
|  |             } | ||||||
|  |             rowSelectMode.set(false); | ||||||
|  |             return input; | ||||||
|  |         } catch (final InterruptedException e) { | ||||||
|  |             rowSelectMode.set(false); | ||||||
|  |             throw new RuntimeException(e); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets a status message, saying that the game has ended in a draw. | ||||||
|  |      */ | ||||||
|  |     public void displayDrawStatus() { | ||||||
|  |         statusLabel.setText("<html>No valid columns found. <br>Hence, game ends with a draw.</html>"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets a status message, saying that the game has ended with a winner. | ||||||
|  |      * | ||||||
|  |      * @param winner the winner of the game | ||||||
|  |      */ | ||||||
|  |     public void displayWinnerStatus(final RobotFamily winner) { | ||||||
|  |         statusLabel.setText("<html>Player %s has won the game!</html>".formatted(winner.getName())); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the {@link #statusLabel} of this {@link InputHandler}. | ||||||
|  |      * | ||||||
|  |      * <p>Use the {@link JLabel#getText()} method to get the current text of the label, and the | ||||||
|  |      * {@link JLabel#setText(String)} method to update the text. | ||||||
|  |      * | ||||||
|  |      * @return the {@link #statusLabel} of this {@link InputHandler} | ||||||
|  |      */ | ||||||
|  |     public JLabel getStatusLabel() { | ||||||
|  |         return statusLabel; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								src/main/resources/h02.properties
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/main/resources/h02.properties
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | FW_WORLD_WIDTH = 7 | ||||||
|  | FW_WORLD_HEIGHT = 6 | ||||||
							
								
								
									
										16
									
								
								src/test/java/h02/ExampleJUnitTest.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/test/java/h02/ExampleJUnitTest.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,16 @@ | ||||||
|  | package h02; | ||||||
|  | 
 | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  | 
 | ||||||
|  | import static org.junit.jupiter.api.Assertions.*; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * An example JUnit test class. | ||||||
|  |  */ | ||||||
|  | public class ExampleJUnitTest { | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testAddition() { | ||||||
|  |         assertEquals(2, 1 + 1); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								version
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								version
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | 0.1.0-SNAPSHOT | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue