aboutsummaryrefslogtreecommitdiff
path: root/racket/aoc2020/day-07/day-07.rkt
blob: f2a1ffe1263f971f768b6d58a864e76ff7a63843 (plain)
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
#lang racket
(require advent-of-code
         threading
         rebellion/collection/entry
         rebellion/collection/multidict)

(define raw-rules (~> (open-aoc-input (find-session) 2020 7) (port->string) (string-split "\n")))

(define (split-rule r)
  (match-define (list head tail) (string-split (string-trim r #px" bags?.") " bags contain "))
  (~>> tail
       (regexp-split #px"( bags?,\\s)" _)
       (map (λ~> (regexp-match* #px"(\\d+) (\\w+ \\w+)" _ #:match-select rest)))
       append*
       (map (λ~> (match _
                   [(list n c) (list (string->symbol c) (string->number n))]
                   ['() '()])))
       (cons (string->symbol head) _)))

(define rules-multidict
  (for*/multidict ([ln (in-list raw-rules)] #:do [(match-define (list* from tos) (split-rule ln))]
                                            [to (in-list tos)])
                  (entry from to)))

;; part 1
(define (bags-that-eventually-contain target)
  (for/fold ([holders (set)]) ([rule (in-multidict-entries rules-multidict)])
    (match rule
      [(entry outside (list (== target) _))
       (set-union (set-add holders outside) (bags-that-eventually-contain outside))]
      [_ holders])))

(define part-1 (set-count (bags-that-eventually-contain '|shiny gold|)))
(~a "Part 1: " part-1)
;; (aoc-submit (find-session) 2020 7 1 part-1)

;; part 2
(define (bags-that-are-contained-by target)
  (for/sum ([holding (in-multidict-entries rules-multidict)])
           (match holding
             [(entry (== target) (list held n)) (+ n (* n (bags-that-are-contained-by held)))]
             [_ 0])))

(define part-2 (bags-that-are-contained-by '|shiny gold|))
(~a "Part 2: " part-2)
;; (aoc-submit (find-session) 2020 7 1 part-2)