import numpy as np
from environment.environment import SokobanEnvImpl
from environment.util import *

env = SokobanEnvImpl(use_default_env=True)

def test_find_player():
    assert find_player(env) == (4,4)

def test_find_boxes():
    assert find_boxes(env) == [(3, 4), (5, 5), (7, 5), (2, 7)]

def test_find_box_targets():
    assert list(sorted(find_box_targets(env))) == [(1, 7), (4, 5), (5, 4), (7, 6)]

def test_pos_in_bound():
    assert pos_in_bound(env, 5, 5) == True
    assert pos_in_bound(env, 11, 5) == False
    assert pos_in_bound(env, 5, 11) == False
    assert pos_in_bound(env, -1, 5) == False
    assert pos_in_bound(env, 5, -1) == False

def test_breadth_first_search():
    assert len(breadth_first_search(env=env)) == 30

def test_breadth_first_search_unsolvable():
    unsolvable_env = SokobanEnvImpl(fixated_env=(
        np.array([[0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 1, 1, 0, 1, 0, 0, 0],
                  [0, 1, 2, 1, 1, 0, 1, 1, 1, 0],
                  [0, 1, 1, 1, 2, 0, 1, 1, 1, 0],
                  [0, 1, 1, 1, 0, 0, 1, 2, 1, 0],
                  [0, 2, 1, 0, 0, 0, 0, 0, 0, 0],
                  [0, 1, 1, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]),
        np.array([[0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 1, 1, 0, 1, 0, 0, 0],
                  [0, 1, 2, 4, 5, 0, 1, 1, 1, 0],
                  [0, 1, 1, 1, 2, 0, 1, 4, 1, 0],
                  [0, 1, 1, 1, 0, 0, 1, 2, 1, 0],
                  [0, 2, 4, 0, 0, 0, 0, 0, 0, 0],
                  [0, 1, 1, 0, 0, 0, 0, 0, 0, 0],
                  [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]),
        {(6, 7): (5, 7), (5, 4): (5, 5), (4, 5): (4, 3), (7, 1): (7, 2)}
    )) # Wall divides playground
    assert breadth_first_search(env=unsolvable_env) == None

def test_find_shortest_paths_to_place_remaining_boxes():
    result = find_shortest_paths_to_place_remaining_boxes(env)
    assert len(result) == 4
    for box in result:
        assert box["shortest_path_to_place"] is not None
