This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
[TestMethod] | |
public void Get_returns_correct_details_for_a_stored_location() | |
{ | |
// Create some locations | |
Location l1 = new Location(4); | |
Location l2 = new Location(2); | |
Location l3 = new Location(7); | |
// Set some random exits between locations | |
l2.SetExitToLocation(0, l1.Id); | |
l2.SetExitToLocation(2, l3.Id); | |
// Set up the database context | |
using (QuesterContext dbcontext = new QuesterContext()) | |
{ | |
// Clear the Locations table | |
dbcontext.RemoveRange(dbcontext.Locations); | |
// Add the locations I prepared earlier | |
dbcontext.Locations.Add(l1); | |
dbcontext.Locations.Add(l2); | |
dbcontext.Locations.Add(l3); | |
// Save my changes to the locations table | |
dbcontext.SaveChanges(); | |
} | |
// Now test that Get() retrieves entries from the database correctly | |
LocationRepository locationRepository = new LocationRepository(); | |
Location retrievedLocation = locationRepository.Get(l2.Id); | |
// Check that we have the correct number of locations | |
Assert.AreEqual(l2.Id, retrievedLocation.Id); | |
Assert.AreEqual(l1.Id, retrievedLocation.ExitInDirection(0)); | |
Assert.AreEqual(l3.Id, retrievedLocation.ExitInDirection(2)); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public Location Get(int id) | |
{ | |
throw new NotImplementedException(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public Location Get(int id) | |
{ | |
Location result = null; | |
using(QuesterContext context = new QuesterContext()) | |
{ | |
result = (from e in context.Locations | |
where e.Id == id | |
select e).FirstOrDefault(); | |
} | |
return result; | |
} |
Stepping through the test in the debugger reveals the following: locations l1 through l3 are all set up with correct Ids and exits. My retrievedLocation, however, has the correct Id, but all exits are set to the default of -1.
Looking at the actual database file, I see that the schema only has a column for the Id - but nothing for the exits. This is not actually all that surprising, when you think about it. I, for one, can't think of a good way to store an array in a database - and I guess the programmers behind Entity Framework couldn't either, so I consider myself in good company here. As a result, the array that holds the information about exits in my Location class simply gets ignored.
I can see two ways around the problem: I can create individual integer fields for each allowable direction. This strikes me as a horrendous solution; it violates several OOD principles. If I ever needed more than, say, four directions, I would have to modify my Locations class to do so, which violates the open/closed principle. Furthermore, the number of directions really shouldn't be defined in the Location class - it belongs in a class that deals with navigation, according to the single responsibility principle.
With the current solution, using an array, I could have worked around it by injecting the number of possible directions into the constructor. With individual integer fields, e.g. int NorthExit, int SouthExit etc. that would be impossible, so I'm not going to go there.
Instead I'm going to use a new class that deals specifically with this problem:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class Exit | |
{ | |
public Exit(int directionId, int destinationLocationId) | |
{ | |
DirectionId = directionId; | |
DestinationLocationId = destinationLocationId; | |
} | |
public int ExitId { get; set; } | |
public int DirectionId { get; set; } | |
public int DestinationLocationId { get; set; } | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class Location | |
{ | |
private List<Exit> _Exits { get; set; } | |
public int LocationId { get; set; } | |
public Location(int id) : this() | |
{ | |
// Store the id of the location | |
LocationId = id; | |
} | |
public Location() | |
{ | |
// Initialise the exits | |
_Exits = new List<Exit>(); | |
} | |
/// <summary> | |
/// Gets the location to which a given exit leads, if applicable | |
/// </summary> | |
/// <param name="direction">Direction of the exit</param> | |
/// <returns>The location the exit leads to or -1 if there is no location in the specified direction</returns> | |
public int ExitInDirection(int direction) | |
{ | |
int result = -1; | |
Exit exitInDirection = (from e in _Exits | |
where e.DirectionId == direction | |
select e).FirstOrDefault(); | |
// If an exit in the specified direction exists, return that | |
if (exitInDirection != null) | |
result = exitInDirection.DestinationLocationId; | |
return result; | |
} | |
/// <summary> | |
/// Connects an exit to the Id of the location it leads to | |
/// </summary> | |
/// <param name="direction">Direction of the exit</param> | |
/// <param name="destinationLocationId">Id of the location the exit should lead to</param> | |
public void SetExitToLocation(int direction, int destinationLocationId) | |
{ | |
_Exits.Add(new Exit(direction, destinationLocationId)); | |
} | |
} |
- Location_is_constructed_with_all_exits_set_to_no_exit()
- Location_returns_no_exit_if_direction_is_negative()
- Location_returns_no_exit_if_direction_is_greater_than_available_directions()
- CreateOneWayExitToLocation_throws_exception_if_direction_is_negative()
- CreateOneWayExitToLocation_throws_exception_if_direction_is_greater_than_available_directions()
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
[TestMethod] | |
public void Location_returns_minus_one_if_no_exit_in_that_direction() | |
{ | |
Location location = new Location(1); | |
Assert.AreEqual(-1, location.ExitInDirection(-1)); | |
} |
Getting around this is a little bit tricky in EF7, so I'll leave that until the next post.