#!/usr/bin/python3 """ Jour 12 du défi Advent Of Code pour l'année 2023 """ import os from functools import cache from aoc_utils import decorators def read_sample(): """récupère les entrées depuis le fichier texte correspondant""" filename = os.path.join(os.path.dirname(__file__ ), "inputs", "day12.txt") with open(filename, 'r') as f: sample = f.read().split('\n') sample = [ i for i in sample if i != '' ] return sample def parse_sample(sample): springs = [] counts = [] for i in range(len(sample)): springs.append(sample[i].split(" ")[0]) counts.append([int(i) for i in sample[i].split(" ")[1].split(",")]) return springs, counts def nb_poss(spring, count): def check_new_elem(elem, size, count): if size > len(count): return False if elem != count[size-1]: return False return True @cache def nb_poss_rec(actual_count_size, cur, pos, first): """ actual_count_size: already added elements cur: current number of # in the spring pos pos: current position in the spring first: first elem of the upcoming spring """ # While loop for first elem if pos < len(spring) and first != '?': if first == '.': if cur > 0: actual_count_size += 1 if not check_new_elem(cur, actual_count_size, count): return 0 cur = 0 elif first == '#': cur += 1 pos += 1 # Exactly the same loop for the other elems while pos < len(spring) and spring[pos] != '?': if spring[pos] == '.': if cur > 0: actual_count_size += 1 if not check_new_elem(cur, actual_count_size, count): return 0 cur = 0 elif spring[pos] == '#': cur += 1 pos += 1 # spring ends with # ? if pos == len(spring) and cur > 0: actual_count_size += 1 if not check_new_elem(cur, actual_count_size, count): return 0 # End of spring ? if pos >= len(spring): if len(count) == actual_count_size: return 1 return 0 # Debugging if spring[pos] != '?': print("weird", spring, count, actual_count_size, pos, spring[pos], len(spring)) return # Recursive calls if spring[pos] == '?': if cur > 0: if len(count) > actual_count_size: if cur == count[actual_count_size]: return nb_poss_rec(actual_count_size, cur, pos, '.') # break the #### with a . elif cur < count[actual_count_size]: return nb_poss_rec(actual_count_size, cur, pos, '#') # continue with # else: return 0 # not possible return nb_poss_rec(actual_count_size, cur, pos, '.') + nb_poss_rec(actual_count_size, cur, pos, '#') return nb_poss_rec(0, 0, 0, spring[0]) def unfold(springs, counts): for i in range(len(springs)): springs[i] = '?'.join([springs[i]]*5) counts[i] = counts[i]*5 @decorators.timeit def part1(sample): """Partie 1 du défi""" springs, counts = parse_sample(sample) possibilities = [] for i in range(len(springs)): possibilities.append(nb_poss(springs[i], counts[i])) return sum(possibilities) @decorators.timeit def part2(sample): """Partie 2 du défi""" springs, counts = parse_sample(sample) unfold(springs, counts) possibilities = [] for i in range(len(springs)): possibilities.append(nb_poss(springs[i], counts[i])) return sum(possibilities) def main(): """Fonction principale""" sample = read_sample() print(f"part1: {part1(sample)}") print(f"part2: {part2(sample)}") if __name__ == "__main__": main()