forked from klipspringr/wordle-cli
-
Notifications
You must be signed in to change notification settings - Fork 1
/
wordle.py
119 lines (100 loc) · 4.46 KB
/
wordle.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
import os
from enum import Enum
from typing import List
class LetterStates(Enum):
NOTGUESSEDYET = 0
NOTPRESENT = 1
INCORRECTPOSITION = 2
CORRECTPOSITION = 3
class Game:
ROUNDS = 6
LENGTH = 5
WIN_STATES = [LetterStates.CORRECTPOSITION for _ in range(LENGTH)]
def __init__(self, path_solutions="data/solutions.txt", path_guesses="data/guesses.txt"):
with open(os.path.join(os.path.dirname(__file__), path_solutions), "r") as f:
self.VALID_SOLUTIONS = tuple(l.upper() for l in f.read().splitlines() if len(l) == self.LENGTH)
with open(os.path.join(os.path.dirname(__file__), path_guesses), "r") as f:
self.VALID_GUESSES = tuple(l.upper() for l in f.read().splitlines() if len(l) == self.LENGTH)
# official list of guesses does not include solutions, so add them, ignoring duplicates (albeit no duplicates in official lists)
self.VALID_GUESSES = tuple(set(self.VALID_SOLUTIONS + self.VALID_GUESSES))
self.POSSIBLE_WORDS = list(self.VALID_GUESSES)
def play(self, player, solution, hints=False):
player.start()
round = 1
while round <= self.ROUNDS:
while True:
guess = player.guess(round)
if player.ASSUME_GUESSES_VALID:
break
elif len(guess) != self.LENGTH or not guess.isalpha():
guess = guess.strip()
player.warn(f"{ guess[:5]+'..' if len(guess) > self.LENGTH else guess } invalid")
elif guess not in self.VALID_GUESSES:
player.warn(f"{ guess } not in dict".strip())
else:
break
states = Game.check_guess(guess, solution)
if hints and states != Game.WIN_STATES:
self.POSSIBLE_WORDS = [w for w in self.POSSIBLE_WORDS if Game.is_same_response(guess, w, states)]
hint = len(self.POSSIBLE_WORDS)
else:
hint = -1
player.handle_response(guess, states, hint)
if states == Game.WIN_STATES:
if hasattr(player, "handle_win"):
player.handle_win(round)
return round
round += 1
if hasattr(player, "handle_loss"):
player.handle_loss(solution)
return None
@staticmethod
def check_guess(guess: str, solution: str) -> List[LetterStates]:
if guess == solution:
return Game.WIN_STATES
# https://mathspp.com/blog/solving-wordle-with-python
# pool is set of letters in the solution available for INCORRECTPOSITION
pool = {}
for g, s in zip(guess, solution):
if g == s:
continue
if s in pool:
pool[s] += 1
else:
pool[s] = 1
states = []
for guess_letter, solution_letter in zip(guess, solution):
if guess_letter == solution_letter:
states.append(LetterStates.CORRECTPOSITION)
elif guess_letter in solution and guess_letter in pool and pool[guess_letter] > 0:
states.append(LetterStates.INCORRECTPOSITION)
pool[guess_letter] -= 1
else:
states.append(LetterStates.NOTPRESENT)
return states
@staticmethod
def is_same_response(guess: str, solution: str, other_response: List[LetterStates]) -> bool:
if guess == solution:
return other_response == Game.WIN_STATES
# https://mathspp.com/blog/solving-wordle-with-python
# pool is set of letters in the solution available for INCORRECTPOSITION
pool = {}
for g, s in zip(guess, solution):
if g == s:
continue
if s in pool:
pool[s] += 1
else:
pool[s] = 1
for guess_letter, solution_letter, other_state in zip(guess, solution, other_response):
if guess_letter == solution_letter:
if other_state != LetterStates.CORRECTPOSITION:
return False
elif guess_letter in solution and guess_letter in pool and pool[guess_letter] > 0:
if other_state != LetterStates.INCORRECTPOSITION:
return False
pool[guess_letter] -= 1
else:
if other_state != LetterStates.NOTPRESENT:
return False
return True