Compare commits
	
		
			48 Commits
		
	
	
		
			ba2aa13999
			...
			v20250128
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| cd54fdf345 | |||
| cced4f04b2 | |||
| 2bdcdcb1dd | |||
| 71fefd760e | |||
| 2461f5b417 | |||
| b83d6bdde4 | |||
| a314e945ed | |||
| 2735074515 | |||
| 647476f6d8 | |||
| 515034a64e | |||
| a7c5845b7a | |||
| 6dd5d2f232 | |||
| 6b0e55e3f9 | |||
| 719dd43307 | |||
| 6241c0b720 | |||
| 3d4729bdb6 | |||
| 7084a62962 | |||
| 8d2cca657b | |||
| 56ebe616ab | |||
| fc555fa581 | |||
| 0e3aa42b78 | |||
| c72bc6b0e5 | |||
| c73c88679a | |||
| 81b44e1702 | |||
| c8f897cf8d | |||
| c0525c2bc8 | |||
| 3f6ba6e9dc | |||
| 74f1fb63e1 | |||
| a7feb7101a | |||
| 28ecc74d7a | |||
| 6d6db0ed28 | |||
| 53953cf47c | |||
| acf6ad1bb9 | |||
| 26b78420a2 | |||
| 582e268d56 | |||
| 5d02036f9e | |||
| 6d2ac3675b | |||
| 6cd8deed23 | |||
| c02aea543c | |||
| c9b05e7026 | |||
| 81edc8962d | |||
| ef2db7184e | |||
| 0c31f7953d | |||
| 8ec56bc3d8 | |||
| 5a9f22edb7 | |||
| 458e5ac101 | |||
| e7800660e9 | |||
| 57ac3a1d16 | 
							
								
								
									
										19
									
								
								.gitea/workflows/demo.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.gitea/workflows/demo.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | name: Gitea Actions Demo | ||||||
|  | run-name: ${{ gitea.actor }} is testing out Gitea Actions 🚀 | ||||||
|  | on: [push] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   Explore-Gitea-Actions: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - run: echo "🎉 The job was automatically triggered by a ${{ gitea.event_name }} event." | ||||||
|  |       - run: echo "🐧 This job is now running on a ${{ runner.os }} server hosted by Gitea!" | ||||||
|  |       - run: echo "🔎 The name of your branch is ${{ gitea.ref }} and your repository is ${{ gitea.repository }}." | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |       - run: echo "💡 The ${{ gitea.repository }} repository has been cloned to the runner." | ||||||
|  |       - run: echo "🖥️ The workflow is now ready to test your code on the runner." | ||||||
|  |       - name: List files in the repository | ||||||
|  |         run: | | ||||||
|  |           ls ${{ gitea.workspace }} | ||||||
|  |       - run: echo "🍏 This job's status is ${{ job.status }}." | ||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | builds | ||||||
| @@ -2,9 +2,17 @@ | |||||||
| # See https://pre-commit.com/hooks.html for more hooks | # See https://pre-commit.com/hooks.html for more hooks | ||||||
| repos: | repos: | ||||||
| -   repo: https://github.com/pre-commit/pre-commit-hooks | -   repo: https://github.com/pre-commit/pre-commit-hooks | ||||||
|     rev: v3.2.0 |     rev: v5.0.0 | ||||||
|     hooks: |     hooks: | ||||||
|     -   id: trailing-whitespace |     -   id: trailing-whitespace | ||||||
|     -   id: end-of-file-fixer |     -   id: end-of-file-fixer | ||||||
|     -   id: check-yaml |     -   id: check-yaml | ||||||
|     -   id: check-added-large-files |     -   id: check-added-large-files | ||||||
|  | -   repo: https://github.com/dnephin/pre-commit-golang | ||||||
|  |     rev: v0.5.1 | ||||||
|  |     hooks: | ||||||
|  |       - id: go-fmt | ||||||
|  |       - id: go-imports | ||||||
|  |       - id: no-go-testing | ||||||
|  |       - id: golangci-lint | ||||||
|  |       - id: go-unit-tests | ||||||
|   | |||||||
							
								
								
									
										141
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | |||||||
|  | # Sudoku Funpark | ||||||
|  | Creating worlds most inefficient Sudoku solver, by trying every option, without any smart approach. | ||||||
|  |  | ||||||
|  | _(This was a learning project to get a better grasp of Golang. But more important; It was fun to do!.)_ | ||||||
|  |  | ||||||
|  | ## Goals | ||||||
|  | * Create the most inefficient Sudoku solver imaginable | ||||||
|  | * Be a learning experience for the Go programming language | ||||||
|  |  | ||||||
|  | I wrote [a blog post](https://blog.ligthert.net/posts/exploration-fun-and-process-cycles-of-sudoku/) about this. | ||||||
|  |  | ||||||
|  | ## Features | ||||||
|  | * Worlds least efficient Sudoku solver | ||||||
|  | * Ability to assign a number of CPU cores to this task | ||||||
|  | * Split workloads among several computers | ||||||
|  |  | ||||||
|  | ## Usage | ||||||
|  | To use the sudoku solver, run the binary with all the `-row` parameters available. Use other parameters to tune our output or CPU usage. | ||||||
|  | ``` | ||||||
|  | Usage of ./sudoku-funpark: | ||||||
|  |   -numcpu int | ||||||
|  |     	Number of CPU cores to assign to this task. (default 12) | ||||||
|  |   -output string | ||||||
|  |     	Type of output. 'human' for human readable. 'flat' for flat as stored in memory output. 'json' for JSON output. (default "human") | ||||||
|  |   -part int | ||||||
|  |     	Process part x in n parts. Cannot be lower than 1, or higher than specified in split. (default 1) | ||||||
|  |   -print string | ||||||
|  |     	'short': normal output;'long': normal output with timestamps; 'silent': Only print the results; (default "short") | ||||||
|  |   -row1 string | ||||||
|  |     	1st row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row2 string | ||||||
|  |     	2nd row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row3 string | ||||||
|  |     	4rd row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row4 string | ||||||
|  |     	4th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row5 string | ||||||
|  |     	5th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row6 string | ||||||
|  |     	6th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row7 string | ||||||
|  |     	7th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row8 string | ||||||
|  |     	8th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -row9 string | ||||||
|  |     	9th row of the sudoku puzzle. (default "000000000") | ||||||
|  |   -split int | ||||||
|  |     	Split the tasks in n parts. This depends on the availability of the first row. (default 1) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Instead of using the 3x3 blocks with 3x3 digits, it uses horizontal rows from top to bottom. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Example | ||||||
|  | To see the solver in action, run the tool with the following parameters. | ||||||
|  |  | ||||||
|  | For a short running (~14 seconds) example: | ||||||
|  | > $ ./sudoku-funpark -row1 769104802 -row2 154800060 -row3 832700154 -row4 600900328 -row5 045328670 -row6 328670945 -row7 597410280 -row8 006283090 -row9 200590006 | ||||||
|  |  | ||||||
|  | For a long running (~1 hours 15 minutes) example: | ||||||
|  | > $ ./sudoku-funpark -row1 769104802 -row2 154800060 -row3 002700150 -row4 600900308 -row5 045328670 -row6 328670945 -row7 597410280 -row8 006283090 -row9 200590006 | ||||||
|  |  | ||||||
|  | The outpot (of the short running parameters) will look something like this: | ||||||
|  | ``` | ||||||
|  | ./sudoku-funpark -row1 769104802 -row2 154800060 -row3 832700154 -row4 600900328 -row5 045328670 -row6 328670945 -row7 597410280 -row8 006283090 -row9 200590006  | ||||||
|  | Loading blocks... Done! (38.957376ms) | ||||||
|  | Populating blocks... Done! (92.087174ms) | ||||||
|  | Number of (potential) solutions: 26542080 | ||||||
|  | Validating solutions | ||||||
|  | Processing: 8% (2131893/26542080); Rate: 2131884/sec for 1.000028115s; Time left (est.): 11s | ||||||
|  | Processing: 16% (4292163/26542080); Rate: 2160219/sec for 1.000087826s; Time left (est.): 10s | ||||||
|  | Processing: 24% (6438334/26542080); Rate: 2146157/sec for 1.000017364s; Time left (est.): 9s | ||||||
|  | Processing: 32% (8529362/26542080); Rate: 2090965/sec for 1.000367121s; Time left (est.): 8s | ||||||
|  | Processing: 40% (10737065/26542080); Rate: 2207530/sec for 1.000072427s; Time left (est.): 7s | ||||||
|  | Processing: 48% (12958905/26542080); Rate: 2221755/sec for 1.000003187s; Time left (est.): 6s | ||||||
|  | Processing: 57% (15163877/26542080); Rate: 2204929/sec for 1.000002717s; Time left (est.): 5s | ||||||
|  | Processing: 65% (17254760/26542080); Rate: 2090742/sec for 1.00008452s; Time left (est.): 4s | ||||||
|  | Processing: 73% (19513142/26542080); Rate: 2258348/sec for 1.000071076s; Time left (est.): 3s | ||||||
|  | Processing: 82% (21795213/26542080); Rate: 2282028/sec for 1.000076024s; Time left (est.): 2s | ||||||
|  | Processing: 90% (24048891/26542080); Rate: 2253645/sec for 1.000146957s; Time left (est.): 1s | ||||||
|  | Processing: 98% (26226252/26542080); Rate: 2177215/sec for 1.000129955s; Time left (est.): 0s | ||||||
|  | Processing: 100% (26542080/26542080); Rate: 315792/sec for 1.000105149s; Time left (est.): 0s | ||||||
|  | Validated solutions (13.001683829s) | ||||||
|  |  | ||||||
|  | Solution #1: | ||||||
|  | ╔═══════════╗ | ||||||
|  | ║769│154│832╢ | ||||||
|  | ║154│832│769╢ | ||||||
|  | ║832│769│154╢ | ||||||
|  | ╟───┼───┼───╢ | ||||||
|  | ║671│945│328╢ | ||||||
|  | ║945│328│671╢ | ||||||
|  | ║328│671│945╢ | ||||||
|  | ╟───┼───┼───╢ | ||||||
|  | ║597│416│283╢ | ||||||
|  | ║416│283│597╢ | ||||||
|  | ║283│597│416╢ | ||||||
|  | ╚═══════════╝ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Caveats | ||||||
|  | While this may very well solve all possible Sudoku puzzles (including the one [designed against brute force algorithms](https://en.wikipedia.org/wiki/Sudoku_solving_algorithms)), the blanks in the puzzle, the harder it is, the more possible solutions there are, the more solutions it needs to parse, the longer it takes. As this is a computational heavy program, the more CPU you throw against it the faster it will solve issues. | ||||||
|  |  | ||||||
|  | ## Generating your own blocks | ||||||
|  | To generate your own blocks, you could use the following code: | ||||||
|  | ```go | ||||||
|  | func (solver *Solver) generate_blocks() []int { | ||||||
|  |  | ||||||
|  | 	var blocks []int | ||||||
|  | 	decvals := [9]int{49, 50, 51, 52, 53, 54, 55, 56, 57} | ||||||
|  |  | ||||||
|  | 	for counter := 123456789; counter <= 987654321; counter++ { | ||||||
|  |  | ||||||
|  | 		// Convert number to string ([]byte) | ||||||
|  | 		digits := strconv.Itoa(counter) | ||||||
|  |  | ||||||
|  | 		// Check if every number is only represented only once | ||||||
|  | 		var valid bool | ||||||
|  | 		valid = true | ||||||
|  | 		for decval := range decvals { | ||||||
|  | 			var count int | ||||||
|  | 			for digit := range digits { | ||||||
|  | 				if digits[digit] == byte(decvals[decval]) { | ||||||
|  | 					count = count + 1 | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if count != 1 { | ||||||
|  | 				valid = false | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if valid { | ||||||
|  | 			blocks = append(blocks, counter) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return blocks | ||||||
|  |  | ||||||
|  | } | ||||||
|  | ``` | ||||||
							
								
								
									
										29
									
								
								Taskfile.yml
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								Taskfile.yml
									
									
									
									
									
								
							| @@ -2,16 +2,43 @@ | |||||||
|  |  | ||||||
| version: '3' | version: '3' | ||||||
|  |  | ||||||
|  | vars: | ||||||
|  |   APP: sudoku-funpark | ||||||
|  |   BUILD_DIR: builds | ||||||
|  |  | ||||||
| tasks: | tasks: | ||||||
|   default: |   default: | ||||||
|     cmds: |     cmds: | ||||||
|       - go run . |       - go run . --help | ||||||
|  |     silent: true | ||||||
|  |   run_short: | ||||||
|  |     cmds: | ||||||
|  |       - go run . -row1 769104802 -row2 154800060 -row3 832700154 -row4 600900328 -row5 045328670 -row6 328670945 -row7 597410280 -row8 006283090 -row9 200590006 | ||||||
|  |     silent: true | ||||||
|  |   run_long: | ||||||
|  |     cmds: | ||||||
|  |       - go run . -row1 769104802 -row2 154800060 -row3 002700150 -row4 600900308 -row5 045328670 -row6 328670945 -row7 597410280 -row8 006283090 -row9 200590006 | ||||||
|     silent: true |     silent: true | ||||||
|   precommit: |   precommit: | ||||||
|     cmds: |     cmds: | ||||||
|  |       - pre-commit autoupdate | ||||||
|       - pre-commit run --all |       - pre-commit run --all | ||||||
|     silent: true |     silent: true | ||||||
|   lint: |   lint: | ||||||
|     cmds: |     cmds: | ||||||
|       - golangci-lint run |       - golangci-lint run | ||||||
|     silent: true |     silent: true | ||||||
|  |   build: | ||||||
|  |     cmds: | ||||||
|  |       - mkdir -p {{.BUILD_DIR}} | ||||||
|  |       - rm {{.BUILD_DIR}}/* || true | ||||||
|  |       - go tool dist list | grep -v android | grep -v ios | grep -v wasip1 | awk -F '/' '{printf "echo Compiling %s/%s; env CGO_ENABLED=1 GOOS=%s GOARCH=%s go build -o {{.BUILD_DIR}}/{{.APP}}.%s-%s\n",$1,$2,$1,$2,$1,$2 }' | sh | ||||||
|  |       - for i in `ls {{.BUILD_DIR}}/*windows*`; do mv -v $i $i.exe; done | ||||||
|  |   gource: | ||||||
|  |     cmds: | ||||||
|  |       - gource --auto-skip-seconds 1 --key -r 60 | ||||||
|  |     silent: true | ||||||
|  |   godoc: | ||||||
|  |     cmds: | ||||||
|  |       - godoc -http=:6060 | ||||||
|  |     silent: true | ||||||
|   | |||||||
							
								
								
									
										41
									
								
								controller/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								controller/types.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | package controller | ||||||
|  |  | ||||||
|  | import "gitea.ligthert.net/golang/sudoku-funpark/outputter" | ||||||
|  |  | ||||||
|  | // Simple interface to store values shared amongst packages. | ||||||
|  | type Controller struct { | ||||||
|  | 	// All possible blocks/rows available | ||||||
|  | 	Blocks []string | ||||||
|  | 	// 1st row of the Sudoku puzzle. | ||||||
|  | 	Row1 string | ||||||
|  | 	// 2nd row of the Sudoku puzzle. | ||||||
|  | 	Row2 string | ||||||
|  | 	// 3rd row of the Sudoku puzzle. | ||||||
|  | 	Row3 string | ||||||
|  | 	// 4th row of the Sudoku puzzle. | ||||||
|  | 	Row4 string | ||||||
|  | 	// 5th row of the Sudoku puzzle. | ||||||
|  | 	Row5 string | ||||||
|  | 	// 6th row of the Sudoku puzzle. | ||||||
|  | 	Row6 string | ||||||
|  | 	// 7th row of the Sudoku puzzle. | ||||||
|  | 	Row7 string | ||||||
|  | 	// 8th row of the Sudoku puzzle. | ||||||
|  | 	Row8 string | ||||||
|  | 	// 9th row of the Sudoku puzzle. | ||||||
|  | 	Row9 string | ||||||
|  | 	// Slice with all found solutions. | ||||||
|  | 	Solutions [][]string | ||||||
|  | 	// Number of CPUs Go routines are allowed to use. | ||||||
|  | 	NumCPUs int | ||||||
|  | 	// Number of parts the search should be split into. | ||||||
|  | 	Split int | ||||||
|  | 	// Which part of the search should the solver focus on. | ||||||
|  | 	Part int | ||||||
|  | 	// Type of output requested | ||||||
|  | 	Output string | ||||||
|  | 	// Select printing method | ||||||
|  | 	Print string | ||||||
|  | 	// Outputter package | ||||||
|  | 	Outputter *outputter.Outputter | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								export/Export.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								export/Export.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | package export | ||||||
|  |  | ||||||
|  | func (export *Export) Export() (render string) { | ||||||
|  | 	// Print the valid solutions | ||||||
|  | 	switch export.Controller.Output { | ||||||
|  | 	case "human": | ||||||
|  | 		render = export.renderHumanReadable() | ||||||
|  | 	case "flat": | ||||||
|  | 		render = export.renderFlat() | ||||||
|  | 	case "json": | ||||||
|  | 		render = export.renderJSON() | ||||||
|  | 	} | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								export/renderFlat.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								export/renderFlat.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | package export | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strconv" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Render output as stored internally. | ||||||
|  | func (export *Export) renderFlat() (render string) { | ||||||
|  | 	for solutionIndex, solution := range export.Controller.Solutions { | ||||||
|  | 		render += fmt.Sprintln("\nSolution #" + strconv.Itoa(solutionIndex+1) + ":") | ||||||
|  | 		render += fmt.Sprintln(solution) | ||||||
|  | 	} | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								export/renderHumanReadable.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								export/renderHumanReadable.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | package export | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strconv" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Render solutions in a human friendly format. | ||||||
|  | func (export *Export) renderHumanReadable() (render string) { | ||||||
|  | 	for solutionIndex, solution := range export.Controller.Solutions { | ||||||
|  | 		render += fmt.Sprintln("\nSolution #" + strconv.Itoa(solutionIndex+1) + ":") | ||||||
|  | 		render += fmt.Sprintln("╔═══════════╗") | ||||||
|  | 		for rowIndex, row := range solution { | ||||||
|  | 			if rowIndex == 3 || rowIndex == 6 { | ||||||
|  | 				render += fmt.Sprintln("╟───┼───┼───╢") | ||||||
|  | 			} | ||||||
|  | 			render += fmt.Sprintln("║" + row[0:3] + "│" + row[3:6] + "│" + row[6:9] + "╢") | ||||||
|  | 		} | ||||||
|  | 		render += fmt.Sprintln("╚═══════════╝") | ||||||
|  | 	} | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								export/renderJSON.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								export/renderJSON.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | package export | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Render JSON output. | ||||||
|  | func (export *Export) renderJSON() (render string) { | ||||||
|  | 	type solution_type map[string]any | ||||||
|  | 	solutions := make([]solution_type, 0) | ||||||
|  |  | ||||||
|  | 	for solutionIndex, solution := range export.Controller.Solutions { | ||||||
|  | 		solutionMap := map[string]any{ | ||||||
|  | 			"order": solutionIndex, | ||||||
|  | 			"row1":  solution[0], | ||||||
|  | 			"row2":  solution[1], | ||||||
|  | 			"row3":  solution[2], | ||||||
|  | 			"row4":  solution[3], | ||||||
|  | 			"row5":  solution[4], | ||||||
|  | 			"row6":  solution[5], | ||||||
|  | 			"row7":  solution[6], | ||||||
|  | 			"row8":  solution[7], | ||||||
|  | 			"row9":  solution[8], | ||||||
|  | 		} | ||||||
|  | 		solutions = append(solutions, solutionMap) | ||||||
|  | 	} | ||||||
|  | 	renderBytes, err := json.Marshal(solutions) | ||||||
|  | 	if err != nil { | ||||||
|  | 		export.Controller.Outputter.Println("ERROR: json.Marshal error:", err) | ||||||
|  | 		export.Controller.Outputter.Println("Printing solution as-is:", solutions) | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | 	render = string(renderBytes) | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								export/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								export/types.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | package export | ||||||
|  |  | ||||||
|  | import "gitea.ligthert.net/golang/sudoku-funpark/controller" | ||||||
|  |  | ||||||
|  | // A simple inteface to export found solutions in different formats. | ||||||
|  | type Export struct { | ||||||
|  | 	Controller *controller.Controller | ||||||
|  | } | ||||||
							
								
								
									
										93
									
								
								flags/ParseFlags.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								flags/ParseFlags.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | package flags | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"flag" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"runtime" | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Parse command-line parameters, test input, store them in the Controller. | ||||||
|  | func (flags *Flags) ParseFlags() { | ||||||
|  |  | ||||||
|  | 	// Define parameters | ||||||
|  | 	flag.StringVar(&flags.Controller.Row1, "row1", "000000000", "1st row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row2, "row2", "000000000", "2nd row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row3, "row3", "000000000", "4rd row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row4, "row4", "000000000", "4th row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row5, "row5", "000000000", "5th row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row6, "row6", "000000000", "6th row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row7, "row7", "000000000", "7th row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row8, "row8", "000000000", "8th row of the sudoku puzzle.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Row9, "row9", "000000000", "9th row of the sudoku puzzle.") | ||||||
|  | 	flag.IntVar(&flags.Controller.NumCPUs, "numcpu", runtime.NumCPU(), "Number of CPU cores to assign to this task.") | ||||||
|  | 	flag.IntVar(&flags.Controller.Split, "split", 1, "Split the tasks in n parts. This depends on the availability of the first row.") | ||||||
|  | 	flag.IntVar(&flags.Controller.Part, "part", 1, "Process part x in n parts. Cannot be lower than 1, or higher than specified in split.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Output, "output", "human", "Type of output. 'human' for human readable. 'flat' for flat as stored in memory output. 'json' for JSON output.") | ||||||
|  | 	flag.StringVar(&flags.Controller.Print, "print", "short", "'short': normal output;'long': normal output with timestamps; 'silent': Only print the results;") | ||||||
|  |  | ||||||
|  | 	// Parse the flags | ||||||
|  | 	flag.Parse() | ||||||
|  |  | ||||||
|  | 	// Process any changes to the CPU usage. | ||||||
|  | 	if flags.Controller.NumCPUs <= 0 { | ||||||
|  | 		fmt.Printf("ERROR: Number of CPU cores must be 1 or higher.\n\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if flags.Controller.NumCPUs != runtime.NumCPU() { | ||||||
|  | 		runtime.GOMAXPROCS(flags.Controller.NumCPUs) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Process rows | ||||||
|  | 	if flags.Controller.Row1 == "000000000" || flags.Controller.Row2 == "000000000" || flags.Controller.Row3 == "000000000" || flags.Controller.Row4 == "000000000" || flags.Controller.Row5 == "000000000" || flags.Controller.Row6 == "000000000" || flags.Controller.Row7 == "000000000" || flags.Controller.Row8 == "000000000" || flags.Controller.Row9 == "000000000" { | ||||||
|  | 		fmt.Printf("ERROR: All parameters must be entered.\n\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Validate the row (never trust user input) | ||||||
|  | 	flags.validateRow("row1", flags.Controller.Row1) | ||||||
|  | 	flags.validateRow("row2", flags.Controller.Row2) | ||||||
|  | 	flags.validateRow("row3", flags.Controller.Row3) | ||||||
|  | 	flags.validateRow("row4", flags.Controller.Row4) | ||||||
|  | 	flags.validateRow("row5", flags.Controller.Row5) | ||||||
|  | 	flags.validateRow("row6", flags.Controller.Row6) | ||||||
|  | 	flags.validateRow("row7", flags.Controller.Row7) | ||||||
|  | 	flags.validateRow("row8", flags.Controller.Row8) | ||||||
|  | 	flags.validateRow("row9", flags.Controller.Row9) | ||||||
|  |  | ||||||
|  | 	// Process workload splitting | ||||||
|  | 	// Ensure split and part are 1 or higher | ||||||
|  | 	if flags.Controller.Split <= 0 || flags.Controller.Part <= 0 { | ||||||
|  | 		fmt.Printf("ERROR: '-split' and '-part' need to be 1 or higher.\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Ensure part is between 1 and split | ||||||
|  | 	if flags.Controller.Part > flags.Controller.Split { | ||||||
|  | 		fmt.Printf("ERROR: '-part' cannot be bigger than `-split`.\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Process output selection | ||||||
|  | 	flags.Controller.Output = strings.ToLower(flags.Controller.Output) | ||||||
|  | 	if flags.Controller.Output != "human" && flags.Controller.Output != "flat" && flags.Controller.Output != "json" { | ||||||
|  | 		fmt.Printf("ERROR: Invalid output, can only be 'human' or 'flat' or 'json'.\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Process print selection | ||||||
|  | 	flags.Controller.Print = strings.ToLower(flags.Controller.Print) | ||||||
|  | 	if flags.Controller.Print != "short" && flags.Controller.Print != "long" && flags.Controller.Print != "silent" { | ||||||
|  | 		fmt.Printf("ERROR: Invalid Print, can only be 'short' or 'long' or 'silent'.\n") | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								flags/printUsage.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								flags/printUsage.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | package flags | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"flag" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Print help information for the end-user | ||||||
|  | func (flags *Flags) printUsage() { | ||||||
|  | 	fmt.Fprintf(flag.CommandLine.Output(), "Usage of %s:\n", os.Args[0]) | ||||||
|  | 	fmt.Fprintf(flag.CommandLine.Output(), "\nPut every row of a Sudoku puzzle as paramters.\nUse '0' for what is currently blank in the puzzle you wish to solve.\n\n") | ||||||
|  | 	fmt.Fprintf(flag.CommandLine.Output(), "Example: %s -row1 ... -row2 ... -row3 ... (etc)\n\n", os.Args[0]) | ||||||
|  | 	flag.PrintDefaults() | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								flags/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								flags/types.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | package flags | ||||||
|  |  | ||||||
|  | import "gitea.ligthert.net/golang/sudoku-funpark/controller" | ||||||
|  |  | ||||||
|  | // Interface to parse command-line parameters | ||||||
|  | type Flags struct { | ||||||
|  | 	Controller *controller.Controller | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								flags/validChar.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								flags/validChar.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | package flags | ||||||
|  |  | ||||||
|  | // Validate if the char provided is 0-9 | ||||||
|  | func (flags *Flags) validChar(char rune) (valid bool) { | ||||||
|  | 	decvals := [10]int{48, 49, 50, 51, 52, 53, 54, 55, 56, 57} | ||||||
|  |  | ||||||
|  | 	for _, value := range decvals { | ||||||
|  | 		if char == rune(value) { | ||||||
|  | 			valid = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										54
									
								
								flags/validateRow.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								flags/validateRow.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | |||||||
|  | package flags | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Validate if a row is properly set. | ||||||
|  | // This check for: | ||||||
|  | // - Correct length | ||||||
|  | // - Correct numbers | ||||||
|  | // - Numbers only present once | ||||||
|  | func (flags *Flags) validateRow(name string, row string) { | ||||||
|  |  | ||||||
|  | 	var found bool | ||||||
|  | 	var double bool | ||||||
|  | 	count := make(map[rune]int) | ||||||
|  |  | ||||||
|  | 	// 1. Make sure the row is 9 in length | ||||||
|  | 	if len(row) != 9 { | ||||||
|  | 		fmt.Printf("ERROR: Invalid length of %s (%s), must be 9 numbers\n\n", name, row) | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 2. Ensure all digits are numbers | ||||||
|  | 	for _, value := range row { | ||||||
|  | 		found = flags.validChar(value) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !found { | ||||||
|  | 		fmt.Printf("ERROR: Invalid character of %s (%s), must be 9 numbers\n\n", name, row) | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 3. Ensure all digits (except zero) are there only once | ||||||
|  | 	for _, digits := range row { | ||||||
|  | 		count[digits] = count[digits] + 1 | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for key, value := range count { | ||||||
|  | 		if value > 1 && key != 48 { | ||||||
|  | 			double = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if double { | ||||||
|  | 		fmt.Printf("ERROR: Double character of %s (%s), numbers between 1 and 9 may only be entered once\n\n", name, row) | ||||||
|  | 		flags.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										52
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								main.go
									
									
									
									
									
								
							| @@ -1,7 +1,55 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import "gitea.ligthert.net/golang/sudoku-funpark/solver" | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"runtime" | ||||||
|  | 	"strconv" | ||||||
|  |  | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/controller" | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/export" | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/flags" | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/outputter" | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/solver" | ||||||
|  | ) | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	solver.Run() | 	// Instantiate the interfaces | ||||||
|  | 	controller := controller.Controller{} | ||||||
|  | 	outp := outputter.Outputter{} | ||||||
|  | 	export := export.Export{Controller: &controller} | ||||||
|  | 	flags := flags.Flags{Controller: &controller} | ||||||
|  | 	solver := solver.Solver{Controller: &controller, Outp: &outp} | ||||||
|  |  | ||||||
|  | 	// Parse and handle flags | ||||||
|  | 	flags.ParseFlags() | ||||||
|  |  | ||||||
|  | 	// Tell outp what kind of output is expected. | ||||||
|  | 	outp.OutputType = controller.Print | ||||||
|  |  | ||||||
|  | 	// Report number of CPUs being used, if set. | ||||||
|  | 	if runtime.NumCPU() != controller.NumCPUs { | ||||||
|  | 		outp.Println("Using " + strconv.Itoa(controller.NumCPUs) + " CPUs, (was " + strconv.Itoa(runtime.NumCPU()) + ")") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Load blocks from CSV file | ||||||
|  | 	solver.LoadBlocks() | ||||||
|  |  | ||||||
|  | 	// Find rows that fit with the entered rows | ||||||
|  | 	solver.PopulateBlocks() | ||||||
|  |  | ||||||
|  | 	// If needed, split the workload | ||||||
|  | 	// May exit and throw an error if the work load isn't viable | ||||||
|  | 	if controller.Split != 1 { | ||||||
|  | 		solver.SelectWorkload() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Print the total number of solutions to validate | ||||||
|  | 	outp.Println("Number of (potential) solutions:", solver.Iter) | ||||||
|  |  | ||||||
|  | 	// Check the number of solutions | ||||||
|  | 	go solver.CheckCombinations() | ||||||
|  | 	solver.Tracker() | ||||||
|  |  | ||||||
|  | 	fmt.Println(export.Export()) | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										80
									
								
								notes.txt
									
									
									
									
									
								
							
							
						
						
									
										80
									
								
								notes.txt
									
									
									
									
									
								
							| @@ -1,80 +0,0 @@ | |||||||
| Compat Matrix |  | ||||||
|  |  | ||||||
| 1: 1 2 3 4 7 |  | ||||||
| 2: 1 2 3 5 8 |  | ||||||
| 3: 1 2 3 6 9 |  | ||||||
| 4: 1 4 5 6 7 |  | ||||||
| 5: 2 4 5 6 8 |  | ||||||
| 6: 3 4 5 6 9 |  | ||||||
| 7: 1 4 7 8 9 |  | ||||||
| 8: 2 5 7 8 9 |  | ||||||
| 9: 3 6 7 8 9 |  | ||||||
|  |  | ||||||
| // counter: 123456789 |  | ||||||
| // 1st Digit: 1 (49) |  | ||||||
| // 2nd Digit: 2 (50) |  | ||||||
| // 3rd Digit: 3 (51) |  | ||||||
| // 4th Digit: 4 (52) |  | ||||||
| // 5th Digit: 5 (53) |  | ||||||
| // 6th Digit: 6 (54) |  | ||||||
| // 7th Digit: 7 (55) |  | ||||||
| // 8th Digit: 8 (56) |  | ||||||
| // 9th Digit: 9 (57) |  | ||||||
|  |  | ||||||
| // 362880 |  | ||||||
|  |  | ||||||
| // blocks := generate_blocks() |  | ||||||
| // fmt.Println(len(blocks)) |  | ||||||
| // print_block(blocks[0]) |  | ||||||
| // print_block(blocks[1]) |  | ||||||
| // for i := range blocks { |  | ||||||
| // 	fmt.Println(blocks[i]) |  | ||||||
| // } |  | ||||||
|  |  | ||||||
| row1 := "769104802" |  | ||||||
| row2 := "154800060" |  | ||||||
| row3 := "002700150" |  | ||||||
| row4 := "600900308" |  | ||||||
| row5 := "045328670" |  | ||||||
| row6 := "328670945" |  | ||||||
| row7 := "597410280" |  | ||||||
| row8 := "006283090" |  | ||||||
| row9 := "200590006" |  | ||||||
|  |  | ||||||
| row1 := "769154832" |  | ||||||
| row2 := "154832769" |  | ||||||
| row3 := "832769154" |  | ||||||
| row4 := "671945328" |  | ||||||
| row5 := "945328671" |  | ||||||
| row6 := "328671945" |  | ||||||
| row7 := "597416283" |  | ||||||
| row8 := "416284597" |  | ||||||
| row9 := "283597416" |  | ||||||
|  |  | ||||||
| 	// for rows1_index := range solver.row1s { |  | ||||||
| 	// 	for rows2_index := range solver.row2s { |  | ||||||
| 	// 		for rows3_index := range solver.row3s { |  | ||||||
| 	// 			for rows4_index := range solver.row4s { |  | ||||||
| 	// 				for rows5_index := range solver.row5s { |  | ||||||
| 	// 					for rows6_index := range solver.row6s { |  | ||||||
| 	// 						for rows7_index := range solver.row7s { |  | ||||||
| 	// 							for rows8_index := range solver.row8s { |  | ||||||
| 	// 								for rows9_index := range solver.row9s { |  | ||||||
| 	// 									go solver.routine_validator(rows1_index, rows2_index, rows3_index, rows4_index, rows5_index, rows6_index, rows7_index, rows8_index, rows9_index) |  | ||||||
| 	// 									// if solver.validate_combination(solver.row1s[rows1_index], solver.row2s[rows2_index], solver.row3s[rows3_index], solver.row4s[rows4_index], solver.row5s[rows5_index], solver.row6s[rows6_index], solver.row7s[rows7_index], solver.row8s[rows8_index], solver.row9s[rows9_index]) { |  | ||||||
| 	// 									// 	solver.solutions = append(solver.solutions, solver.render_combination(solver.row1s[rows1_index], solver.row2s[rows2_index], solver.row3s[rows3_index], solver.row4s[rows4_index], solver.row5s[rows5_index], solver.row6s[rows6_index], solver.row7s[rows7_index], solver.row8s[rows8_index], solver.row9s[rows9_index])) |  | ||||||
| 	// 									// } |  | ||||||
| 	// 									// solver.counter = solver.counter + 1 |  | ||||||
| 	// 									// if solver.counter%1000000 == 0 { |  | ||||||
| 	// 									// 	percentage = (float32(solver.counter) / (float32(solver.iter) / 100)) |  | ||||||
| 	// 									// 	fmt.Println("Processing:", percentage, "%; Procs:", runtime.NumGoroutine()) |  | ||||||
| 	// 									// } |  | ||||||
| 	// 								} |  | ||||||
| 	// 							} |  | ||||||
| 	// 						} |  | ||||||
| 	// 					} |  | ||||||
| 	// 				} |  | ||||||
| 	// 			} |  | ||||||
| 	// 		} |  | ||||||
| 	// 	} |  | ||||||
| 	// } |  | ||||||
							
								
								
									
										17
									
								
								outputter/Printf.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								outputter/Printf.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | package outputter | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (outputter *Outputter) Printf(format string, args ...any) { | ||||||
|  | 	switch outputter.OutputType { | ||||||
|  | 	case "short": | ||||||
|  | 		fmt.Printf(format, args...) | ||||||
|  | 	case "long": | ||||||
|  | 		log.Printf(format, args...) | ||||||
|  | 	case "silent": | ||||||
|  | 		// Do nothing | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								outputter/Println.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								outputter/Println.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | package outputter | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (outputter *Outputter) Println(msg ...any) { | ||||||
|  | 	switch outputter.OutputType { | ||||||
|  | 	case "short": | ||||||
|  | 		fmt.Println(msg...) | ||||||
|  | 	case "long": | ||||||
|  | 		log.Println(msg...) | ||||||
|  | 	case "silent": | ||||||
|  | 		// Do nothing | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								outputter/types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								outputter/types.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | package outputter | ||||||
|  |  | ||||||
|  | type Outputter struct { | ||||||
|  | 	OutputType string | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								solver/CheckCombinations.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								solver/CheckCombinations.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Iterate through all combination of blocks and validate them. | ||||||
|  | func (solver *Solver) CheckCombinations() { | ||||||
|  | 	for rows1Index := range solver.row1s { | ||||||
|  | 		for rows2Index := range solver.row2s { | ||||||
|  | 			for rows3Index := range solver.row3s { | ||||||
|  | 				for rows4Index := range solver.row4s { | ||||||
|  | 					for rows5Index := range solver.row5s { | ||||||
|  | 						for rows6Index := range solver.row6s { | ||||||
|  | 							for rows7Index := range solver.row7s { | ||||||
|  | 								for rows8Index := range solver.row8s { | ||||||
|  | 									for rows9Index := range solver.row9s { | ||||||
|  | 										go solver.validator(rows1Index, rows2Index, rows3Index, rows4Index, rows5Index, rows6Index, rows7Index, rows8Index, rows9Index) | ||||||
|  | 									} | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										39
									
								
								solver/LoadBlocks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								solver/LoadBlocks.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"embed" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Dirty AF. Not happy with this. 😒 | ||||||
|  | // | ||||||
|  | //go:embed blocks.csv | ||||||
|  | var f embed.FS | ||||||
|  |  | ||||||
|  | // Load all possible blocks from CSV in to memory | ||||||
|  | func (solver *Solver) LoadBlocks() { | ||||||
|  |  | ||||||
|  | 	defer solver.timeTrack(time.Now(), "Done!") | ||||||
|  | 	solver.Outp.Printf("Loading blocks... ") | ||||||
|  |  | ||||||
|  | 	var blocks []string | ||||||
|  |  | ||||||
|  | 	file, err := f.ReadFile("blocks.csv") | ||||||
|  | 	if err != nil { | ||||||
|  | 		panic(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	temp := strings.Split(string(file), "\n") | ||||||
|  |  | ||||||
|  | 	for _, line := range temp { | ||||||
|  | 		block := string(line) | ||||||
|  | 		if block == "\n" || block == "" { // Ignore new-line at the end of the file. | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		blocks = append(blocks, block) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	solver.Controller.Blocks = blocks | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								solver/PopulateBlocks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								solver/PopulateBlocks.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Find all possible blocks that can be used to find a solution. | ||||||
|  | func (solver *Solver) PopulateBlocks() { | ||||||
|  |  | ||||||
|  | 	defer solver.timeTrack(time.Now(), "Done!") | ||||||
|  | 	solver.Outp.Printf("Populating blocks... ") | ||||||
|  |  | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row1, &solver.row1s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row2, &solver.row2s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row3, &solver.row3s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row4, &solver.row4s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row5, &solver.row5s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row6, &solver.row6s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row7, &solver.row7s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row8, &solver.row8s) | ||||||
|  | 	solver.findBlocks(&solver.Controller.Row9, &solver.row9s) | ||||||
|  |  | ||||||
|  | 	// This calculates and stores the total number of solutions to validate. | ||||||
|  | 	solver.Iter = uint64(len(solver.row1s)) * uint64(len(solver.row2s)) * uint64(len(solver.row3s)) * uint64(len(solver.row4s)) * uint64(len(solver.row5s)) * uint64(len(solver.row6s)) * uint64(len(solver.row7s)) * uint64(len(solver.row8s)) * uint64(len(solver.row9s)) | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								solver/SelectWorkload.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								solver/SelectWorkload.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"os" | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Renders workload for an agent. | ||||||
|  | // Checks if this feature can be used, otherwise exits. | ||||||
|  | // Modify solver.row1s so it limits the workload to what is only desired | ||||||
|  | func (solver *Solver) SelectWorkload() { | ||||||
|  | 	if solver.Controller.Split > len(solver.row1s) { | ||||||
|  | 		solver.Outp.Println("ERROR: Unable to divide the workload in " + strconv.Itoa(solver.Controller.Split) + " parts, when only " + strconv.Itoa(len(solver.row1s)) + " are available.\n\n") | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 	defer solver.timeTrack(time.Now(), "Workload set") | ||||||
|  | 	solver.Outp.Println("Setting workload") | ||||||
|  | 	solver.Outp.Println("We are agent " + strconv.Itoa(solver.Controller.Part) + " of " + strconv.Itoa(solver.Controller.Split)) | ||||||
|  | 	workloads := solver.splitWorkload() | ||||||
|  | 	solver.setWorkload(workloads) | ||||||
|  | } | ||||||
							
								
								
									
										102
									
								
								solver/Tracker.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								solver/Tracker.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Keep track and output progress. | ||||||
|  | // Calculate rates, display percentages, estimate the ETA till completion. | ||||||
|  | func (solver *Solver) Tracker() { | ||||||
|  |  | ||||||
|  | 	// Add time tracking | ||||||
|  | 	defer solver.timeTrack(time.Now(), "Validated solutions") | ||||||
|  | 	solver.Outp.Println("Validating solutions") | ||||||
|  |  | ||||||
|  | 	// Determine if the main-loop is done | ||||||
|  | 	var done bool | ||||||
|  |  | ||||||
|  | 	// Tracking progress in percentages | ||||||
|  | 	var percentage float32 | ||||||
|  | 	// Tracking progress in validated solutions | ||||||
|  | 	var track int | ||||||
|  |  | ||||||
|  | 	// Tracking the rate, starting point | ||||||
|  | 	var rateStart uint64 | ||||||
|  | 	// Tracking the rate, difference between previous iterations | ||||||
|  | 	var rateDiff uint64 | ||||||
|  |  | ||||||
|  | 	// Tracking duration | ||||||
|  | 	var timerStart = time.Now() | ||||||
|  |  | ||||||
|  | 	// Estimation how long it will take | ||||||
|  | 	var est_fin string | ||||||
|  |  | ||||||
|  | 	// While not needed for rateDiff anymore, it makes estimation calculations more accurate. ☹️ | ||||||
|  | 	time.Sleep(time.Second) | ||||||
|  |  | ||||||
|  | 	// for solver.Iter != solver.counter { // Start for-loop | ||||||
|  | 	for !done { | ||||||
|  |  | ||||||
|  | 		// Determine how far we are. | ||||||
|  | 		percentage = (float32(solver.counter.Load()) / (float32(solver.Iter) / 100)) | ||||||
|  |  | ||||||
|  | 		// Reset the loop | ||||||
|  | 		rateDiff = solver.counter.Load() - rateStart | ||||||
|  |  | ||||||
|  | 		if track <= int(percentage) || rateDiff == 0 { // Start if-statement | ||||||
|  |  | ||||||
|  | 			// Make sure something happened, making rateStart the only reliable variable | ||||||
|  | 			if solver.Iter == solver.counter.Load() { | ||||||
|  | 				percentage = 100 | ||||||
|  | 				solver.counter.Store(solver.Iter) | ||||||
|  | 				done = true | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			timer_elapsed := time.Since(timerStart) | ||||||
|  | 			solver.rates = append(solver.rates, rateDiff) | ||||||
|  | 			rate_avg := solver.calcAVG() | ||||||
|  |  | ||||||
|  | 			// Estimate when this is finished | ||||||
|  | 			if rateDiff == 0 { | ||||||
|  | 				est_fin = "N/A" | ||||||
|  | 			} else { | ||||||
|  | 				duration_int := (solver.Iter - solver.counter.Load()) / rate_avg | ||||||
|  | 				duration_string := strconv.Itoa(int(duration_int)) + "s" | ||||||
|  | 				est, err := time.ParseDuration(duration_string) | ||||||
|  | 				if err != nil { | ||||||
|  | 					est_fin = "parse error" | ||||||
|  | 				} else { | ||||||
|  | 					est_fin = est.String() | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// Printing the progress | ||||||
|  | 			solver.Outp.Println("Processing: " + strconv.Itoa(int(percentage)) + "% (" + strconv.FormatUint(solver.counter.Load(), 10) + "/" + strconv.Itoa(int(solver.Iter)) + "); Rate: " + strconv.FormatUint(rateDiff, 10) + "/sec for " + timer_elapsed.String() + "; Time left (est.): " + est_fin) | ||||||
|  |  | ||||||
|  | 			// After we are done printing, exit this for-loop | ||||||
|  | 			if percentage == 100 { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// Wrap up the loop or break | ||||||
|  | 			if int(percentage) > track { | ||||||
|  | 				track = int(percentage) | ||||||
|  | 			} else { | ||||||
|  | 				track = track + 1 | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			timerStart = time.Now() | ||||||
|  |  | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Resert the rate counter | ||||||
|  | 		rateStart = solver.counter.Load() | ||||||
|  |  | ||||||
|  | 		// Sleep for a second | ||||||
|  | 		if solver.Iter != solver.counter.Load() { | ||||||
|  | 			time.Sleep(1 * time.Second) | ||||||
|  | 		} | ||||||
|  | 	} // End for-loop | ||||||
|  |  | ||||||
|  | } | ||||||
| Can't render this file because it is too large. | 
| @@ -1,34 +0,0 @@ | |||||||
| package solver |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/csv" |  | ||||||
| 	"io" |  | ||||||
| 	"log" |  | ||||||
| 	"os" |  | ||||||
| 	"strconv" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func (solver *Solver) load_blocks() { |  | ||||||
| 	var blocks []int |  | ||||||
|  |  | ||||||
| 	file, err := os.Open("blocks.csv") |  | ||||||
| 	if err != nil { |  | ||||||
| 		panic(err) |  | ||||||
| 	} |  | ||||||
| 	defer file.Close() |  | ||||||
|  |  | ||||||
| 	r := csv.NewReader(file) |  | ||||||
| 	for { |  | ||||||
| 		record, err := r.Read() |  | ||||||
| 		if err == io.EOF { |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Fatal(err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		block, _ := strconv.Atoi(record[0]) |  | ||||||
| 		blocks = append(blocks, block) |  | ||||||
| 	} |  | ||||||
| 	solver.blocks = blocks |  | ||||||
| } |  | ||||||
							
								
								
									
										14
									
								
								solver/calcAVG.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								solver/calcAVG.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Calculate the average rate in a stored slice of rates. | ||||||
|  | func (solver *Solver) calcAVG() (avg uint64) { | ||||||
|  | 	var avgSum uint64 | ||||||
|  |  | ||||||
|  | 	for _, value := range solver.rates { | ||||||
|  | 		avgSum += uint64(value) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	avg = avgSum / uint64(len(solver.rates)) | ||||||
|  |  | ||||||
|  | 	return | ||||||
|  | } | ||||||
							
								
								
									
										37
									
								
								solver/findBlocks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								solver/findBlocks.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // The actual function that finds the blocks matching the partial blocks. | ||||||
|  | func (solver *Solver) findBlocks(row *string, rows *[]string) { | ||||||
|  | 	// Declare selection | ||||||
|  | 	var selection []string | ||||||
|  | 	var currBlocks []string | ||||||
|  | 	funcRow := *row | ||||||
|  |  | ||||||
|  | 	for letter := range funcRow { | ||||||
|  |  | ||||||
|  | 		if len(selection) == 0 { | ||||||
|  | 			currBlocks = solver.Controller.Blocks | ||||||
|  | 		} else { | ||||||
|  | 			currBlocks = selection | ||||||
|  | 			selection = nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for _, block := range currBlocks { | ||||||
|  |  | ||||||
|  | 			currRow := block | ||||||
|  |  | ||||||
|  | 			if funcRow[letter] == currRow[letter] { | ||||||
|  | 				foundRow := currRow | ||||||
|  | 				selection = append(selection, foundRow) | ||||||
|  | 			} | ||||||
|  | 			if funcRow[letter] == '0' { | ||||||
|  | 				foundRow := currRow | ||||||
|  | 				selection = append(selection, foundRow) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 		} // End for-loop | ||||||
|  |  | ||||||
|  | 	} // End for-loop | ||||||
|  |  | ||||||
|  | 	*rows = selection | ||||||
|  | } | ||||||
| @@ -1,138 +0,0 @@ | |||||||
| package solver |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"runtime" |  | ||||||
| 	"strconv" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func (solver *Solver) find_blocks(row *string, rows *[]int) { |  | ||||||
| 	// Declare selection |  | ||||||
| 	var selection []int |  | ||||||
| 	var curr_blocks []int |  | ||||||
| 	func_row := *row |  | ||||||
|  |  | ||||||
| 	// fmt.Println(row) |  | ||||||
| 	for letter := range func_row { |  | ||||||
|  |  | ||||||
| 		if len(selection) == 0 { |  | ||||||
| 			curr_blocks = solver.blocks |  | ||||||
| 		} else { |  | ||||||
| 			curr_blocks = selection |  | ||||||
| 			selection = nil |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		for block := range curr_blocks { |  | ||||||
|  |  | ||||||
| 			curr_row := strconv.Itoa(curr_blocks[block]) |  | ||||||
|  |  | ||||||
| 			if func_row[letter] == curr_row[letter] { |  | ||||||
| 				found_row, _ := strconv.Atoi(curr_row) |  | ||||||
| 				selection = append(selection, found_row) |  | ||||||
| 			} |  | ||||||
| 			if func_row[letter] == '0' { |  | ||||||
| 				found_row, _ := strconv.Atoi(curr_row) |  | ||||||
| 				selection = append(selection, found_row) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 		} // End for-loop |  | ||||||
|  |  | ||||||
| 	} // End for-loop |  | ||||||
|  |  | ||||||
| 	*rows = selection |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) check_combinations() { |  | ||||||
| 	for rows1_index := range solver.row1s { |  | ||||||
| 		for rows2_index := range solver.row2s { |  | ||||||
| 			for rows3_index := range solver.row3s { |  | ||||||
| 				for rows4_index := range solver.row4s { |  | ||||||
| 					for rows5_index := range solver.row5s { |  | ||||||
| 						for rows6_index := range solver.row6s { |  | ||||||
| 							for rows7_index := range solver.row7s { |  | ||||||
| 								for rows8_index := range solver.row8s { |  | ||||||
| 									for rows9_index := range solver.row9s { |  | ||||||
| 										go solver.routine_validator(rows1_index, rows2_index, rows3_index, rows4_index, rows5_index, rows6_index, rows7_index, rows8_index, rows9_index) |  | ||||||
| 									} |  | ||||||
| 								} |  | ||||||
| 							} |  | ||||||
| 						} |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_validator(rows1_index int, rows2_index int, rows3_index int, rows4_index int, rows5_index int, rows6_index int, rows7_index int, rows8_index int, rows9_index int) { |  | ||||||
|  |  | ||||||
| 	var percentage float32 |  | ||||||
|  |  | ||||||
| 	if solver.validate_combination(solver.row1s[rows1_index], solver.row2s[rows2_index], solver.row3s[rows3_index], solver.row4s[rows4_index], solver.row5s[rows5_index], solver.row6s[rows6_index], solver.row7s[rows7_index], solver.row8s[rows8_index], solver.row9s[rows9_index]) { |  | ||||||
| 		solver.solutions = append(solver.solutions, solver.render_combination(solver.row1s[rows1_index], solver.row2s[rows2_index], solver.row3s[rows3_index], solver.row4s[rows4_index], solver.row5s[rows5_index], solver.row6s[rows6_index], solver.row7s[rows7_index], solver.row8s[rows8_index], solver.row9s[rows9_index])) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	solver.counter = solver.counter + 1 |  | ||||||
|  |  | ||||||
| 	if solver.counter%1000000 == 0 { |  | ||||||
| 		percentage = (float32(solver.counter) / (float32(solver.iter) / 100)) |  | ||||||
| 		fmt.Println("Processing:", percentage, "%; Procs:", runtime.NumGoroutine()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) validate_combination(row1 int, row2 int, row3 int, row4 int, row5 int, row6 int, row7 int, row8 int, row9 int) bool { |  | ||||||
| 	var retval bool |  | ||||||
| 	retval = true |  | ||||||
|  |  | ||||||
| 	row1s := strconv.Itoa(row1) |  | ||||||
| 	row2s := strconv.Itoa(row2) |  | ||||||
| 	row3s := strconv.Itoa(row3) |  | ||||||
| 	row4s := strconv.Itoa(row4) |  | ||||||
| 	row5s := strconv.Itoa(row5) |  | ||||||
| 	row6s := strconv.Itoa(row6) |  | ||||||
| 	row7s := strconv.Itoa(row7) |  | ||||||
| 	row8s := strconv.Itoa(row8) |  | ||||||
| 	row9s := strconv.Itoa(row9) |  | ||||||
|  |  | ||||||
| 	for index := range 9 { |  | ||||||
| 		if row1s[index] == row2s[index] || row1s[index] == row3s[index] || row1s[index] == row4s[index] || row1s[index] == row5s[index] || row1s[index] == row6s[index] || row1s[index] == row7s[index] || row1s[index] == row8s[index] || row1s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row2s[index] == row1s[index] || row2s[index] == row3s[index] || row2s[index] == row4s[index] || row2s[index] == row5s[index] || row2s[index] == row6s[index] || row2s[index] == row7s[index] || row2s[index] == row8s[index] || row2s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row3s[index] == row1s[index] || row3s[index] == row2s[index] || row3s[index] == row4s[index] || row3s[index] == row5s[index] || row3s[index] == row6s[index] || row3s[index] == row7s[index] || row3s[index] == row8s[index] || row3s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row4s[index] == row1s[index] || row4s[index] == row2s[index] || row4s[index] == row3s[index] || row4s[index] == row5s[index] || row4s[index] == row6s[index] || row4s[index] == row7s[index] || row4s[index] == row8s[index] || row4s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row5s[index] == row1s[index] || row5s[index] == row2s[index] || row5s[index] == row3s[index] || row5s[index] == row4s[index] || row5s[index] == row6s[index] || row5s[index] == row7s[index] || row5s[index] == row8s[index] || row5s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row6s[index] == row1s[index] || row6s[index] == row2s[index] || row6s[index] == row3s[index] || row6s[index] == row4s[index] || row6s[index] == row5s[index] || row6s[index] == row7s[index] || row6s[index] == row8s[index] || row6s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row7s[index] == row1s[index] || row7s[index] == row2s[index] || row7s[index] == row3s[index] || row7s[index] == row4s[index] || row5s[index] == row6s[index] || row7s[index] == row6s[index] || row7s[index] == row8s[index] || row7s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row8s[index] == row1s[index] || row8s[index] == row2s[index] || row8s[index] == row3s[index] || row8s[index] == row4s[index] || row8s[index] == row5s[index] || row8s[index] == row6s[index] || row8s[index] == row7s[index] || row8s[index] == row9s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if row9s[index] == row1s[index] || row9s[index] == row2s[index] || row9s[index] == row3s[index] || row9s[index] == row4s[index] || row9s[index] == row5s[index] || row9s[index] == row6s[index] || row9s[index] == row7s[index] || row9s[index] == row8s[index] { |  | ||||||
| 			retval = false |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return retval |  | ||||||
| } |  | ||||||
							
								
								
									
										21
									
								
								solver/setWorkload.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								solver/setWorkload.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Set the workload by setting solver.row1s | ||||||
|  | func (solver *Solver) setWorkload(agents []int) { | ||||||
|  | 	var start int = 0 | ||||||
|  | 	var finish int = 0 | ||||||
|  | 	for key, value := range agents { | ||||||
|  | 		if key == solver.Controller.Part-1 { | ||||||
|  | 			finish = start + value | ||||||
|  | 			break | ||||||
|  | 		} else { | ||||||
|  | 			start += value | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Set the shortened set of instructions | ||||||
|  | 	solver.row1s = solver.row1s[start:finish] | ||||||
|  |  | ||||||
|  | 	// Recalculate how much we need to grind through | ||||||
|  | 	solver.Iter = uint64(len(solver.row1s)) * uint64(len(solver.row2s)) * uint64(len(solver.row3s)) * uint64(len(solver.row4s)) * uint64(len(solver.row5s)) * uint64(len(solver.row6s)) * uint64(len(solver.row7s)) * uint64(len(solver.row8s)) * uint64(len(solver.row9s)) | ||||||
|  | } | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| package solver |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func Run() { |  | ||||||
| 	solver := Solver{} |  | ||||||
| 	solver.row1 = "769104802" |  | ||||||
| 	solver.row2 = "154800060" |  | ||||||
| 	solver.row3 = "002700150" |  | ||||||
| 	solver.row4 = "600900308" |  | ||||||
| 	solver.row5 = "045328670" |  | ||||||
| 	solver.row6 = "328670945" |  | ||||||
| 	solver.row7 = "597410280" |  | ||||||
| 	solver.row8 = "006283090" |  | ||||||
| 	solver.row9 = "200590006" |  | ||||||
|  |  | ||||||
| 	solver.load_blocks() |  | ||||||
| 	//fmt.Println("Total blocks:", len(solver.blocks)) |  | ||||||
|  |  | ||||||
| 	// Find rows that fit with the entered rows |  | ||||||
| 	solver.find_blocks(&solver.row1, &solver.row1s) |  | ||||||
| 	solver.find_blocks(&solver.row2, &solver.row2s) |  | ||||||
| 	solver.find_blocks(&solver.row3, &solver.row3s) |  | ||||||
| 	solver.find_blocks(&solver.row4, &solver.row4s) |  | ||||||
| 	solver.find_blocks(&solver.row5, &solver.row5s) |  | ||||||
| 	solver.find_blocks(&solver.row6, &solver.row6s) |  | ||||||
| 	solver.find_blocks(&solver.row7, &solver.row7s) |  | ||||||
| 	solver.find_blocks(&solver.row8, &solver.row8s) |  | ||||||
| 	solver.find_blocks(&solver.row9, &solver.row9s) |  | ||||||
|  |  | ||||||
| 	solver.iter = int64(len(solver.row1s)) * int64(len(solver.row2s)) * int64(len(solver.row3s)) * int64(len(solver.row4s)) * int64(len(solver.row5s)) * int64(len(solver.row6s)) * int64(len(solver.row7s)) * int64(len(solver.row8s)) * int64(len(solver.row9s)) |  | ||||||
| 	fmt.Println("Number of iterations:", solver.iter) |  | ||||||
|  |  | ||||||
| 	solver.check_combinations() |  | ||||||
|  |  | ||||||
| 	fmt.Println(solver.solutions) |  | ||||||
|  |  | ||||||
| } |  | ||||||
							
								
								
									
										19
									
								
								solver/splitWorkload.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								solver/splitWorkload.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Determine how workload should be split among the agents | ||||||
|  | func (solver *Solver) splitWorkload() []int { | ||||||
|  | 	agents := make([]int, solver.Controller.Split) | ||||||
|  | 	var tracker int | ||||||
|  | 	var tasks int = len(solver.row1s) | ||||||
|  |  | ||||||
|  | 	for tasks != 0 { | ||||||
|  | 		agents[tracker] += 1 | ||||||
|  | 		tasks -= 1 | ||||||
|  | 		tracker += 1 | ||||||
|  | 		if tracker == solver.Controller.Split { | ||||||
|  | 			tracker = 0 | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return agents | ||||||
|  | } | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| package solver |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"strconv" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func (solver *Solver) render_combination(row1 int, row2 int, row3 int, row4 int, row5 int, row6 int, row7 int, row8 int, row9 int) string { |  | ||||||
|  |  | ||||||
| 	row1s := strconv.Itoa(row1) |  | ||||||
| 	row2s := strconv.Itoa(row2) |  | ||||||
| 	row3s := strconv.Itoa(row3) |  | ||||||
| 	row4s := strconv.Itoa(row4) |  | ||||||
| 	row5s := strconv.Itoa(row5) |  | ||||||
| 	row6s := strconv.Itoa(row6) |  | ||||||
| 	row7s := strconv.Itoa(row7) |  | ||||||
| 	row8s := strconv.Itoa(row8) |  | ||||||
| 	row9s := strconv.Itoa(row9) |  | ||||||
|  |  | ||||||
| 	return row1s + "\n" + row2s + "\n" + row3s + "\n" + row4s + "\n" + row5s + "\n" + row6s + "\n" + row7s + "\n" + row8s + "\n" + row9s + "\n" |  | ||||||
| } |  | ||||||
							
								
								
									
										12
									
								
								solver/timeTrack.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								solver/timeTrack.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // A simple function to track time | ||||||
|  | // Use with `defer` | ||||||
|  | func (solver *Solver) timeTrack(start time.Time, msg string) { | ||||||
|  | 	elapsed := time.Since(start) | ||||||
|  | 	solver.Outp.Printf("%s (%s)\n", msg, elapsed) | ||||||
|  | } | ||||||
| @@ -1,26 +1,39 @@ | |||||||
| package solver | package solver | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"sync/atomic" | ||||||
|  |  | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/controller" | ||||||
|  | 	"gitea.ligthert.net/golang/sudoku-funpark/outputter" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Solve a given Sudoku puzzle by iterating through all possible solutions. | ||||||
| type Solver struct { | type Solver struct { | ||||||
| 	blocks    []int | 	Controller *controller.Controller | ||||||
| 	row1      string | 	// Slice of possible blocks for the 1st row. | ||||||
| 	row2      string | 	row1s []string | ||||||
| 	row3      string | 	// Slice of possible blocks for the 2nd row. | ||||||
| 	row4      string | 	row2s []string | ||||||
| 	row5      string | 	// Slice of possible blocks for the 3rd row. | ||||||
| 	row6      string | 	row3s []string | ||||||
| 	row7      string | 	// Slice of possible blocks for the 4th row. | ||||||
| 	row8      string | 	row4s []string | ||||||
| 	row9      string | 	// Slice of possible blocks for the 5th row. | ||||||
| 	row1s     []int | 	row5s []string | ||||||
| 	row2s     []int | 	// Slice of possible blocks for the 6th row. | ||||||
| 	row3s     []int | 	row6s []string | ||||||
| 	row4s     []int | 	// Slice of possible blocks for the 7th row. | ||||||
| 	row5s     []int | 	row7s []string | ||||||
| 	row6s     []int | 	// Slice of possible blocks for the 8th row. | ||||||
| 	row7s     []int | 	row8s []string | ||||||
| 	row8s     []int | 	// Slice of possible blocks for the 9th row. | ||||||
| 	row9s     []int | 	row9s []string | ||||||
| 	iter      int64 | 	// Maximum number of possible solutions with the current set of rows. | ||||||
| 	counter   int64 | 	Iter uint64 | ||||||
| 	solutions []string | 	// Progress counter, needs atomic due to the number of updates. | ||||||
|  | 	counter atomic.Uint64 | ||||||
|  | 	// Slice of rates for accurate duration estimation. | ||||||
|  | 	rates []uint64 | ||||||
|  | 	// Reference to Outputter interface | ||||||
|  | 	Outp *outputter.Outputter | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										47
									
								
								solver/validateCombination.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								solver/validateCombination.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Validate combination | ||||||
|  | func (solver *Solver) validateCombination(row1 string, row2 string, row3 string, row4 string, row5 string, row6 string, row7 string, row8 string, row9 string) (retval bool) { | ||||||
|  | 	retval = true | ||||||
|  |  | ||||||
|  | 	for index := range 9 { | ||||||
|  | 		if row1[index] == row2[index] || row1[index] == row3[index] || row1[index] == row4[index] || row1[index] == row5[index] || row1[index] == row6[index] || row1[index] == row7[index] || row1[index] == row8[index] || row1[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row2[index] == row1[index] || row2[index] == row3[index] || row2[index] == row4[index] || row2[index] == row5[index] || row2[index] == row6[index] || row2[index] == row7[index] || row2[index] == row8[index] || row2[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row3[index] == row1[index] || row3[index] == row2[index] || row3[index] == row4[index] || row3[index] == row5[index] || row3[index] == row6[index] || row3[index] == row7[index] || row3[index] == row8[index] || row3[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row4[index] == row1[index] || row4[index] == row2[index] || row4[index] == row3[index] || row4[index] == row5[index] || row4[index] == row6[index] || row4[index] == row7[index] || row4[index] == row8[index] || row4[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row5[index] == row1[index] || row5[index] == row2[index] || row5[index] == row3[index] || row5[index] == row4[index] || row5[index] == row6[index] || row5[index] == row7[index] || row5[index] == row8[index] || row5[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row6[index] == row1[index] || row6[index] == row2[index] || row6[index] == row3[index] || row6[index] == row4[index] || row6[index] == row5[index] || row6[index] == row7[index] || row6[index] == row8[index] || row6[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row7[index] == row1[index] || row7[index] == row2[index] || row7[index] == row3[index] || row7[index] == row4[index] || row5[index] == row6[index] || row7[index] == row6[index] || row7[index] == row8[index] || row7[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row8[index] == row1[index] || row8[index] == row2[index] || row8[index] == row3[index] || row8[index] == row4[index] || row8[index] == row5[index] || row8[index] == row6[index] || row8[index] == row7[index] || row8[index] == row9[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if row9[index] == row1[index] || row9[index] == row2[index] || row9[index] == row3[index] || row9[index] == row4[index] || row9[index] == row5[index] || row9[index] == row6[index] || row9[index] == row7[index] || row9[index] == row8[index] { | ||||||
|  | 			retval = false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return retval | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								solver/validator.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								solver/validator.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | package solver | ||||||
|  |  | ||||||
|  | // Validate the provided rows and verify it is a valid solution. | ||||||
|  | func (solver *Solver) validator(rows1Index int, rows2Index int, rows3Index int, rows4Index int, rows5Index int, rows6Index int, rows7Index int, rows8Index int, rows9Index int) { | ||||||
|  |  | ||||||
|  | 	solver.counter.Add(1) | ||||||
|  |  | ||||||
|  | 	if solver.validateCombination(solver.row1s[rows1Index], solver.row2s[rows2Index], solver.row3s[rows3Index], solver.row4s[rows4Index], solver.row5s[rows5Index], solver.row6s[rows6Index], solver.row7s[rows7Index], solver.row8s[rows8Index], solver.row9s[rows9Index]) { | ||||||
|  | 		solver.Controller.Solutions = append(solver.Controller.Solutions, []string{solver.row1s[rows1Index], solver.row2s[rows2Index], solver.row3s[rows3Index], solver.row4s[rows4Index], solver.row5s[rows5Index], solver.row6s[rows6Index], solver.row7s[rows7Index], solver.row8s[rows8Index], solver.row9s[rows9Index]}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										94
									
								
								unused.txt
									
									
									
									
									
								
							
							
						
						
									
										94
									
								
								unused.txt
									
									
									
									
									
								
							| @@ -1,94 +0,0 @@ | |||||||
| // Processing |  | ||||||
| func (solver *Solver) routine_row1(index1 int) { |  | ||||||
| 	for index2 := range solver.row2s { |  | ||||||
| 		go solver.routine_row2(index1, index2) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row2(index1 int, index2 int) { |  | ||||||
| 	for index3 := range solver.row3s { |  | ||||||
| 		go solver.routine_row3(index1, index2, index3) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row3(index1 int, index2 int, index3 int) { |  | ||||||
| 	for index4 := range solver.row4s { |  | ||||||
| 		go solver.routine_row4(index1, index2, index3, index4) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row4(index1 int, index2 int, index3 int, index4 int) { |  | ||||||
| 	for index5 := range solver.row5s { |  | ||||||
| 		go solver.routine_row5(index1, index2, index3, index4, index5) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row5(index1 int, index2 int, index3 int, index4 int, index5 int) { |  | ||||||
| 	for index6 := range solver.row6s { |  | ||||||
| 		go solver.routine_row6(index1, index2, index3, index4, index5, index6) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row6(index1 int, index2 int, index3 int, index4 int, index5 int, index6 int) { |  | ||||||
| 	for index7 := range solver.row7s { |  | ||||||
| 		go solver.routine_row7(index1, index2, index3, index4, index5, index6, index7) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row7(index1 int, index2 int, index3 int, index4 int, index5 int, index6 int, index7 int) { |  | ||||||
| 	for index8 := range solver.row8s { |  | ||||||
| 		go solver.routine_row8(index1, index2, index3, index4, index5, index6, index7, index8) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row8(index1 int, index2 int, index3 int, index4 int, index5 int, index6 int, index7 int, index8 int) { |  | ||||||
| 	for index9 := range solver.row9s { |  | ||||||
| 		go solver.routine_row9(index1, index2, index3, index4, index5, index6, index7, index8, index9) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (solver *Solver) routine_row9(index1 int, index2 int, index3 int, index4 int, index5 int, index6 int, index7 int, index8 int, index9 int) { |  | ||||||
| 	go solver.routine_validator(index1, index2, index3, index4, index5, index6, index7, index8, index9) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // blocks.go |  | ||||||
| func (solver *Solver) generate_blocks() []int { |  | ||||||
|  |  | ||||||
| 	var blocks []int |  | ||||||
| 	decvals := [9]int{49, 50, 51, 52, 53, 54, 55, 56, 57} |  | ||||||
|  |  | ||||||
| 	for counter := 123456789; counter <= 987654321; counter++ { |  | ||||||
|  |  | ||||||
| 		// Convert number to string ([]byte) |  | ||||||
| 		digits := strconv.Itoa(counter) |  | ||||||
|  |  | ||||||
| 		// Check if every number is only represented only once |  | ||||||
| 		var valid bool |  | ||||||
| 		valid = true |  | ||||||
| 		for decval := range decvals { |  | ||||||
| 			var count int |  | ||||||
| 			for digit := range digits { |  | ||||||
| 				if digits[digit] == byte(decvals[decval]) { |  | ||||||
| 					count = count + 1 |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if count != 1 { |  | ||||||
| 				valid = false |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if valid { |  | ||||||
| 			blocks = append(blocks, counter) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return blocks |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // stash.go |  | ||||||
| func (solver *Solver) print_block(block int) { |  | ||||||
| 	digits := strconv.Itoa(block) |  | ||||||
| 	fmt.Printf("%c %c %c\n%c %c %c\n%c %c %c\n\n", digits[0], digits[1], digits[2], digits[3], digits[4], digits[5], digits[6], digits[7], digits[8]) |  | ||||||
| } |  | ||||||
		Reference in New Issue
	
	Block a user