123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169 |
- package main
- import (
- "bufio"
- "fmt"
- "log"
- "os"
- "strings"
- )
- type dish struct {
- ingredients []string
- allergens []string
- }
- func readDish(line string) dish {
- var food dish
- parts := strings.Split(line, " (contains ")
- if len(parts) != 2 {
- log.Fatalf("Invalid line: %s", line)
- }
- for _, ing := range strings.Split(parts[0], " ") {
- food.ingredients = append(food.ingredients, ing)
- }
- cleanedPart2 := strings.TrimSuffix(parts[1], ")")
- for _, allergen := range strings.Split(cleanedPart2, ", ") {
- food.allergens = append(food.allergens, allergen)
- }
- return food
- }
- func readFile(file *os.File) []dish {
- var foods []dish
- scanner := bufio.NewScanner(file)
- for scanner.Scan() {
- line := scanner.Text()
- if line == "" {
- break
- }
- foods = append(foods, readDish(line))
- }
- if err := scanner.Err(); err != nil {
- log.Fatalf("Scanner error: %s", err)
- }
- return foods
- }
- type alerg struct {
- count int
- ingredients map[string]int
- }
- func processFoods(foods []dish) map[string]alerg {
- allergensPossibleForIngredients := make(map[string]alerg)
- for _, food := range foods {
- for _, allergen := range food.allergens {
- var currentAllergen alerg
- if _, ok := allergensPossibleForIngredients[allergen]; !ok {
- currentAllergen = alerg{count: 0, ingredients: make(map[string]int)}
- } else {
- currentAllergen = allergensPossibleForIngredients[allergen]
- }
- currentAllergen.count++
- for _, ingredient := range food.ingredients {
- currentAllergen.ingredients[ingredient]++
- }
- allergensPossibleForIngredients[allergen] = currentAllergen
- }
- }
- return allergensPossibleForIngredients
- }
- func findAllergicIngredients(allergensPossibleForIngredients map[string]alerg) map[string][]string {
- highestAllergens := make(map[string][]string)
- for allergen, item := range allergensPossibleForIngredients {
- var highestIngredients []string
- for key, value := range item.ingredients {
- if value == item.count {
- highestIngredients = append(highestIngredients, key)
- }
- }
- highestAllergens[allergen] = highestIngredients
- }
- return highestAllergens
- }
- func analyse(highestAllergens map[string][]string) map[string]bool {
- foundIngredients := make(map[string]bool)
- found := 0
- target := len(highestAllergens)
- for found < target {
- currentAllergens := make(map[string][]string)
- for key, value := range highestAllergens {
- if len(value) == 1 {
- foundIngredients[value[0]] = true
- found++
- continue
- }
- var newList []string
- for _, item := range value {
- if foundIngredients[item] {
- continue
- }
- newList = append(newList, item)
- }
- currentAllergens[key] = newList
- }
- highestAllergens = currentAllergens
- }
- return foundIngredients
- }
- func part1(foods []dish, badIngredients map[string]bool) int {
- sum := 0
- for _, food := range foods {
- for _, ingredient := range food.ingredients {
- if !badIngredients[ingredient] {
- sum++
- }
- }
- }
- return sum
- }
- func main() {
- if len(os.Args) < 2 {
- log.Fatal("You need to specify a file!")
- }
- filePath := os.Args[1]
- file, err := os.Open(filePath)
- if err != nil {
- log.Fatalf("Failed to open %s!\n", filePath)
- }
- foods := readFile(file)
- if err := file.Close(); err != nil {
- log.Fatalf("Failed to close file: %s", err)
- }
- allergens := processFoods(foods)
- highestAllergens := findAllergicIngredients(allergens)
- badIngredients := analyse(highestAllergens)
- fmt.Println("Part1:", part1(foods, badIngredients))
- }
|