wordle_decoder 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.rubocop.yml +19 -0
- data/CHANGELOG.md +5 -0
- data/Gemfile +14 -0
- data/Gemfile.lock +45 -0
- data/README.md +19 -0
- data/Rakefile +72 -0
- data/lib/least_common_words.txt +9972 -0
- data/lib/less_common_words.txt +981 -0
- data/lib/most_common_words.txt +2019 -0
- data/lib/wordle_answers.json +1 -0
- data/lib/wordle_decoder/guess.rb +83 -0
- data/lib/wordle_decoder/version.rb +5 -0
- data/lib/wordle_decoder/word.rb +122 -0
- data/lib/wordle_decoder/word_position.rb +150 -0
- data/lib/wordle_decoder/word_search.rb +76 -0
- data/lib/wordle_decoder/wordle_share.rb +122 -0
- data/lib/wordle_decoder.rb +66 -0
- data/lib/words_to_frequency_score.json +1 -0
- data/sig/wordle_decoder.rbs +4 -0
- metadata +78 -0
@@ -0,0 +1 @@
|
|
1
|
+
["cigar","rebut","sissy","humph","awake","blush","focal","evade","naval","serve","heath","dwarf","model","karma","stink","grade","quiet","bench","abate","feign","major","death","fresh","crust","stool","colon","abase","marry","react","batty","pride","floss","helix","croak","staff","paper","unfed","whelp","trawl","outdo","adobe","crazy","sower","repay","digit","crate","cluck","spike","mimic","pound","maxim","linen","unmet","flesh","booby","forth","first","stand","belly","ivory","seedy","print","yearn","drain","bribe","stout","panel","crass","flume","offal","agree","error","swirl","argue","bleed","delta","flick","totem","wooer","front","shrub","parry","biome","lapel","start","greet","goner","golem","lusty","loopy","round","audit","lying","gamma","labor","islet","civic","forge","corny","moult","basic","salad","agate","spicy","spray","essay","fjord","spend","kebab","guild","aback","motor","alone","hatch","hyper","thumb","dowry","ought","belch","dutch","pilot","tweed","comet","jaunt","enema","steed","abyss","growl","fling","dozen","boozy","erode","world","gouge","click","briar","great","altar","pulpy","blurt","coast","duchy","groin","fixer","group","rogue","badly","smart","pithy","gaudy","chill","heron","vodka","finer","surer","radio","rouge","perch","retch","wrote","clock","tilde","store","prove","bring","solve","cheat","grime","exult","usher","epoch","triad","break","rhino","viral","conic","masse","sonic","vital","trace","using","peach","champ","baton","brake","pluck","craze","gripe","weary","picky","acute","ferry","aside","tapir","troll","unify","rebus","boost","truss","siege","tiger","banal","slump","crank","gorge","query","drink","favor","abbey","tangy","panic","solar","shire","proxy","point","robot","prick","wince","crimp","knoll","sugar","whack","mount","perky","could","wrung","light","those","moist","shard","pleat","aloft","skill","elder","frame","humor","pause","ulcer","ultra","robin","cynic","aroma","caulk","shake","dodge","swill","tacit","other","thorn","trove","bloke","vivid","spill","chant","choke","rupee","nasty","mourn","ahead","brine","cloth","hoard","sweet","month","lapse","watch","today","focus","smelt","tease","cater","movie","saute","allow","renew","their","slosh","purge","chest","depot","epoxy","nymph","found","shall","harry","stove","lowly","snout","trope","fewer","shawl","natal","comma","foray","scare","stair","black","squad","royal","chunk","mince","shame","cheek","ample","flair","foyer","cargo","oxide","plant","olive","inert","askew","heist","shown","zesty","hasty","trash","fella","larva","forgo","story","hairy","train","homer","badge","midst","canny","fetus","butch","farce","slung","tipsy","metal","yield","delve","being","scour","glass","gamer","scrap","money","hinge","album","vouch","asset","tiara","crept","bayou","atoll","manor","creak","showy","phase","froth","depth","gloom","flood","trait","girth","piety","payer","goose","float","donor","atone","primo","apron","blown","cacao","loser","input","gloat","awful","brink","smite","beady","rusty","retro","droll","gawky","hutch","pinto","gaily","egret","lilac","sever","field","fluff","hydro","flack","agape","voice","stead","stalk","berth","madam","night","bland","liver","wedge","augur","roomy","wacky","flock","angry","bobby","trite","aphid","tryst","midge","power","elope","cinch","motto","stomp","upset","bluff","cramp","quart","coyly","youth","rhyme","buggy","alien","smear","unfit","patty","cling","glean","label","hunky","khaki","poker","gruel","twice","twang","shrug","treat","unlit","waste","merit","woven","octal","needy","clown","widow","irony","ruder","gauze","chief","onset","prize","fungi","charm","gully","inter","whoop","taunt","leery","class","theme","lofty","tibia","booze","alpha","thyme","eclat","doubt","parer","chute","stick","trice","alike","sooth","recap","saint","liege","glory","grate","admit","brisk","soggy","usurp","scald","scorn","leave","twine","sting","bough","marsh","sloth","dandy","vigor","howdy","enjoy","valid","ionic","equal","unset","floor","catch","spade","stein","exist","quirk","denim","grove","spiel","mummy","fault","foggy","flout","carry","sneak","libel","waltz","aptly","piney","inept","aloud","photo","dream","stale","vomit","ombre","fanny","unite","snarl","baker","there","glyph","pooch","hippy","spell","folly","louse","gulch","vault","godly","threw","fleet","grave","inane","shock","crave","spite","valve","skimp","claim","rainy","musty","pique","daddy","quasi","arise","aging","valet","opium","avert","stuck","recut","mulch","genre","plume","rifle","count","incur","total","wrest","mocha","deter","study","lover","safer","rivet","funny","smoke","mound","undue","sedan","pagan","swine","guile","gusty","equip","tough","canoe","chaos","covet","human","udder","lunch","blast","stray","manga","melee","lefty","quick","paste","given","octet","risen","groan","leaky","grind","carve","loose","sadly","spilt","apple","slack","honey","final","sheen","eerie","minty","slick","derby","wharf","spelt","coach","erupt","singe","price","spawn","fairy","jiffy","filmy","stack","chose","sleep","ardor","nanny","niece","woozy","handy","grace","ditto","stank","cream","usual","diode","valor","angle","ninja","muddy","chase","reply","prone","spoil","heart","shade","diner","arson","onion","sleet","dowel","couch","palsy","bowel","smile","evoke","creek","lance","eagle","idiot","siren","built","embed","award","dross","annul","goody","frown","patio","laden","humid","elite","lymph","edify","might","reset","visit","gusto","purse","vapor","crock","write","sunny","loath","chaff","slide","queer","venom","stamp","sorry","still","acorn","aping","pushy","tamer","hater","mania","awoke","brawn","swift","exile","birch","lucky","freer","risky","ghost","plier","lunar","winch","snare","nurse","house","borax","nicer","lurch","exalt","about","savvy","toxin","tunic","pried","inlay","chump","lanky","cress","eater","elude","cycle","kitty","boule","moron","tenet","place","lobby","plush","vigil","index","blink","clung","qualm","croup","clink","juicy","stage","decay","nerve","flier","shaft","crook","clean","china","ridge","vowel","gnome","snuck","icing","spiny","rigor","snail","flown","rabid","prose","thank","poppy","budge","fiber","moldy","dowdy","kneel","track","caddy","quell","dumpy","paler","swore","rebar","scuba","splat","flyer","horny","mason","doing","ozone","amply","molar","ovary","beset","queue","cliff","magic","truce","sport","fritz","edict","twirl","verse","llama","eaten","range","whisk","hovel","rehab","macaw","sigma","spout","verve","sushi","dying","fetid","brain","buddy","thump","scion","candy","chord","basin","march","crowd","arbor","gayly","musky","stain","dally","bless","bravo","stung","title","ruler","kiosk","blond","ennui","layer","fluid","tatty","score","cutie","zebra","barge","matey","bluer","aider","shook","river","privy","betel","frisk","bongo","begun","azure","weave","genie","sound","glove","braid","scope","wryly","rover","assay","ocean","bloom","irate","later","woken","silky","wreck","dwelt","slate","smack","solid","amaze","hazel","wrist","jolly","globe","flint","rouse","civil","vista","relax","cover","alive","beech","jetty","bliss","vocal","often","dolly","eight","joker","since","event","ensue","shunt","diver","poser","worst","sweep","alley","creed","anime","leafy","bosom","dunce","stare","pudgy","waive","choir","stood","spoke","outgo","delay","bilge","ideal","clasp","seize","hotly","laugh","sieve","block","meant","grape","noose","hardy","shied","drawl","daisy","putty","strut","burnt","tulip","crick","idyll","vixen","furor","geeky","cough","naive","shoal","stork","bathe","aunty","check","prime","brass","outer","furry","razor","elect","evict","imply","demur","quota","haven","cavil","swear","crump","dough","gavel","wagon","salon","nudge","harem","pitch","sworn","pupil","excel","stony","cabin","unzip","queen","trout","polyp","earth","storm","until","taper","enter","child","adopt","minor","fatty","husky","brave","filet","slime","glint","tread","steal","regal","guest","every","murky","share","spore","hoist","buxom","inner","otter","dimly","level","sumac","donut","stilt","arena","sheet","scrub","fancy","slimy","pearl","silly","porch","dingo","sepia","amble","shady","bread","friar","reign","dairy","quill","cross","brood","tuber","shear","posit","blank","villa","shank","piggy","freak","which","among","fecal","shell","would","algae","large","rabbi","agony","amuse","bushy","copse","swoon","knife","pouch","ascot","plane","crown","urban","snide","relay","abide","viola","rajah","straw","dilly","crash","amass","third","trick","tutor","woody","blurb","grief","disco","where","sassy","beach","sauna","comic","clued","creep","caste","graze","snuff","frock","gonad","drunk","prong","lurid","steel","halve","buyer","vinyl","utile","smell","adage","worry","tasty","local","trade","finch","ashen","modal","gaunt","clove","enact","adorn","roast","speck","sheik","missy","grunt","snoop","party","touch","mafia","emcee","array","south","vapid","jelly","skulk","angst","tubal","lower","crest","sweat","cyber","adore","tardy","swami","notch","groom","roach","hitch","young","align","ready","frond","strap","puree","realm","venue","swarm","offer","seven","dryer","diary","dryly","drank","acrid","heady","theta","junto","pixie","quoth","bonus","shalt","penne","amend","datum","build","piano","shelf","lodge","suing","rearm","coral","ramen","worth","psalm","infer","overt","mayor","ovoid","glide","usage","poise","randy","chuck","prank","fishy","tooth","ether","drove","idler","swath","stint","while","begat","apply","slang","tarot","radar","credo","aware","canon","shift","timer","bylaw","serum","three","steak","iliac","shirk","blunt","puppy","penal","joist","bunny","shape","beget","wheel","adept","stunt","stole","topaz","chore","fluke","afoot","bloat","bully","dense","caper","sneer","boxer","jumbo","lunge","space","avail","short","slurp","loyal","flirt","pizza","conch","tempo","droop","plate","bible","plunk","afoul","savoy","steep","agile","stake","dwell","knave","beard","arose","motif","smash","broil","glare","shove","baggy","mammy","swamp","along","rugby","wager","quack","squat","snaky","debit","mange","skate","ninth","joust","tramp","spurn","medal","micro","rebel","flank","learn","nadir","maple","comfy","remit","gruff","ester","least","mogul","fetch","cause","oaken","aglow","meaty","gaffe","shyly","racer","prowl","thief","stern","poesy","rocky","tweet","waist","spire","grope","havoc","patsy","truly","forty","deity","uncle","swish","giver","preen","bevel","lemur","draft","slope","annoy","lingo","bleak","ditty","curly","cedar","dirge","grown","horde","drool","shuck","crypt","cumin","stock","gravy","locus","wider","breed","quite","chafe","cache","blimp","deign","fiend","logic","cheap","elide","rigid","false","renal","pence","rowdy","shoot","blaze","envoy","posse","brief","never","abort","mouse","mucky","sulky","fiery","media","trunk","yeast","clear","skunk","scalp","bitty","cider","koala","duvet","segue","creme","super","grill","after","owner","ember","reach","nobly","empty","speed","gipsy","recur","smock","dread","merge","burst","kappa","amity","shaky","hover","carol","snort","synod","faint","haunt","flour","chair","detox","shrew","tense","plied","quark","burly","novel","waxen","stoic","jerky","blitz","beefy","lyric","hussy","towel","quilt","below","bingo","wispy","brash","scone","toast","easel","saucy","value","spice","honor","route","sharp","bawdy","radii","skull","phony","issue","lager","swell","urine","gassy","trial","flora","upper","latch","wight","brick","retry","holly","decal","grass","shack","dogma","mover","defer","sober","optic","crier","vying","nomad","flute","hippo","shark","drier","obese","bugle","tawny","chalk","feast","ruddy","pedal","scarf","cruel","bleat","tidal","slush","semen","windy","dusty","sally","igloo","nerdy","jewel","shone","whale","hymen","abuse","fugue","elbow","crumb","pansy","welsh","syrup","terse","suave","gamut","swung","drake","freed","afire","shirt","grout","oddly","tithe","plaid","dummy","broom","blind","torch","enemy","again","tying","pesky","alter","gazer","noble","ethos","bride","extol","decor","hobby","beast","idiom","utter","these","sixth","alarm","erase","elegy","spunk","piper","scaly","scold","hefty","chick","sooty","canal","whiny","slash","quake","joint","swept","prude","heavy","wield","femme","lasso","maize","shale","screw","spree","smoky","whiff","scent","glade","spent","prism","stoke","riper","orbit","cocoa","guilt","humus","shush","table","smirk","wrong","noisy","alert","shiny","elate","resin","whole","hunch","pixel","polar","hotel","sword","cleat","mango","rumba","puffy","filly","billy","leash","clout","dance","ovate","facet","chili","paint","liner","curio","salty","audio","snake","fable","cloak","navel","spurt","pesto","balmy","flash","unwed","early","churn","weedy","stump","lease","witty","wimpy","spoof","saner","blend","salsa","thick","warty","manic","blare","squib","spoon","probe","crepe","knack","force","debut","order","haste","teeth","agent","widen","icily","slice","ingot","clash","juror","blood","abode","throw","unity","pivot","slept","troop","spare","sewer","parse","morph","cacti","tacky","spool","demon","moody","annex","begin","fuzzy","patch","water","lumpy","admin","omega","limit","tabby","macho","aisle","skiff","basis","plank","verge","botch","crawl","lousy","slain","cubic","raise","wrack","guide","foist","cameo","under","actor","revue","fraud","harpy","scoop","climb","refer","olden","clerk","debar","tally","ethic","cairn","tulle","ghoul","hilly","crude","apart","scale","older","plain","sperm","briny","abbot","rerun","quest","crisp","bound","befit","drawn","suite","itchy","cheer","bagel","guess","broad","axiom","chard","caput","leant","harsh","curse","proud","swing","opine","taste","lupus","gumbo","miner","green","chasm","lipid","topic","armor","brush","crane","mural","abled","habit","bossy","maker","dusky","dizzy","lithe","brook","jazzy","fifty","sense","giant","surly","legal","fatal","flunk","began","prune","small","slant","scoff","torus","ninny","covey","viper","taken","moral","vogue","owing","token","entry","booth","voter","chide","elfin","ebony","neigh","minim","melon","kneed","decoy","voila","ankle","arrow","mushy","tribe","cease","eager","birth","graph","odder","terra","weird","tried","clack","color","rough","weigh","uncut","ladle","strip","craft","minus","dicey","titan","lucid","vicar","dress","ditch","gypsy","pasta","taffy","flame","swoop","aloof","sight","broke","teary","chart","sixty","wordy","sheer","leper","nosey","bulge","savor","clamp","funky","foamy","toxic","brand","plumb","dingy","butte","drill","tripe","bicep","tenor","krill","worse","drama","hyena","think","ratio","cobra","basil","scrum","bused","phone","court","camel","proof","heard","angel","petal","pouty","throb","maybe","fetal","sprig","spine","shout","cadet","macro","dodgy","satyr","rarer","binge","trend","nutty","leapt","amiss","split","myrrh","width","sonar","tower","baron","fever","waver","spark","belie","sloop","expel","smote","baler","above","north","wafer","scant","frill","awash","snack","scowl","frail","drift","limbo","fence","motel","ounce","wreak","revel","talon","prior","knelt","cello","flake","debug","anode","crime","salve","scout","imbue","pinky","stave","vague","chock","fight","video","stone","teach","cleft","frost","prawn","booty","twist","apnea","stiff","plaza","ledge","tweak","board","grant","medic","bacon","cable","brawl","slunk","raspy","forum","drone","women","mucus","boast","toddy","coven","tumor","truer","wrath","stall","steam","axial","purer","daily","trail","niche","mealy","juice","nylon","plump","merry","flail","papal","wheat","berry","cower","erect","brute","leggy","snipe","sinew","skier","penny","jumpy","rally","umbra","scary","modem","gross","avian","greed","satin","tonic","parka","sniff","livid","stark","trump","giddy","reuse","taboo","avoid","quote","devil","liken","gloss","gayer","beret","noise","gland","dealt","sling","rumor","opera","thigh","tonga","flare","wound","white","bulky","etude","horse","circa","paddy","inbox","fizzy","grain","exert","surge","gleam","belle","salvo","crush","fruit","sappy","taker","tract","ovine","spiky","frank","reedy","filth","spasm","heave","mambo","right","clank","trust","lumen","borne","spook","sauce","amber","lathe","carat","corer","dirty","slyly","affix","alloy","taint","sheep","kinky","wooly","mauve","flung","yacht","fried","quail","brunt","grimy","curvy","cagey","rinse","deuce","state","grasp","milky","bison","graft","sandy","baste","flask","hedge","girly","swash","boney","coupe","endow","abhor","welch","blade","tight","geese","miser","mirth","cloud","cabal","leech","close","tenth","pecan","droit","grail","clone","guise","ralph","tango","biddy","smith","mower","payee","serif","drape","fifth","spank","glaze","allot","truck","kayak","virus","testy","tepee","fully","zonal","metro","curry","grand","banjo","axion","bezel","occur","chain","nasal","gooey","filer","brace","allay","pubic","raven","plead","gnash","flaky","munch","dully","eking","thing","slink","hurry","theft","shorn","pygmy","ranch","wring","lemon","shore","mamma","froze","newer","style","moose","antic","drown","vegan","chess","guppy","union","lever","lorry","image","cabby","druid","exact","truth","dopey","spear","cried","chime","crony","stunk","timid","batch","gauge","rotor","crack","curve","latte","witch","bunch","repel","anvil","soapy","meter","broth","madly","dried","scene","known","magma","roost","woman","thong","punch","pasty","downy","knead","whirl","rapid","clang","anger","drive","goofy","email","music","stuff","bleep","rider","mecca","folio","setup","verso","quash","fauna","gummy","happy","newly","fussy","relic","guava","ratty","fudge","femur","chirp","forte","alibi","whine","petty","golly","plait","fleck","felon","gourd","brown","thrum","ficus","stash","decry","wiser","junta","visor","daunt","scree","impel","await","press","whose","turbo","stoop","speak","mangy","eying","inlet","crone","pulse","mossy","staid","hence","pinch","teddy","sully","snore","ripen","snowy","attic","going","leach","mouth","hound","clump","tonal","bigot","peril","piece","blame","haute","spied","undid","intro","basal","shine","gecko","rodeo","guard","steer","loamy","scamp","scram","manly","hello","vaunt","organ","feral","knock","extra","condo","adapt","willy","polka","rayon","skirt","faith","torso","match","mercy","tepid","sleek","riser","twixt","peace","flush","catty","login","eject","roger","rival","untie","refit","aorta","adult","judge","rower","artsy","rural","shave"]
|
@@ -0,0 +1,83 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class WordleDecoder
|
4
|
+
class Guess
|
5
|
+
def initialize(start_word, first_word_position, word_positions)
|
6
|
+
@start_word = start_word
|
7
|
+
@first_word_position = first_word_position
|
8
|
+
@word_positions = word_positions
|
9
|
+
end
|
10
|
+
|
11
|
+
def score
|
12
|
+
@score ||= words_with_scores.sum(&:last)
|
13
|
+
end
|
14
|
+
|
15
|
+
def word_scores
|
16
|
+
@word_scores ||= words_with_scores.map(&:last)
|
17
|
+
end
|
18
|
+
|
19
|
+
def words
|
20
|
+
@words ||= words_with_scores.map { |w, _s| w.to_s }
|
21
|
+
end
|
22
|
+
|
23
|
+
def words_with_scores
|
24
|
+
@words_with_scores ||= select_words_with_scores
|
25
|
+
end
|
26
|
+
|
27
|
+
def inspect
|
28
|
+
"<#{self.class.name} score: #{score}, word_scores: #{word_scores}, words: #{words}>"
|
29
|
+
end
|
30
|
+
|
31
|
+
private
|
32
|
+
|
33
|
+
#
|
34
|
+
# Greatly penalize words that have multiple of the same black letters
|
35
|
+
# Greatly penalize words that have the same black letters as any seen word
|
36
|
+
# Greatly penalize words that have the same yellow letter/index pair as any seen word
|
37
|
+
# Reward words that have yellow letters that match yellow letters in seen words, but in different positions
|
38
|
+
# Reward words that have yellow letters that match green letters in seen words
|
39
|
+
# Penalize words that have yellow letters that don't appear in seen words
|
40
|
+
# Penalize words that have green letters that don't appear in seen words
|
41
|
+
# Rewards words based on commonality
|
42
|
+
# Reward/penalize words based on line indexes and common letters
|
43
|
+
#
|
44
|
+
def select_words_with_scores
|
45
|
+
selected_words = [@start_word]
|
46
|
+
selected_word_scores = [@start_word.score]
|
47
|
+
seen_black_chars = @start_word.black_chars
|
48
|
+
seen_yellow_chars = @start_word.yellow_chars
|
49
|
+
seen_green_chars = @start_word.green_chars
|
50
|
+
seen_yellow_char_index_pairs = @start_word.yellow_char_index_pairs
|
51
|
+
@word_positions.each do |word_position|
|
52
|
+
potential_words = word_position.potential_words
|
53
|
+
potential_words = word_position.frequent_potential_words if potential_words.empty?
|
54
|
+
words_with_score_array = potential_words.map do |word|
|
55
|
+
word_score = word.score
|
56
|
+
next([word, word_score]) if word_score.negative?
|
57
|
+
next([word, -95]) unless (seen_black_chars & word.black_chars).empty?
|
58
|
+
next([word, -90]) unless (seen_yellow_char_index_pairs & word.yellow_char_index_pairs).empty?
|
59
|
+
|
60
|
+
word_score += (seen_yellow_chars & word.yellow_chars).count
|
61
|
+
word_score += (seen_green_chars & word.yellow_chars).count
|
62
|
+
word_score -= (word.yellow_chars - seen_yellow_chars - seen_green_chars).count
|
63
|
+
word_score -= (word.green_chars - seen_green_chars).count
|
64
|
+
[word, word_score]
|
65
|
+
end
|
66
|
+
|
67
|
+
best_word, best_score = words_with_score_array.max_by { _2 }
|
68
|
+
selected_words << best_word
|
69
|
+
selected_word_scores << best_score
|
70
|
+
seen_black_chars.concat(best_word.black_chars)
|
71
|
+
seen_yellow_chars.concat(best_word.yellow_chars)
|
72
|
+
seen_green_chars.concat(best_word.green_chars)
|
73
|
+
seen_yellow_char_index_pairs.concat(best_word.yellow_char_index_pairs)
|
74
|
+
end
|
75
|
+
|
76
|
+
selected_words.zip(selected_word_scores)
|
77
|
+
end
|
78
|
+
|
79
|
+
def normalize_confidence_score(word, score)
|
80
|
+
(word.confidence_score + score).clamp(word.confidence_score, 99)
|
81
|
+
end
|
82
|
+
end
|
83
|
+
end
|
@@ -0,0 +1,122 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class WordleDecoder
|
4
|
+
class Word
|
5
|
+
def initialize(word_str, word_position, commonality = nil)
|
6
|
+
@word_str = word_str
|
7
|
+
@word_position = word_position
|
8
|
+
@commonality = commonality
|
9
|
+
end
|
10
|
+
|
11
|
+
def score
|
12
|
+
@score ||= commonality_score + common_letter_score +
|
13
|
+
frequency_score - pentalty_score
|
14
|
+
end
|
15
|
+
|
16
|
+
def pentalty_score
|
17
|
+
guessed_same_letter_twice? ? 100 : 0
|
18
|
+
end
|
19
|
+
|
20
|
+
def confidence_score(guess_score)
|
21
|
+
position_score = @word_position.confidence_score
|
22
|
+
(position_score + guess_score).clamp(position_score, 99).round
|
23
|
+
end
|
24
|
+
|
25
|
+
def chars
|
26
|
+
@chars ||= @word_str.split("")
|
27
|
+
end
|
28
|
+
|
29
|
+
COMMON_LETTERS = %w[s e a o r i l t n].freeze
|
30
|
+
PENALTY_LETTERS_COUNT = 5
|
31
|
+
|
32
|
+
def common_letter_score
|
33
|
+
if @word_position.line_index <= 1
|
34
|
+
[(chars & COMMON_LETTERS).count, 3].min
|
35
|
+
else
|
36
|
+
letters_count = PENALTY_LETTERS_COUNT + @word_position.line_index
|
37
|
+
-(black_chars & COMMON_LETTERS.first(letters_count)).count
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
COMMONALITY_SCORES = { most: 2, less: 1, least: 0 }.freeze
|
42
|
+
|
43
|
+
def commonality_score
|
44
|
+
COMMONALITY_SCORES[@commonality] || 0
|
45
|
+
end
|
46
|
+
|
47
|
+
def frequency_score
|
48
|
+
@frequency_score ||= WordSearch.frequency_score(@word_str)
|
49
|
+
end
|
50
|
+
|
51
|
+
def green_chars
|
52
|
+
@green_chars ||= find_chars(@word_position.green_letter_positions)
|
53
|
+
end
|
54
|
+
|
55
|
+
def yellow_chars
|
56
|
+
@yellow_chars ||= find_chars(@word_position.yellow_letter_positions)
|
57
|
+
end
|
58
|
+
|
59
|
+
def yellow_char_index_pairs
|
60
|
+
@yellow_char_index_pairs ||= find_char_index_pairs(@word_position.yellow_letter_positions)
|
61
|
+
end
|
62
|
+
|
63
|
+
def black_chars
|
64
|
+
@black_chars ||= find_chars(@word_position.black_letter_positions)
|
65
|
+
end
|
66
|
+
|
67
|
+
def possible?
|
68
|
+
return true if yellow_chars.empty?
|
69
|
+
|
70
|
+
answer_chars = @word_position.answer_chars.dup
|
71
|
+
delete_green_chars!(answer_chars)
|
72
|
+
|
73
|
+
yellow_chars.all? do |yellow_char|
|
74
|
+
answer_char_index = answer_chars.index(yellow_char)
|
75
|
+
answer_chars.delete_at(answer_char_index) if answer_char_index
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
def to_s
|
80
|
+
@word_str
|
81
|
+
end
|
82
|
+
|
83
|
+
def to_terminal
|
84
|
+
@word_position.letter_positions.map do |letter_position|
|
85
|
+
char = @word_str[letter_position.index]
|
86
|
+
case letter_position.hint_char
|
87
|
+
when "g"
|
88
|
+
"{{green:#{char}}}"
|
89
|
+
when "y"
|
90
|
+
"{{yellow:#{char}}}"
|
91
|
+
else
|
92
|
+
char
|
93
|
+
end
|
94
|
+
end.join
|
95
|
+
end
|
96
|
+
|
97
|
+
private
|
98
|
+
|
99
|
+
def guessed_same_letter_twice?
|
100
|
+
black_chars.count != black_chars.uniq.count || !(black_chars & yellow_chars).empty?
|
101
|
+
end
|
102
|
+
|
103
|
+
def delete_green_chars!(answer_chars)
|
104
|
+
green_chars&.each do |green_char|
|
105
|
+
answer_char_index = answer_chars.index(green_char)
|
106
|
+
answer_chars.delete_at(answer_char_index) if answer_char_index
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
def find_chars(letter_positions)
|
111
|
+
return [] unless letter_positions
|
112
|
+
|
113
|
+
letter_positions.map { |lp| @word_str[lp.index] }
|
114
|
+
end
|
115
|
+
|
116
|
+
def find_char_index_pairs(letter_positions)
|
117
|
+
return [] unless letter_positions
|
118
|
+
|
119
|
+
letter_positions.map { |lp| [@word_str[lp.index], lp.index] }
|
120
|
+
end
|
121
|
+
end
|
122
|
+
end
|
@@ -0,0 +1,150 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class WordleDecoder
|
4
|
+
class WordPosition
|
5
|
+
EMOJI_HINT_CHARS = { "⬛" => "b",
|
6
|
+
"⬜" => "b",
|
7
|
+
"🟨" => "y",
|
8
|
+
"🟩" => "g" }.freeze
|
9
|
+
|
10
|
+
def initialize(hint_line, line_index, answer_chars)
|
11
|
+
@hint_chars = normalize_hint_chars(hint_line)
|
12
|
+
@answer_chars = answer_chars
|
13
|
+
@line_index = line_index
|
14
|
+
@letter_positions = initialize_letter_positions(@hint_chars, @answer_chars)
|
15
|
+
end
|
16
|
+
|
17
|
+
attr_reader :hint_chars,
|
18
|
+
:answer_chars,
|
19
|
+
:line_index,
|
20
|
+
:letter_positions
|
21
|
+
|
22
|
+
def potential_words
|
23
|
+
@potential_words ||= initialize_potential_words
|
24
|
+
end
|
25
|
+
|
26
|
+
def frequent_potential_words
|
27
|
+
@frequent_potential_words ||= find_10_frequent_potential_words
|
28
|
+
end
|
29
|
+
|
30
|
+
BASE_INCONFIDENCE = 0.05
|
31
|
+
|
32
|
+
def confidence_score
|
33
|
+
return 1 if potential_words.empty?
|
34
|
+
|
35
|
+
score = (100 * (1.0 / potential_words.count.to_f))
|
36
|
+
(score - (score * BASE_INCONFIDENCE)).round
|
37
|
+
end
|
38
|
+
|
39
|
+
def green_letter_positions
|
40
|
+
@green_letter_positions ||= select_letter_positions_by_hint("g")
|
41
|
+
end
|
42
|
+
|
43
|
+
def yellow_letter_positions
|
44
|
+
@yellow_letter_positions ||= select_letter_positions_by_hint("y")
|
45
|
+
end
|
46
|
+
|
47
|
+
def black_letter_positions
|
48
|
+
@black_letter_positions ||= select_letter_positions_by_hint("b")
|
49
|
+
end
|
50
|
+
|
51
|
+
private
|
52
|
+
|
53
|
+
def initialize_potential_words
|
54
|
+
potential_words = []
|
55
|
+
WordSearch::COMMONALITY_OPTIONS.each do |commonality|
|
56
|
+
word_strings = compute_words_from_hints(commonality)
|
57
|
+
next if word_strings.empty?
|
58
|
+
|
59
|
+
word_strings = remove_impossible_words(word_strings, commonality)
|
60
|
+
next if word_strings.empty?
|
61
|
+
|
62
|
+
new_words = word_strings.map! { |str| Word.new(str, self, commonality) }
|
63
|
+
new_words.select!(&:possible?)
|
64
|
+
potential_words.concat(new_words)
|
65
|
+
end
|
66
|
+
potential_words
|
67
|
+
end
|
68
|
+
|
69
|
+
def compute_words_from_hints(commonality)
|
70
|
+
words = nil
|
71
|
+
[green_letter_positions, yellow_letter_positions].each do |letters|
|
72
|
+
words = filter_by_words(words, letters, commonality) if letters
|
73
|
+
end
|
74
|
+
words || []
|
75
|
+
end
|
76
|
+
|
77
|
+
def filter_by_words(words, letters, commonality)
|
78
|
+
letters.each do |letter|
|
79
|
+
if words
|
80
|
+
words &= letter.potential_words(commonality)
|
81
|
+
else
|
82
|
+
words = letter.potential_words(commonality)
|
83
|
+
end
|
84
|
+
end
|
85
|
+
words
|
86
|
+
end
|
87
|
+
|
88
|
+
def remove_impossible_words(words, commonality)
|
89
|
+
black_letter_positions&.each do |letter|
|
90
|
+
words -= letter.impossible_words(commonality)
|
91
|
+
end
|
92
|
+
words
|
93
|
+
end
|
94
|
+
|
95
|
+
def select_letter_positions_by_hint(hint_char)
|
96
|
+
@letter_positions.select { |lg| lg.hint_char == hint_char }
|
97
|
+
end
|
98
|
+
|
99
|
+
def find_10_frequent_potential_words
|
100
|
+
word_strings = WordSearch.most_frequent_words_without_chars(@answer_chars, 10)
|
101
|
+
word_strings.map { |str| Word.new(str, self) }
|
102
|
+
end
|
103
|
+
|
104
|
+
def normalize_hint_chars(hint_line)
|
105
|
+
hint_line.each_char.map { |c| EMOJI_HINT_CHARS[c] || c }
|
106
|
+
end
|
107
|
+
|
108
|
+
def initialize_letter_positions(hint_chars, answer_chars)
|
109
|
+
hint_chars.each_with_index.map do |hint_char, index|
|
110
|
+
LetterPosition.new(index, hint_char, hint_chars, answer_chars)
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
114
|
+
class LetterPosition
|
115
|
+
def initialize(index, hint_char, hint_chars, answer_chars)
|
116
|
+
@index = index
|
117
|
+
@hint_char = hint_char
|
118
|
+
@hint_chars = hint_chars
|
119
|
+
@answer_chars = answer_chars
|
120
|
+
@answer_char = @answer_chars[index]
|
121
|
+
end
|
122
|
+
|
123
|
+
attr_reader :hint_char,
|
124
|
+
:answer_char,
|
125
|
+
:index
|
126
|
+
|
127
|
+
def potential_words(commonality)
|
128
|
+
case hint_char
|
129
|
+
when "g"
|
130
|
+
WordSearch.char_at_index(@answer_char, @index, commonality)
|
131
|
+
when "y"
|
132
|
+
if @hint_chars.count("g") == 3
|
133
|
+
must_be_char_index = @hint_chars.index.with_index { |h, i| h != "g" && i != @index }
|
134
|
+
must_be_char = @answer_chars[must_be_char_index]
|
135
|
+
WordSearch.char_at_index(must_be_char, @index, commonality)
|
136
|
+
else
|
137
|
+
# TODO: if yellow char is in the word as green already, and there's only one isntance of it
|
138
|
+
# in final word, can assume it's not that char.
|
139
|
+
chars = @answer_chars - [@answer_char]
|
140
|
+
WordSearch.chars_at_index(chars, @index, commonality)
|
141
|
+
end
|
142
|
+
end
|
143
|
+
end
|
144
|
+
|
145
|
+
def impossible_words(commonality)
|
146
|
+
WordSearch.chars_at_index(@answer_chars, @index, commonality)
|
147
|
+
end
|
148
|
+
end
|
149
|
+
end
|
150
|
+
end
|
@@ -0,0 +1,76 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'json'
|
4
|
+
|
5
|
+
class WordleDecoder
|
6
|
+
class WordSearch
|
7
|
+
COMMONALITY_OPTIONS = %i[most less least].freeze
|
8
|
+
|
9
|
+
class << self
|
10
|
+
def char_at_index(char, index, commonality)
|
11
|
+
case commonality
|
12
|
+
when :most
|
13
|
+
most_common_letter_to_words_arrays[index][char]
|
14
|
+
when :less
|
15
|
+
less_common_letter_to_words_arrays[index][char]
|
16
|
+
when :least
|
17
|
+
least_common_letter_to_words_arrays[index][char]
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
def chars_at_index(chars, index, commonality)
|
22
|
+
chars.uniq.flat_map { |c| char_at_index(c, index, commonality) }
|
23
|
+
end
|
24
|
+
|
25
|
+
def frequency_score(word)
|
26
|
+
words_to_frequency_score_hash[word]
|
27
|
+
end
|
28
|
+
|
29
|
+
def words_to_frequency_score_hash
|
30
|
+
@words_to_frequency_score_hash ||= load_words_to_frequency_score_hash
|
31
|
+
end
|
32
|
+
|
33
|
+
def most_frequent_words_without_chars(without_chars, limit)
|
34
|
+
regex = /#{without_chars.join("|")}/
|
35
|
+
words = []
|
36
|
+
words_to_frequency_score_hash.each_key.reverse_each do |str|
|
37
|
+
words << str unless str.match?(regex)
|
38
|
+
|
39
|
+
return(words) if words.count == limit
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
private
|
44
|
+
|
45
|
+
def load_words_to_frequency_score_hash
|
46
|
+
JSON.parse(file_path("words_to_frequency_score.json"))
|
47
|
+
end
|
48
|
+
|
49
|
+
def most_common_letter_to_words_arrays
|
50
|
+
@most_common_letter_to_words_arrays ||= build_letter_to_words_hashes("most_common_words.txt")
|
51
|
+
end
|
52
|
+
|
53
|
+
def less_common_letter_to_words_arrays
|
54
|
+
@less_common_letter_to_words_arrays ||= build_letter_to_words_hashes("less_common_words.txt")
|
55
|
+
end
|
56
|
+
|
57
|
+
def least_common_letter_to_words_arrays
|
58
|
+
@least_common_letter_to_words_arrays ||= build_letter_to_words_hashes("least_common_words.txt")
|
59
|
+
end
|
60
|
+
|
61
|
+
def build_letter_to_words_hashes(words_file_name)
|
62
|
+
words = load_wordle_words(words_file_name)
|
63
|
+
Array.new(5) { |index| words.group_by { |word| word[index] } }
|
64
|
+
end
|
65
|
+
|
66
|
+
def load_wordle_words(file_name)
|
67
|
+
file_path(file_name).split("\n")
|
68
|
+
end
|
69
|
+
|
70
|
+
def file_path(file_name)
|
71
|
+
file_path = File.join(File.dirname(__FILE__), "..", file_name)
|
72
|
+
File.read(file_path)
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
@@ -0,0 +1,122 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class WordleDecoder
|
4
|
+
class WordleShare
|
5
|
+
ANSWER_LINES = ["🟩🟩🟩🟩🟩", "ggggg"].freeze
|
6
|
+
|
7
|
+
def self.final_line?(input_lines)
|
8
|
+
ANSWER_LINES.any? { input_lines.include?(_1) }
|
9
|
+
end
|
10
|
+
|
11
|
+
def self.wordle_answers
|
12
|
+
@wordle_answers ||= load_worldle_ansers
|
13
|
+
end
|
14
|
+
|
15
|
+
def self.load_worldle_ansers
|
16
|
+
file_path = File.join(File.dirname(__FILE__), "..", "wordle_answers.json")
|
17
|
+
JSON.parse File.read(file_path)
|
18
|
+
end
|
19
|
+
|
20
|
+
def initialize(input, answer_input = nil)
|
21
|
+
@input = input
|
22
|
+
self.answer_input = answer_input
|
23
|
+
end
|
24
|
+
|
25
|
+
attr_reader :input,
|
26
|
+
:answer_input
|
27
|
+
|
28
|
+
attr_accessor :answer
|
29
|
+
|
30
|
+
def answer_input=(val)
|
31
|
+
@answer_input = val
|
32
|
+
@answer = normalize_answer_input(val)
|
33
|
+
end
|
34
|
+
|
35
|
+
GAME_DAY_REGEX = /wordle\s(\d+)\s/i.freeze
|
36
|
+
|
37
|
+
def find_answer
|
38
|
+
title_line = input_lines.detect { |line| line.match?(GAME_DAY_REGEX) }
|
39
|
+
game_day = title_line.match(GAME_DAY_REGEX).captures.first&.to_i
|
40
|
+
self.answer = self.class.wordle_answers[game_day]
|
41
|
+
end
|
42
|
+
|
43
|
+
def answer_chars
|
44
|
+
@answer_chars ||= answer&.strip&.split("")
|
45
|
+
end
|
46
|
+
|
47
|
+
def hint_lines
|
48
|
+
@hint_lines ||= parse_hint_lines!
|
49
|
+
end
|
50
|
+
|
51
|
+
def wordle_lines
|
52
|
+
@wordle_lines ||= parse_wordle_lines!
|
53
|
+
end
|
54
|
+
|
55
|
+
def input_lines
|
56
|
+
@input_lines ||= normalize_input_lines(parse_input_lines!)
|
57
|
+
end
|
58
|
+
|
59
|
+
def to_terminal
|
60
|
+
"{{blue:>}} #{wordle_lines.join("\n ")}"
|
61
|
+
end
|
62
|
+
|
63
|
+
def decoder
|
64
|
+
@decoder ||= WordleDecoder.new(self)
|
65
|
+
end
|
66
|
+
|
67
|
+
def inspect
|
68
|
+
"<#{self.class.name} input: #{input}, answer_input: #{answer_input}" \
|
69
|
+
" answer_chars: #{answer_chars.inspect} hint_lines: #{hint_lines.inspect}>"
|
70
|
+
end
|
71
|
+
|
72
|
+
private
|
73
|
+
|
74
|
+
def parse_hint_lines!
|
75
|
+
hint_lines = wordle_lines.dup
|
76
|
+
hint_lines.pop if ANSWER_LINES.include?(hint_lines.last)
|
77
|
+
hint_lines
|
78
|
+
end
|
79
|
+
|
80
|
+
VALID_HINT_CHARS = WordPosition::EMOJI_HINT_CHARS.to_a.flatten.uniq
|
81
|
+
|
82
|
+
def parse_wordle_lines!
|
83
|
+
input_lines.select do |line|
|
84
|
+
line.each_char.all? { |c| VALID_HINT_CHARS.include?(c) }
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
def parse_input_lines!
|
89
|
+
case input
|
90
|
+
when String
|
91
|
+
convert_input_string_to_input_lines
|
92
|
+
when Array
|
93
|
+
input
|
94
|
+
else
|
95
|
+
raise Error, "Input must be a String or Array"
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
def normalize_input_lines(input_lines)
|
100
|
+
input_lines.filter_map do |input_line|
|
101
|
+
line = input_line.strip
|
102
|
+
line unless line.empty?
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
def convert_input_string_to_input_lines
|
107
|
+
hint_input = input.downcase
|
108
|
+
if hint_input.include?("\n")
|
109
|
+
hint_input.split("\n")
|
110
|
+
else
|
111
|
+
hint_input.chars.each_slice(5).map(&:join)
|
112
|
+
end
|
113
|
+
end
|
114
|
+
|
115
|
+
VALID_ANSWER_CHARS = ("a".."z").freeze
|
116
|
+
|
117
|
+
def normalize_answer_input(val)
|
118
|
+
val = val&.strip&.downcase
|
119
|
+
val if val && val.length == 5 && val.each_char.all? { |c| VALID_ANSWER_CHARS.include?(c) }
|
120
|
+
end
|
121
|
+
end
|
122
|
+
end
|