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

Hill Climbing Algorithm

livebook/day12.livemd

Hill Climbing Algorithm

Mix.install([
  {:kino, "~> 0.8.0"},
  {:kino_vega_lite, "~> 0.1.7"},
  {:vega_lite, "~> 0.1.6"},
  {:libgraph, "~> 0.16.0"}
])

alias VegaLite, as: Vl
:ok

Input

input = Kino.Input.textarea("Input:")
input =
  Kino.Input.read(input) |> String.trim() |> String.split("\n") |> Enum.map(&String.to_charlist/1)

grid =
  input
  |> Enum.with_index()
  |> Enum.reduce({%{}, nil, nil}, fn {line, row}, acc ->
    line
    |> Enum.with_index()
    |> Enum.reduce(
      acc,
      fn
        {?S, col}, _acc = {map, _s, e} ->
          {Map.put(map, {row, col}, ?a), {row, col}, e}

        {?E, col}, _acc = {map, s, _e} ->
          {Map.put(map, {row, col}, ?z), s, {row, col}}

        {ch, col}, _acc = {map, s, e} ->
          {Map.put(map, {row, col}, ch), s, e}
      end
    )
  end)
{map, start, end_} = grid

graph =
  Enum.reduce(map, Graph.new(type: :directed), fn {{row, col}, height}, graph ->
    graph |> Graph.add_vertex({row, col}, pos: {row, col}, height: height)
  end)

graph =
  Enum.reduce(map, graph, fn {{row, col}, height}, graph ->
    neighbours =
      [{row - 1, col}, {row, col + 1}, {row + 1, col}, {row, col - 1}]
      |> Enum.filter(fn {r, c} ->
        neighbour_height = Map.get(map, {r, c}, nil)
        neighbour_height != nil and neighbour_height <= height + 1
      end)

    edges = for n <- neighbours, do: {{row, col}, n}

    graph |> Graph.add_edges(edges)
  end)
render_graph = fn graph ->
  tmp = Path.join(System.tmp_dir!(), :crypto.strong_rand_bytes(10) |> Base.encode32())

  {:ok, dot} = Graph.to_dot(graph)
  File.write!(tmp <> ".dot", dot)
  {"", 0} = System.cmd("/opt/homebrew/bin/dot", ["-T", "png", tmp <> ".dot", "-o", tmp <> ".png"])

  png_data = File.read!(tmp <> ".png")
  image = Kino.Image.new(png_data, :png)

  # File.rm!(tmp <> ".dot")
  # File.rm!(tmp <> ".png")

  image
end
# render_graph.(graph)
Graph.edges(graph) |> Enum.count()
Graph.vertices(graph) |> Enum.count()

Part 1

shortest = Graph.Pathfinding.dijkstra(graph, start, end_)
length(shortest) - 1
shortest
render_map = fn map, shortest ->
  heights = map |> Enum.map(fn {{row, col}, height} -> %{x: col, y: row, height: height} end)

  path =
    shortest
    |> Enum.map(fn {row, col} ->
      text = [Map.get(map, {row, col})] |> List.to_string()
      %{x: col, y: row, label: text}
    end)

  height_layer =
    Vl.new()
    |> Vl.mark(:rect)
    |> Vl.data(values: heights)
    |> Vl.encode_field(:x, "x")
    |> Vl.encode_field(:y, "y")
    |> Vl.encode_field(:color, "height",
      type: :ordinal,
      legend: nil,
      scale: [scheme: "greens"]
    )

  shortest_layer =
    Vl.new()
    |> Vl.mark(:text)
    |> Vl.data(values: path)
    |> Vl.encode_field(:x, "x")
    |> Vl.encode_field(:y, "y")
    |> Vl.encode_field(:text, "label")

  Vl.new(width: 600, height: 600)
  |> Vl.layers([height_layer, shortest_layer])
end
render_map.(map, shortest)

Part 2

starts =
  Enum.filter(map, fn
    {_, ?a} -> true
    _ -> false
  end)
  |> Enum.map(fn {rc, _} -> rc end)
paths = for start <- starts, do: Graph.Pathfinding.dijkstra(graph, start, end_)
shortest = paths |> Enum.filter(&amp;is_list/1) |> Enum.min_by(&amp;length/1)
length(shortest) - 1