JamCoders

💾 Download
In [ ]:
%config InteractiveShell.ast_node_interactivity="none"
%pip install networkx
%pip install matplotlib
%pip install pygraphviz
import networkx as nx
import warnings
import base64
warnings.simplefilter(action='ignore', category=FutureWarning)

def f(globals, locals):
    import base64
    CODE = (
        "ZGVmIG1ha2VfcHJpbnRfbG9jYWxzKCk6IAogICAgIyBJbiBhIGZ1bmN0aW9uIHRvIHByZXZlbnQgbG9jYWxzIGFuZCBpbXBvcnRzIGZyb20gbGVha2luZy4KICAgIGdsb2JhbCBtYWtlX3ByaW50X2xvY2FscwogICAgZGVsIG1ha2VfcHJpbnRfbG9jYWxzICAjIE9ubHkgcnVuIHRoaXMgZnVuY3Rpb24gb25jZS4KCiAgICBpbXBvcnQgSVB5dGhvbgogICAgaW1wb3J0IGFzdAogICAgaW1wb3J0IGluc3BlY3QKCiAgICBjbGFzcyBWaXNpdG9yKGFzdC5Ob2RlVmlzaXRvcik6CiAgICAgICAgZGVmIF9faW5pdF9fKHNlbGYpOgogICAgICAgICAgICBzZWxmLnZhcmlhYmxlcyA9IHNldCgpCiAgICAgICAgZGVmIHZpc2l0X05hbWUoc2VsZiwgbmFtZSk6CiAgICAgICAgICAgIHNlbGYudmFyaWFibGVzLmFkZChuYW1lLmlkKQogICAgICAgICMgVE9ETzogUG9zc2libHkgZGV0ZWN0IHdoZXRoZXIgdmFyaWFibGVzIGFyZSBhc3NpZ25lZCB0by4KCiAgICBBTExPV19UWVBFUyA9IFtpbnQsIGZsb2F0LCBzdHIsIGJvb2wsIGxpc3QsIGRpY3QsIHR1cGxlLCByYW5nZV0KICAgIGRlZiBmaWx0ZXJfdmFyaWFibGVzKHZhcmlhYmxlcywgbG9jYWxzKToKICAgICAgICBmb3IgdiBpbiB2YXJpYWJsZXM6CiAgICAgICAgICAgIGlmIHYgaW4gbG9jYWxzIGFuZCB0eXBlKGxvY2Fsc1t2XSkgaW4gQUxMT1dfVFlQRVM6CiAgICAgICAgICAgICAgICB5aWVsZCB2CiAgCiAgICAjIFVuZm9ydHVuYXRlbHksIHRoZXJlIGRvZXNuJ3Qgc2VlbSB0byBiZSBhIHN1cHBvcnRlZCB3YXkgb2YgZ2V0dGluZwogICAgIyB0aGUgY3VycmVudCBjZWxsJ3MgY29kZSB2aWEgdGhlIHB1YmxpYyBJUHl0aG9uIEFQSXMuIEhvd2V2ZXIsIGJlY2F1c2UKICAgICMgd2UgYXJlIGdldHRpbmcgY2FsbGVkIGZyb20gSVB5dGhvbiBpdHNlbGYgYW5kIHdlIGFyZSBhbHJlYWR5IGluc3BlY3RpbmcKICAgICMgdGhlIHN0YWNrdHJhY2UsIHdlIG1pZ2h0IGFzIHdlbGwganVzdCBwZWVrIGludG8gaXRzIGZyYW1lLi4uCiAgICBpZiBJUHl0aG9uLl9fdmVyc2lvbl9fID09ICI1LjUuMCI6CiAgICAgICAgIyBjb2xhYgogICAgICAgIGRlZiBnZXRfYXN0KGZyYW1lKToKICAgICAgICAgICAgcmV0dXJuIGFzdC5Nb2R1bGUoZnJhbWUuZl9iYWNrLmZfYmFjay5mX2xvY2Fsc1sibm9kZWxpc3QiXSkKICAgICAgICBkZWYgZmluZF9sb2NhbHMoZnJhbWUpOgogICAgICAgICAgICByZXR1cm4gZnJhbWUuZl9sb2NhbHMKICAgICAgICBkZWYgYXRfdG9wX2xldmVsKGZyYW1lKToKICAgICAgICAgICAgcmV0dXJuIGZyYW1lLmZfYmFjay5mX2NvZGUuY29fZmlsZW5hbWUuZW5kc3dpdGgoIklQeXRob24vY29yZS9pbnRlcmFjdGl2ZXNoZWxsLnB5IikKCiAgICBlbGlmIElQeXRob24uX192ZXJzaW9uX18gPT0gIjguNC4wIjoKICAgICAgICAjIGxhYiBjb21wdXRlcnMKICAgICAgICBkZWYgZ2V0X2FzdChmcmFtZSk6CiAgICAgICAgICAgIHJldHVybiBhc3QuTW9kdWxlKGZyYW1lLmZfYmFjay5mX2JhY2suZl9sb2NhbHNbIm5vZGVsaXN0Il0pCiAgICAgICAgZGVmIGZpbmRfbG9jYWxzKGZyYW1lKToKICAgICAgICAgICAgcmV0dXJuIGZyYW1lLmZfbG9jYWxzCiAgICAgICAgZGVmIGF0X3RvcF9sZXZlbChmcmFtZSk6CiAgICAgICAgICAgIHJldHVybiBmcmFtZS5mX2JhY2suZl9jb2RlLmNvX2ZpbGVuYW1lLmVuZHN3aXRoKCJJUHl0aG9uL2NvcmUvaW50ZXJhY3RpdmVzaGVsbC5weSIpCiAgICBlbHNlOgogICAgICAgIHByaW50KGYicHJpbnRfbG9jYWxzKCkgbm90IHN1cHBvcnRlZCBvbiBJUHl0aG9uIHZlcnNpb24ge0lQeXRob24uX192ZXJzaW9uX199IikKCiAgICBkZWYgZ2V0X2NlbGxfbmFtZXMoZnJhbWUpOgogICAgICAgIHRyZWUgPSBnZXRfYXN0KGZyYW1lKQogICAgICAgIHZpc2l0b3IgPSBWaXNpdG9yKCkKICAgICAgICB2aXNpdG9yLnZpc2l0KHRyZWUpCiAgICAgICAgcmV0dXJuIGZpbHRlcl92YXJpYWJsZXModmlzaXRvci52YXJpYWJsZXMsIGZyYW1lLmZfbG9jYWxzKQoKICAgIGRlZiBmaW5kX3doaWNoKGZyYW1lKToKICAgICAgICAjIEZyYW1lIGlzIHRoZSBmcmFtZSB3aG9zZSBsb2NhbHMgd2UgYXJlIGludGVyZXN0ZWQgaW4gcHJpbnRpbmcuCiAgICAgICAgaWYgYXRfdG9wX2xldmVsKGZyYW1lKToKICAgICAgICAgICAgIyBUaGUgcGFyZW50IGZyYW1lIG9mIHRoZSBpbnRlcmVzdGVkIGZyYW1lIGlzIGEgbW9kdWxlLCBtb3N0IGxpa2VseQogICAgICAgICAgICAjICJpbnRlcmFjdGl2ZXNoZWxsIi4gVGhpcyBtZWFucyB3ZSBhcmUgaW4gdGhlIGdsb2JhbCBzY29wZSwgc2luY2UKICAgICAgICAgICAgIyBvbmx5IHRoZSBnbG9iYWwgc2NvcGUgc2hvdWxkIGJlIGRpcmVjdGx5IHJ1biBieSB0aGUgaW50ZXJhY3RpdmUgc2hlbGwuCiAgICAgICAgICAgIHJldHVybiBzZXQoZ2V0X2NlbGxfbmFtZXMoZnJhbWUpKQogICAgICAgICMgVGhlIHBhcmVudCBmcmFtZSBpcyBub3QgYSBtb2R1bGUsIHNvIHdlIGFyZSBpbiBhIGxvY2FsIHNjb3BlLgogICAgICAgIHJldHVybiBzZXQoZnJhbWUuZl9sb2NhbHMpCgogICAgZGVmIHByaW50X2xvY2Fscygqd2hpY2gsIHR5cGVzPUZhbHNlKToKICAgICAgICAiIiJQcmludCB0aGUgbG9jYWwgdmFyaWFibGVzIGluIHRoZSBjYWxsZXIncyBmcmFtZS4iIiIKICAgICAgICBpbXBvcnQgaW5zcGVjdAogICAgICAgICMgY3VycmVudGZyYW1lKCkgZnJhbWUgaXMgcHJpbnRfbG9jYWxzLiBXZSB3YW50IHRoZSBjYWxsZXIncyBmcmFtZQogICAgICAgIGZyYW1lID0gaW5zcGVjdC5jdXJyZW50ZnJhbWUoKS5mX2JhY2sKICAgICAgICBsb2NhbHMgPSBmaW5kX2xvY2FscyhmcmFtZSkKICAgICAgICB3aGljaCA9IHNldCh3aGljaCkgaWYgd2hpY2ggZWxzZSBmaW5kX3doaWNoKGZyYW1lKQogICAgICAgIGxsID0ge2s6IHYgZm9yIGssIHYgaW4gbG9jYWxzLml0ZW1zKCkgaWYgayBpbiB3aGljaH0KICAgICAgICBpZiBub3QgbGw6CiAgICAgICAgICAgIHByaW50KCJwcmludF9sb2NhbHM6IG5vIGxvY2FscyIpCiAgICAgICAgICAgIHJldHVybgogICAgICAgIGlmIHR5cGVzOgogICAgICAgICAgICBwcmludCgiXG4iLmpvaW4oZiJ7a306IHt0eXBlKHYpLl9fbmFtZV9ffSDihpAge3Z9IiBmb3IgaywgdiBpbiBsbC5pdGVtcygpKSkKICAgICAgICBlbHNlOgogICAgICAgICAgICBwcmludCgiOyAiLmpvaW4oZiJ7a30g4oaQIHtyZXByKHYpfSIgZm9yIGssIHYgaW4gbGwuaXRlbXMoKSkpCgogICAgcmV0dXJuIHByaW50X2xvY2FscwoKcHJpbnRfbG9jYWxzID0gbWFrZV9wcmludF9sb2NhbHMoKQ==",
        "def make_pretty_assert():
    import sys
    import IPython
    import ast
    import inspect
    import io
    import itertools
    import functools
    import copy
    
    global make_pretty_assert
    del make_pretty_assert

    if IPython.__version__ == "5.5.0":
        # colab TODO
        def get_assert_line(frame):
            return frame.f_back.f_back.f_locals["node"]
        def at_top_level(frame):
            code = frame.f_back.f_back.f_code
            return code.co_filename.endswith("IPython/core/interactiveshell.py")
        def can_annotate(frame):
            return False

    elif IPython.__version__ == "8.4.0":
        # lab computers
        def get_assert_line(frame):
            return frame.f_back.f_back.f_locals["node"]
        def at_top_level(frame):
            code = frame.f_back.f_back.f_code
            return code.co_filename.endswith("IPython/core/interactiveshell.py") and code.co_name == "run_ast_nodes"
        def can_annotate(frame):
            return at_top_level(frame)

    def is_literal(a):
        try:
            ast.literal_eval(a)
            return True
        except ValueError:
            return False
        return False


    def annotate_call(frame):
        if not can_annotate(frame):
            return
        
        expr: ast.Expr = get_assert_line(frame)
        for kwarg in expr.value.keywords:
            if kwarg.arg == "got":
                got = kwarg.value

        if isinstance(got, ast.Call):
            if not isinstance(got.func, ast.Name):
                return
            name = got.func.id
            func = frame.f_locals[name]
            if not hasattr(func, DEBUGGABLE_ARGS):
                return
            args = getattr(func, DEBUGGABLE_ARGS)
            kwargs = getattr(func, DEBUGGABLE_KWARGS)
            if args is None or kwargs is None:
                return
            boundargs = inspect.signature(func).bind(*args, **kwargs)

            result = io.StringIO()
            pos = result.write(name)
            pos += result.write("(")

            sep = ', '
            arg_tuples = []  # (ast_arg_str, evaluated_arg, start, limit)

            # Use an iterator here. If a boundarg is not consumed here,
            # it may be consumed later by a pos+kw arg.
            boundargs_iter = iter(boundargs.args)
            for ast_arg, arg in zip(got.args, boundargs_iter):
                ast_arg_str = ast.unparse(ast_arg)
                arg_tuples.append((ast_arg_str, arg, pos, pos + len(ast_arg_str)))
                pos += len(ast_arg_str) + len(sep)

            for ast_kwarg, kwarg in itertools.zip_longest(got.keywords, boundargs_iter):
                ast_arg_str = ast.unparse(ast_kwarg)
                arg_tuples.append((ast_arg_str, kwarg, pos + len(ast_kwarg.arg) + len('='), pos + len(ast_arg_str)))
                pos += len(ast_arg_str) + len(sep)

            pos -= len(sep)
            result.write(sep.join(t[0] for t in arg_tuples))
            result.write(')')
            result_str = result.getvalue()

            yield result_str

            nonliterals = []  # (evaluated_arg, result_str, start, limit)
            for _, evaluated_arg, start, limit in arg_tuples:
                if not is_literal(result_str[start:limit]):
                    nonliterals.append((evaluated_arg, result_str[start:limit], start, limit))

            underlines = io.StringIO()
            pos = 0
            for _, _, start, limit in nonliterals:
                if (limit - start) < 3:
                    pos += underlines.write(' ' * (start - pos))
                    pos += underlines.write('↑')
                    pos += underlines.write('↑' * (limit - pos))
                    continue
                idx = start + (limit - start) // 2
                pos += underlines.write(' ' * (start - pos))
                #pos += underlines.write('╙')
                pos += underlines.write('↑' * (idx - pos))
                pos += underlines.write('↑')
                pos += underlines.write('↑' * (limit - pos))
                #pos += underlines.write('╜')
            yield underlines.getvalue()

            def make_bars_buf(nonliterals, end, evaled):
                buf = io.StringIO()
                pos = 0
                for i in range(len(nonliterals) - 1):
                    _, _, start, limit = nonliterals[i]
                    if (limit - start) < 3:
                        idx = start
                    else:
                        idx = start + (limit - start) // 2
                    pos += buf.write(' ' * (idx - pos))
                    pos += buf.write('│')
                    pos += buf.write(' ' * (limit - pos))

                _, result_str, start, limit = nonliterals[-1]
                if (limit - start) < 3:
                    idx = start
                else:
                    idx = start + (limit - start) // 2                
                pos += buf.write(' ' * (idx - pos))
                pos += buf.write('└')
                pos += buf.write('─' * (end - 1 - pos))
                pos += buf.write('╴')
                pos += buf.write(result_str)
                pos += buf.write(' ≔ ')
                pos += buf.write(str(evaled))

                return buf, pos

            while nonliterals:
                buf, pos = make_bars_buf(nonliterals, len(result_str) + 4, nonliterals[-1][0])
                yield buf.getvalue()
                last = nonliterals.pop()

    def assert_equal(*, want, got, out=sys.stdout):
        if want == got:
            print("Test case passed.")
            return

        frame = inspect.currentframe().f_back

        box_padding = 2
        header = " Test case failed. "
        want_line = f"{box_padding * ' '}Want: {repr(want)} (type: {type(want).__name__})"
        got_line = f"{box_padding * ' '}Got:  {repr(got)} (type: {type(got).__name__})"
        
        if can_annotate(frame):
            assert_line = f"{box_padding * ' '}>>> {ast.unparse(get_assert_line(frame))}"
        else:
            assert_line = ""
            
        debug_lines = list(annotate_call(frame))
        if debug_lines:
            got_line += " ← "
            padding = len(got_line)
            got_line += debug_lines[0]

        padded_debug_lines = []
        for i, line in enumerate(debug_lines):
                if i == 0:
                    continue
                padded_debug_lines.append(' ' * padding + line)
                
        line_max_len = max(len(l) + box_padding for l in (assert_line, want_line, got_line, *padded_debug_lines))
        line_max_len = max(32, line_max_len)
        header_dashes = line_max_len - len(header)


        print(file=out)
        print('-' * (header_dashes // 2), end="", file=out)
        print(header, end="", file=out)
        print('-' * ((header_dashes + 1) // 2), end="", file=out)
        print(file=out)
        print(file=out)

        
        if assert_line:
            print(assert_line, file=out)
            print(file=out)

        print(want_line, file=out)
        print(got_line, file=out)
        for line in padded_debug_lines:
            print(line, file=out)
        
        print(file=out)
        print('-' * line_max_len, file=out)
        print(file=out)
        
    DEBUGGABLE_ARGS = "_debuggable_args"
    DEBUGGABLE_KWARGS = "_debuggable_kwargs"

    def debuggable(f):
        @functools.wraps(f)
        def g(*args, **kwargs):
            try:
                args_copy = copy.deepcopy(args)
                kwargs_copy = copy.deepcopy(kwargs)
            except Exception:
                args_copy = None
                kwargs_copy = None
            result = f(*args, **kwargs)
            setattr(g, DEBUGGABLE_ARGS, args_copy)
            setattr(g, DEBUGGABLE_KWARGS, kwargs_copy)
            return result
        return g
    
    return assert_equal, debuggable

assert_equal, debuggable = make_pretty_assert()
",
    )
    for code in CODE:
        exec(base64.b64decode(code), globals, locals)
f(globals(), locals())
del f

def uh(b64):
    return eval(base64.b64decode(b64))

def h(code):
    return base64.b64encode(code)

def sae(*, want, got):
    assert_equal(want=sorted(want), got=sorted(got))

Day 17, Lab B: Dijkstra's Algorithm

1 Dijkstra's Algorithm

Dijkstra's algorithm finds the shortest path between a given source node and all the other nodes in a graph. Thus, it allows us to find the shortest path between any two nodes.

Edges connecting the nodes in a graph are weighted with a number. The weights can represent the "cost" to travel between the two nodes via that edge.

For example, to get from UWI Mona to Downtown Kingston, there are many paths. UWI Mona and Downtown Kingston are nodes in a graph. The different streets are edges. Each street might take a different amount of time, so the weight of the edges in the graph represent time in this example. Shortest path algorithms like Dijkstra's Algorithm helps us to find the fastest path to get from UWI Mona to Downtown Kingston.

1.1 Algorithm Overview

Dijkstra's works by visiting nodes in increasing order of their distance from the source node. Dijkstra's repeatedly visits the unvisited node with the smallest distance from the shortest node. When Dijkstra's visits a node $u$, it inspects all its neighbors $v$ to see if a new shortest path to $v$ can be discovered using the edge from $u \to v$.

After visiting all nodes, Dijkstra's returns the dist dictionary, which contains the distance (length of the shortest path) from the source node to all other nodes.

1.2 An Example

For example, in the following weighted undirected graph, we have five nodes: $A$, $B$, $C$, $D$, and $E$. Let's see how we can use Dijkstra's algorithm to find the distances and shortest paths from node $A$ to all other nodes.

First, we initialize the distance from $A$ to all other nodes to $\infty$, and the distance from $A$ to itself to $0$. As Dijkstra's traverses the graph, these distances will converge to their true values. We also set all nodes to unvisited.

Node Distance Visited
$A$ $0$ False
$B$ $\infty$ False
$C$ $\infty$ False
$D$ $\infty$ False
$E$ $\infty$ False

d1.png

Now, we start traversing the graph by visiting the closest (by distance) unvisited node. Looking at the table of distances, that node is node $A$:

Node Distance Visited
$A$ $0$ False
$B$ $\infty$ False
$C$ $\infty$ False
$D$ $\infty$ False
$E$ $\infty$ False

First, we set $A$ to visited:

Node Distance Visited
$A$ $0$ True
$B$ $\infty$ False
$C$ $\infty$ False
$D$ $\infty$ False
$E$ $\infty$ False

Next, we look at all of $A$'s neighbors and see whether we have discovered a new shortest path to that neighbor. $A$ has two neighbors, $B$, and $D$.

  1. Edge $A \to B$: We know the shortest path to $A$ has $0$ distance (by looking up that distance in the distance dictionary). The edge $A \to B$ has length $6$, so there exists a path from the source node $A$ to $B$ with length $6 = 0 \text{ (distance from $A$)} + 6 \text{ (length of edge $A \to B$)}$. This is shorter than the current known distance from $A$ to $B$, $\infty$ (again looking up in the distance table). So we can set the distance to $B$ to $6$.
  2. Edge $A \to D$. By the same logic as above, there must exist a path from $A$ to $D$ with length $1$. Since $1 < \infty$ (the current distance for $D$), so we can update the distance to $D$ to $1$ as well.

After the above distance updates, the new distance and visited table looks like:

Node Distance Visited
$A$ $0$ True
$B$ $6$ False
$C$ $\infty$ False
$D$ $1$ False
$E$ $\infty$ False

d2.png

Here's our distance and visited table on the second iteration of Dijkstra's:

Node Distance Visited
$A$ $0$ True
$B$ $6$ False
$C$ $\infty$ False
$D$ $1$ False
$E$ $\infty$ False

Next, the unvisited node with the shortest distance from the source node is $D$ with distance $1$. Thus, we visit $D$:

Node Distance Visited
$A$ $0$ True
$B$ $6$ False
$C$ $\infty$ False
$D$ $1$ True
$E$ $\infty$ False

Next, we consider $D$'s neighbors. $D$ has three neighbors: $A$, $B$, and $E$.

  1. Edge $D \to A$: This edge has weight $1$. We know that there is a path from $A$ to $D$ with length $1$ (because that is $D$'s distance in the distance table). Then there must be a path from $A$ to $A$ with length $2 = 1 + 1$. But $2$ is larger than the distance already stored in the table (which again is the length of the shortest path), so we don't update $A$'s distance in the table.
  2. Edge $D \to B$: This edge has weight $2$. That means that there must be a path from $A$ to $B$ through $D$ with length $3 = 1 + 2$. This is better than the distance to $B$ stored in the above table ($3 < 6$), so we update $B$'s distance in the table again.
  3. Edge $D \to E$. This edge has weight $1$. That means that there must be a path from $A$ to $E$ through $D$ with length $2 = 1 + 1$. This is better than the length of the current best path to $E$, $\infty$, so we also update $E$'s distance in the table.

Our new table looks like

Node Distance Visited
$A$ $0$ True
$B$ $3$ False
$C$ $\infty$ False
$D$ $1$ True
$E$ $2$ False

d3.png

On the start of our third iteration of Dijkstra's, our table looks like

Node Distance Visited
$A$ $0$ True
$B$ $3$ False
$C$ $\infty$ False
$D$ $1$ True
$E$ $2$ False

What is the next closest unvisited node?

Musée Rodin 1.jpg

If you answered $E$, you're correct. We visit $E$ and consider all its neighbors:

  1. Edge $E \to B$: weight $2$. The distance from $A$ to $E$ is $2$, so there exists a path from $A$ to $B$ through $E$ with length $4 = \text{dist}[E] + 2 = 2 + 2$. But $4$ is not smaller than the shortest distance to $B$ in our table ($3$), so we do not update this distance in the table.
  2. Edge $E \to C$: weight $5$. This means there is a path from $A$ to $C$ through $E$ with length $7 = \text{dist}[E] + 5 = 2 + 5$. Since $7$ is better than our current distance from $A$ to $C$, we update it in the table.
  3. Edge $E \to D$: weight $1$. This means there is a path from $A$ to $D$ through $E$ with length $3$. But this is no better than our current best distance from $A$ to $D$ of length $1$, so we don't update $D$'s distance in the table.

Now, our distance table looks like:

Node Distance Visited
$A$ $0$ True
$B$ $3$ False
$C$ $7$ False
$D$ $1$ True
$E$ $2$ True

d4.png

Next, we visit the next closest unvisited node, $B$. Think through what happens at this step. After we visit $B$, our table looks like

Node Distance Visited
$A$ $0$ True
$B$ $3$ True
$C$ $7$ False
$D$ $1$ True
$E$ $2$ True

d5.png

Finally, the only unvisited node is C. We update our table with the same logic above. (We've omitted it for brevity; try working through the logic yourself.)

Node Distance Visited
$A$ $0$ True
$B$ $3$ True
$C$ $7$ True
$D$ $1$ True
$E$ $2$ True

Thus, the distance from $A$ to all nodes is stored in the above table.

1.2 Graph, with weights

We reimplemented the class Graph from yesterday afternoon's lab. We want to make our graphs weighted so that we will do our Dijkstra's algorithm on them. We've modified the functions to handle the added weights to the edges. In addition, there's a new get_edge_weight function.

Take a moment to look at the implementation and notice the difference from yesterday's lab.

In [ ]:
class Graph:
    """Represents a graph consisting of nodes and edges."""
    
    def __init__(self):
        """Initializes an empty graph."""
        # Neighbors maps a node to its neighbors.
        self.neighbors = {}
        
        # Weights maps a edge represented as a node tuple (e.g. ("A", "B"))
        # to the weight of that edge.
        self.weights = {}
        
    
    def add_node(self, node_label):
        """Adds a node to the graph.
        
        The node is associated with a string node_label which is used to add
        edges to it later. If a node associated with node_label already exists,
        does nothing.

        Arguments:
            node_label (string).

        Effects: Modifies the graph. If the node already exists, does nothing.
        """
        # First, check if a node associated with node_label already exists
        # Hint: The expression "k in dict" is True if dict has a key called 'k' (else False).
        # If the node doesn't already exist, add it by initializing an empty adjacency list []
        if not node_label in self.neighbors:
            self.neighbors[node_label] = []
            
    
    def add_edge(self, node1, node2, weight):
        """Connects node1 and node2 with an edge.

        Arguments:
            node1 (string): Label of the first node
            node2 (string): Label of the second node.
        
        Effects:
            Modifies the graph by adding an edge from node1 to node2. If an edge
            already exists, does nothing.
        """
        # First, check if the nodes are already connected
        # Hint: The expression "x in list" is True if list contains the element 'x' (else False).
        # If they aren't already connected, connect them by adding each to the other's adjacency list.
        if not node2 in self.neighbors[node1] and not node1 in self.neighbors[node2]:
            self.neighbors[node1] += [node2]
            self.neighbors[node2] += [node1]
            
        self.weights[(node1, node2)] = weight
        self.weights[(node2, node1)] = weight
        
      
    def get_nodes(self):
        """Returns a list of all nodes in the graph.
        
        Returns: (list[str]): Labels of all nodes in the graph.
        """
        return list(self.neighbors.keys()) 
    
    
    def get_neighbors(self, node):
        """Returns list of neighbors of node.
        
        Arguments:
            node (str): The node whose neighbors to retrieve.
            
        Returns (list[str]): A list of that node's neighbors.
        """
        return self.neighbors[node]
    

    def get_edge_weight(self, node1, node2) :
        """Returns the weight of the edge connecting node1 and node2.
        
        Arguments:
            node1 (str): The first node in the edge.
            node2 (str): The second node in the edge.
            
        Returns: int or None.
        """
        edge = (node1, node2)
        if edge in self.weights:
            return self.weights[edge]
        
        return None
                
    
    def draw(self):
        """Prints a visual representation of the graph.
        
        Effects: Prints the graph as output.
        """
        # Create set of edges
        # Create nx graph
        g = nx.Graph()
        for node in self.get_nodes():
            g.add_node(node)
            
        for (u, v), weight in self.weights.items():
            if u > v:
                continue
            g.add_edge(u, v, weight=weight)
            
        pos = nx.nx_agraph.graphviz_layout(g, prog="dot")
        edge_labels = nx.get_edge_attributes(g, "weight")
        nx.draw_networkx_nodes(g, pos)
        nx.draw_networkx_labels(g, pos)
        nx.draw_networkx_edges(g, pos)
        nx.draw_networkx_edge_labels(g, pos, edge_labels)
        
graph = Graph()

graph.add_node("A")
graph.add_node("B")
graph.add_node("C")
graph.add_node("D")
graph.add_node("E")

graph.add_edge("A", "D", 1)
graph.add_edge("A", "B", 6)
graph.add_edge("D", "E", 1)
graph.add_edge("D", "B", 2)
graph.add_edge("E", "B", 2)
graph.add_edge("E", "C", 5)
graph.add_edge("B", "C", 5)

graph.draw()

1.3 Some Help

To help you implement Dijkstra's, first implement the helper functions below.

In [ ]:
def initialize_visited(graph):
    """Initializes a visited dictionary.
    
    The visited dictionary should map each node to whether it has been
    visited or not. At the time of initialization, no nodes should be
    marked as visited.
    
    Arguments:
        graph (Graph): The graph whose nodes are keys in this visited mapping.
        
    Returns (dict[str, bool]):
        A visited dictionary where no nodes are marked as visited
    """
    visited = {}
    # YOUR CODE HERE


want_visited = {
    "A": False,
    "B": False,
    "C": False,
    "D": False,
    "E": False,
}
assert_equal(want=want_visited, got=initialize_visited(graph))
In [ ]:
def initialize_dist(graph, source):
    """Initializes the dist dictionary.
    
    The dist dictionary should map each node in the graph to the length
    of the shortest known path from the source to that node. At the
    time of initialization, the length of the shortest known path to all
    nodes should be infinite (float('inf')), except the length of the
    shortest path to the source node itself, which should be 0.
    
    Arguments:
        graph (Graph): The graph whose nodes are keys in this distance mapping.
        source: The source node.
    
    Returns (dict[str, number]):
        A dist dictionary where all nodes are infinite distance away, except the
          source node, which is 0 distance away.
    """
    dist = {}
    # YOUR CODE HERE


want_dist = {
    "A": 0,
    "B": float('inf'),
    "C": float('inf'),
    "D": float('inf'),
    "E": float('inf'),
}
assert_equal(want=want_dist, got=initialize_dist(graph, "A"))
In [ ]:
def find_closest_unvisited(visited, dist):
    """Finds the label of a unvisited node with a least distance.
    
    If there are multiple unvisited nodes with the same distance, return
    any one. If there are no unvisited nodes, return None.
    
    Arguments:
        visited (dict[str, bool]): A visited mapping.
        dist (dict[str, number]): A dist mapping.
    """
    # YOUR CODE HERE
    

visited = {
    "A": True,
    "B": False,
    "C": False,
    "D": True,
    "E": True,
}
dist = {
    "A": 0,
    "B": 3,
    "C": 7,
    "D": 1,
    "E": 2,
}
assert_equal(want="B", got=find_closest_unvisited(visited, dist))

1.4 Dijkstra's

Now, using the above functions and the Graph class, implement Dijkstra's algorithm.

In [ ]:
@debuggable
def dijkstra(graph, source) :
    """Computes the length of the shortest paths from the source node to all nodes.
    
    Arguments:
        graph (Graph): The input graph to find shortest paths.
        source (str): The label of the source node.
        
    Returns (dict[str, number]):
        A mapping from node labels to the length of the shortest path from the source to
          that node. If there is no path from the source to a node, its distance from the
          source is infinite.
    """
    visited = initialize_visited(graph)
    dist = initialize_dist(graph, source)
  
    current = ___________________________________                # YOUR CODE HERE
    while current is not None:
        # Mark the current node as visited.
        _________________________                                # YOUR CODE HERE
        
        neighbors = _________________________                    # YOUR CODE HERE
        for neighbor in neighbors:
            weight = _________________________                   # YOUR CODE HERE
            possible_path_length = _________________________     # YOUR CODE HERE
            if _________________ < _________________:            # YOUR CODE HERE
                dist[neighbor] = _________________               # YOUR CODE HERE
    
        current = ___________________________________            # YOUR CODE HERE
    
    return dist
In [ ]:
graph1 = Graph()

graph1.add_node("A")
graph1.add_node("B")
graph1.add_node("C")
graph1.add_node("D")
graph1.add_node("E")
graph1.add_node("F")  # Disconnected!

graph1.add_edge("A", "D", 1)
graph1.add_edge("A", "B", 6)
graph1.add_edge("D", "E", 1)
graph1.add_edge("D", "B", 2)
graph1.add_edge("E", "B", 2)
graph1.add_edge("E", "C", 5)
graph1.add_edge("B", "C", 5)

graph1.draw()

dists = dijkstra(graph1, "A")
assert_equal(want=0, got=dists["A"])
assert_equal(want=3, got=dists["B"])
assert_equal(want=7, got=dists["C"])
assert_equal(want=1, got=dists["D"])
assert_equal(want=2, got=dists["E"])
assert_equal(want=float('inf'), got=dists["F"])
In [ ]:
graph2 = Graph()
graph2.add_node("A")
graph2.add_node("B")
graph2.add_node("C")
graph2.add_node("D")
graph2.add_node("E")

graph2.add_edge("A", "B", 10)
graph2.add_edge("B", "C", 2)
graph2.add_edge("A", "C", 3)
graph2.add_edge("C", "D", 9)
graph2.add_edge("B", "D", 1)

graph2.draw()

distsA = dijkstra(graph2, "A")
assert_equal(want=0, got=distsA["A"])
assert_equal(want=5, got=distsA["B"])
assert_equal(want=3, got=distsA["C"])
assert_equal(want=6, got=distsA["D"])
assert_equal(want=float('inf'), got=distsA["E"])

distsB = dijkstra(graph2, "B")
assert_equal(want=5, got=distsB["A"])
assert_equal(want=0, got=distsB["B"])
assert_equal(want=2, got=distsB["C"])
assert_equal(want=1, got=distsB["D"])
assert_equal(want=float('inf'), got=distsB["E"])
In [ ]: