-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path21.1.py
52 lines (47 loc) · 1.88 KB
/
21.1.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
import re
import numpy as np
from math import prod
with open('input/21.txt') as f:
content = [line.strip() for line in f.readlines()]
foods = {}
allergens = {}
ingredient_names = set()
all_ingredients = []
for x, line in enumerate(content):
foods[x] = {}
allergen_index = line.index('(')
foods[x]['ingredients'] = line[:allergen_index].split()
ingredient_names.update(set(foods[x]['ingredients']))
all_ingredients.extend(foods[x]['ingredients'])
foods[x]['allergens'] = line[allergen_index+len('contains ')+1:-1].split(', ')
# build up our allergens with possible values
for i, food in foods.items():
for allergen in food['allergens']:
if allergen not in allergens:
allergens[allergen] = set(food['ingredients'])
else:
allergens[allergen] = allergens[allergen].intersection(set(food['ingredients']))
# sort them by length of possibilities, should begin with a 1:1 match
sorted_allergens = [(k, allergens[k]) for k in sorted(allergens, key=lambda a: len(allergens[a]))]
while any(len(v) > 1 for v in allergens.values()):
for allergen, ingredients in sorted_allergens:
# allergen, ingredient = [(i, next(iter(v))) for i, v in allergens.items() if len(v) == 1][0]
if len(ingredients) == 1:
ingredient = next(iter(ingredients))
# remove ingredient from other allergens
for a in allergens.keys():
if a == allergen:
continue
try:
allergens[a].remove(ingredient)
except:
continue
# find the safe ingredients
for i, food in foods.items():
for allergen, ingredients in allergens.items():
ingredient = next(iter(ingredients))
if ingredient in all_ingredients:
all_ingredients = list(filter(lambda a: a != ingredient, all_ingredients))
# count the remaining occurences for each ingredient
ingredient_count = {k: all_ingredients.count(k) for k in ingredient_names}
print(sum(ingredient_count.values()))