Powered by AppSignal & Oban Pro
Would you like to see your link here? Contact us

Advent of Code 2021 - Day 12

advent-of-code/2021/day12.livemd

Advent of Code 2021 - Day 12

Mix.install([
  {:kino, "~> 0.6.1"},
  {:libgraph, "~> 0.13.3"}
])

alias :digraph, as: Digraph

Instructions

https://adventofcode.com/2021/day/12

Test Input

test_input1 = """
start-A
start-b
A-c
A-b
b-d
A-end
b-end
"""
test_input2 = """
dc-end
HN-start
start-kj
dc-start
dc-HN
LN-dc
HN-end
kj-sa
kj-HN
kj-dc
"""
test_input3 = """
fs-end
he-DX
fs-he
start-DX
pj-DX
end-zg
zg-sl
zg-pj
pj-he
RW-he
fs-DX
pj-RW
zg-RW
start-pj
he-WI
zg-he
pj-fs
start-RW
"""

Puzzle Input

Paste your own input from Advent of Code or copy from sdball 2021 Advent of Code Day 12 input

puzzle_text = Kino.Input.textarea("Please paste your puzzle input:")
puzzle_input = Kino.Input.read(puzzle_text)

——— Exploration ———

Test Input - Exploring Erlang digraph

graph = Digraph.new()
cave_start = Digraph.add_vertex(graph, "start")
cave_end = Digraph.add_vertex(graph, "end")
cave_A = Digraph.add_vertex(graph, "A")
cave_b = Digraph.add_vertex(graph, "b")
cave_c = Digraph.add_vertex(graph, "c")
cave_d = Digraph.add_vertex(graph, "d")

Digraph.add_edge(graph, cave_start, cave_A)
Digraph.add_edge(graph, cave_start, cave_b)
Digraph.add_edge(graph, cave_A, cave_c)
Digraph.add_edge(graph, cave_A, cave_b)
Digraph.add_edge(graph, cave_b, cave_d)
Digraph.add_edge(graph, cave_A, cave_end)
Digraph.add_edge(graph, cave_b, cave_end)

Digraph.get_path(graph, cave_start, cave_end)
Digraph.vertices(graph)
Digraph.edges(graph)

Test Input - Exploring Elixir libgraph

graph = Graph.new()
graph1 =
  test_input1
  |> String.split("\n", trim: true)
  |> Enum.map(fn connection ->
    String.split(connection, "-")
  end)
  |> Enum.reduce(Graph.new(), fn [cave, exit], graph ->
    graph =
      graph
      |> Graph.add_vertices([cave, exit])
      |> Graph.add_edge(cave, exit)

    if cave != "start" and exit != "end" do
      Graph.add_edge(graph, exit, cave)
    else
      graph
    end
  end)
  |> then(fn graph ->
    graph
    |> Graph.vertices()
    |> Enum.reduce(graph, fn cave, graph ->
      if cave == String.downcase(cave) do
        Graph.label_vertex(graph, cave, :small)
      else
        graph
      end
    end)
  end)
Graph.info(graph1)
Graph.edges(graph1)
Graph.neighbors(graph1, "start")
Graph.out_edges(graph1, "start")
Graph.out_edges(graph1, "A")
Graph.get_paths(graph1, "start", "end")
Graph.reachable_neighbors(graph1, ["start"])
Graph.get_paths(graph1, "d", "end")
Graph.vertices(graph1)
|> Enum.reject(fn cave -> cave in ~w(start end) end)
|> Enum.reduce([], fn cave, paths ->
  Graph.get_paths(graph1, "start", cave) ++ paths
end)
|> Enum.map(fn path ->
  [last | leading] = Enum.reverse(path)

  Graph.get_paths(graph1, last, "end")
  |> Enum.map(fn path ->
    Enum.reverse(leading) ++ path
  end)
end)
|> Enum.reduce([], fn ps, paths ->
  paths ++ ps
end)
|> Enum.uniq()
|> Enum.reject(fn path ->
  Enum.frequencies(path)
  |> Map.filter(fn {cave, visits} ->
    cave == String.downcase(cave) and visits > 1
  end)
  |> Map.keys()
  |> length != 0
end)

Ok, it’s clear I don’t have enough Graph data knowledge to effectively work purely within the Graph data and API. But maybe I can still use it as the data structure for recursion.

Test Input - Recursion over Graph

graph1
Graph.out_edges(graph1, "start")
Graph.out_neighbors(graph1, "start")
Graph.vertex_labels(graph1, "b")

VerboseCaveSystem Module

defmodule VerboseCaveSystem do
  def count_paths(graph) do
    outs = Graph.out_neighbors(graph, "start")
    count_paths(outs, graph, MapSet.new(), ["start"], 0)
  end

  def count_paths(["end" | caves], graph, visited_small_caves, path, path_count) do
    IO.puts(
      "I reached the end of the caves! My path so far #{inspect(Enum.reverse(["end" | path]))}"
    )

    IO.puts("EXITED\n")
    count_paths(caves, graph, visited_small_caves, path, path_count + 1)
  end

  def count_paths([cave | caves], graph, visited_small_caves, path, path_count) do
    path_count =
      cond do
        cave in visited_small_caves ->
          IO.puts(
            "Oh no I'm back in the small cave #{cave}. My path so far #{inspect(Enum.reverse([cave | path]))}"
          )

          IO.puts("END OF PATH\n")
          path_count

        :small in Graph.vertex_labels(graph, cave) ->
          IO.puts(
            "Oh nice, a small cave called #{cave}. My path so far #{inspect(Enum.reverse([cave | path]))}"
          )

          outs = Graph.out_neighbors(graph, cave)

          count_paths(
            outs,
            graph,
            MapSet.put(visited_small_caves, cave),
            [cave | path],
            path_count
          )

        true ->
          IO.puts(
            "I am in a large cave called #{cave}. My path so far #{inspect(Enum.reverse([cave | path]))}"
          )

          outs = Graph.out_neighbors(graph, cave)
          count_paths(outs, graph, visited_small_caves, [cave | path], path_count)
      end

    if path_count > 0 do
      IO.puts("Wow I found #{path_count} paths to the exit!")

      if length(caves) > 0 do
        IO.puts("But I have more caves to explore: #{inspect(caves)}\n")
      end
    else
      IO.puts("All those paths and I didn't find any new ways to the exit.")
      IO.puts("But I have more caves to explore: #{inspect(caves)}\n")
    end

    count_paths(caves, graph, visited_small_caves, path, path_count)
  end

  def count_paths(_caves = [], _graph, _visited_small_caves, _path, path_count), do: path_count
end
VerboseCaveSystem.count_paths(graph1)

——— Solving ———

Input Parser

defmodule Input do
  def to_graph(input) do
    input
    |> String.split("\n", trim: true)
    |> Enum.map(fn connection ->
      String.split(connection, "-")
    end)
    |> Enum.reduce(Graph.new(), fn [cave1, cave2], graph ->
      graph
      |> Graph.add_vertices([cave1, cave2])
      |> maybe_add_edge(cave1, cave2)
      |> maybe_add_edge(cave2, cave1)
    end)
    |> then(fn graph ->
      graph
      |> Graph.vertices()
      |> Enum.reduce(graph, fn cave1, graph ->
        if cave1 == String.downcase(cave1) do
          Graph.label_vertex(graph, cave1, :small)
        else
          graph
        end
      end)
    end)
  end

  defp maybe_add_edge(graph, "end", _), do: graph
  defp maybe_add_edge(graph, _, "start"), do: graph

  defp maybe_add_edge(graph, cave1, cave2) do
    Graph.add_edge(graph, cave1, cave2)
  end
end

——— Part 1 ———

CaveSystem module

defmodule CaveSystem do
  def count_paths(graph) do
    outs = Graph.out_neighbors(graph, "start")
    count_paths(outs, graph, MapSet.new(), ["start"], 0)
  end

  def count_paths(["end" | caves], graph, visited_small_caves, path, path_count) do
    count_paths(caves, graph, visited_small_caves, path, path_count + 1)
  end

  def count_paths([cave | caves], graph, visited_small_caves, path, path_count) do
    path_count =
      cond do
        cave in visited_small_caves ->
          path_count

        :small in Graph.vertex_labels(graph, cave) ->
          outs = Graph.out_neighbors(graph, cave)

          count_paths(
            outs,
            graph,
            MapSet.put(visited_small_caves, cave),
            [cave | path],
            path_count
          )

        true ->
          outs = Graph.out_neighbors(graph, cave)
          count_paths(outs, graph, visited_small_caves, [cave | path], path_count)
      end

    count_paths(caves, graph, visited_small_caves, path, path_count)
  end

  def count_paths(_caves = [], _graph, _visited_small_caves, _path, path_count), do: path_count
end

Part 1 - Test Input 1

test_input1
|> Input.to_graph()
|> CaveSystem.count_paths()

Part 1 - Test Input 2

test_input2
|> Input.to_graph()
|> CaveSystem.count_paths()

Part 1 - Test Input 3

test_input3
|> Input.to_graph()
|> CaveSystem.count_paths()

Part 1 - Puzzle Input

puzzle_input
|> Input.to_graph()
|> CaveSystem.count_paths()

——— Part 2 ———

OneSmallVisitTwiceCaveSystem Module

defmodule OneSmallVisitTwiceCaveSystem do
  def count_paths(graph) do
    outs = Graph.out_neighbors(graph, "start")
    count_paths(outs, graph, MapSet.new(), false, ["start"], 0)
  end

  def count_paths(["end" | caves], graph, visited_small_caves, twice?, path, path_count) do
    count_paths(caves, graph, visited_small_caves, twice?, path, path_count + 1)
  end

  def count_paths([cave | caves], graph, visited_small_caves, twice?, path, path_count) do
    path_count =
      cond do
        cave in visited_small_caves and twice? ->
          path_count

        cave in visited_small_caves ->
          outs = Graph.out_neighbors(graph, cave)

          count_paths(
            outs,
            graph,
            MapSet.put(visited_small_caves, cave),
            true,
            [cave | path],
            path_count
          )

        :small in Graph.vertex_labels(graph, cave) ->
          outs = Graph.out_neighbors(graph, cave)

          count_paths(
            outs,
            graph,
            MapSet.put(visited_small_caves, cave),
            twice?,
            [cave | path],
            path_count
          )

        true ->
          outs = Graph.out_neighbors(graph, cave)
          count_paths(outs, graph, visited_small_caves, twice?, [cave | path], path_count)
      end

    count_paths(caves, graph, visited_small_caves, twice?, path, path_count)
  end

  def count_paths(_caves = [], _graph, _visited_small_caves, _twice, _path, path_count),
    do: path_count
end

Part 2 - Test Input 1

test_input1
|> Input.to_graph()
|> OneSmallVisitTwiceCaveSystem.count_paths()

Part 2 - Test Input 2

test_input2
|> Input.to_graph()
|> OneSmallVisitTwiceCaveSystem.count_paths()

Part 2 - Test Input 3

test_input3
|> Input.to_graph()
|> OneSmallVisitTwiceCaveSystem.count_paths()

Part 2 - Puzzle Input

puzzle_input
|> Input.to_graph()
|> OneSmallVisitTwiceCaveSystem.count_paths()