On this page:
<day11>
11.1 What’s the next password that meets the criteria?
<day11-setup>
<day11-q1>
11.2 What’s the next valid password after that?
<day11-q2>
11.3 Testing Day 11
<day11-test>
9.0.0.1

11 Day 11🔗ℹ

 (require aoc-racket/day11) package: aoc-racket

The puzzle. Our input is a short alphabetic key that represents a password.

11.1 What’s the next password that meets the criteria?🔗ℹ

Though the password is alphabetic, we can increment it as we would a numerical password, by changing the rightmost letter to the next letter (for instance x to y, y to z). When we reach z, we roll over to a, and “carry over” the surplus by incrementing the letter to the left.

Furthermore, like Day 5, the puzzle provides certain criteria that must be met:

  • The password must have a sequence of three consecutive letters (like bcd).

  • The password may not contain i, o, or l.

  • The password must contain two different, non-overlapping pairs of letters.

As in Day 5, we’ll use regexp-match to implement tests for these conditions. We’ll also use regexp-replace* to build the function that increments a password alphabetically. Then it’s a simple matter of looking at passwords until we find one that works.

The increment-password function works by using the observation that if the password ends in any number of zs, you have to roll them over to a and increment the letter to the left. Otherwise, you can just increment the last letter — which is actually the same rule, but with zero zs. This logic can all be captured in one regular expression — #rx"^(.*?)(.)(z*)$".

The three-consecutive-letters? test works by converting the letters to numbers and creating a list of the differences betweeen adjacent values. Any three consecutive letters will differ by value of 1. So if the list of differences contains the subsequence '(1 1), then the string has three consecutive letters.

(require racket rackunit)
(provide (all-defined-out))
 
(define (increment-password password)
  (define (increment-letter c)
    ((compose1 ~a integer->char add1 char->integer car string->list) c))
 
  (match-define (list _ prefix letter-to-increment trailing-zs)
    (regexp-match #rx"^(.*?)(.)(z*)$" password))
 
  (string-append* (list prefix (increment-letter letter-to-increment)
                        (regexp-replace* #rx"z" trailing-zs "a"))))
 
(define (three-consecutive-letters? str)
  (define ints (map char->integer (string->list str)))
  (let loop ([differences (map - (cdr ints) (drop-right ints 1))])
    (if (empty? differences)
        #f
        (or (list-prefix? '(1 1) differences) (loop (cdr differences))))))
 
(define (no-iol? str)
  (not (regexp-match #rx"[iol]" str)))
 
(define (two-nonoverlapping-doubles? str)
  (regexp-match #px"(\\w)\\1.*?(\\w)\\2" str))
 
(define (valid? password)
  (and (three-consecutive-letters? password)
       (no-iol? password)
       (two-nonoverlapping-doubles? password)))
 
(define (find-next-valid-password starting-password)
  (define candidate-pw (increment-password starting-password))
  (if (valid? candidate-pw)
      candidate-pw
      (find-next-valid-password candidate-pw)))

(define (q1 input-key)
  (find-next-valid-password input-key))

11.2 What’s the next valid password after that?🔗ℹ

We take the answer to question 1 and use it as input to the same function.

(define (q2 input-key)
  (find-next-valid-password (q1 input-key)))

11.3 Testing Day 11🔗ℹ

(module+ test
  (define input-key (file->string "day11-input.txt"))
  (check-equal? (q1 input-key) "hxbxxyzz")
  (check-equal? (q2 input-key) "hxcaabcc"))