Skip to content

2. Entity Framework

You may earn 4 points by completing this exercise.

Use GitHub Classroom to get your git repository. You can find the invitation link in Moodle. Clone the repository created via the link. It contains a skeleton and the expected structure of your submission. You need to create a branch, named solution, and work on this branch. After completing the exercises and verifying them, commit and push your submission.

Check the required software and tools here. This homework uses MSSQL database.

Entity Framework Core

We are using Entity Framework Core in this exercise. This is different from Entity Framework used in the seminar exercises; this is a platform-independent technology.

Exercise 0: Neptun code

Your very first task is to type your Neptun code into neptun.txt in the root of the repository.

Exercise 1: Database mapping using Code First model and queries (2 points)

Prepare the (partial) mapping of the database using Entity Framework Code First modeling. The Entity Framework Core package is part of the project, so you can start coding. The central class for database access is the DbContext. This class already exists with the name ProductDBContext.

  1. Map the product entity. Create a new class with the name DbProduct with the following code. (The Db prefix indicates that this class is within the scope of the database. This will be relevant in the next exercise.) We rely on conventions as much as possible: use property names that match the column names to make mapping automatic.

    using System.ComponentModel.DataAnnotations.Schema;
    
    namespace ef
    {
        [Table("Product")]
        public class DbProduct
        {
            [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
            public int ID { get; set; }
            public string Name { get; set; }
            public double Price { get; set; }
            public int Stock { get; set; }
        }
    }
    

    Open the source code of class ProductDbContext and uncomment the Products property.

  2. Create a new class with the name DbVat in namespace ef for mapping the VAT database table similarly as seen before. Do not forget to add a new DbSet property into ProductContext with the name Vat.

  3. Map the Product - VAT connection.

    Add a new get-set property into class DbProduct with name Vat and type DbVat. Use the ForeignKey attribute on this property, to indicate the foreign key used to store the relationship ("VatID").

    Create the “other side” of this one-to-many connection from class DbVat to DbProduct. This should be a new property of type System.Collections.Generic.List with name Products. (See an example in the link above.)

There are unit tests available in the solution. The test codes are commented out because it does not compile until you write the code. Select the whole test code and use Edit / Advanced / Uncomment Selection. You can run the unit tests in Visual Studio, or if you are using another IDE (e.g., VS Code, or dotnet cli), then run the tests using the cli. You may update the database connection string in class TestConnectionStringHelper if needed.

Tests

The tests presume that the database is in its initial state. Re-run the database initialization script to restore this state.

Do NOT change the unit tests. You may temporarily alter the unit tests if you need to, but make sure to reset your changes before committing.

If the tests do not compile

If the test code does not compile, you may have used slightly different property names. Fix these in your code and not in the tests!

OnConfiguring

You need no connection string in the DbContext. The constructor handles the connection to the database. Do not create OnConfiguring method in this class!

SUBMISSION

Upload the changed C# source code.

Create a screenshot displaying the successfully executed unit tests. You can run the tests in Visual Studio or using dotnet cli. Make sure that the screenshot includes the source code of the DbContext and the test execution outcome! Save the screenshot as f1.png and upload as part of your submission!

If you are using dotnet cli to run the tests, make sure to display the test names too. Use the -v n command line switch to set detailed logging.

The image does not need to show the exact same source code that you submit; there can be some minor changes here and there. That is, if the tests run successfully and you create the screenshot, then later you make some minor change to the source, there is no need for you to update the screenshot.

Exercise 2: Repository implementation using Entity Framework (2 points)

This exercise can be solved after completing the first exercise.

The Entity Framework DbContext created above has some drawbacks. For example, we need to trigger loading related entities using Include in every query, and the mapped entities are bound to precisely match the database schema. In complex applications, the DbContext is frequently wrapped in a repository that handles all peculiarities of the data access layer.

Implement class ProductRepository that helps with listing and inserting products. You are provided with a so-called model class representing the product entity, only in a more user-friendly way: it contains the tax percentage value directly. An instance of this class is built from database entities, but represents all information in one instance instead of having to handle a product and a VAT record separately. Class Model.Product contains most properties of class DbProduct, but instead of the navigation property to DbVat it contains the referenced percentage value (VAT.Percentage) directly.

Implement the methods of class `ProductRepository.

  • List shall return all products mapped to instances of Model.Product.
  • Insert shall insert a new product into the database. This method shall find the matching VAT record in the database based on the tax percentage value in the model class; if there is no match, it shall insert a new VAT record too! The method shall return the ID of the newly inserted ID (as generated by the database).
  • Delete should delete a product record matched by the ID. You should only delete the product record - no referenced records shall be removed. If delete is blocked by foreign keys, let the caller handle the error. The return value of the method should be false if no record with the specified ID exist; a true return value shall indicate successful deletion.
  • Do not change the definition of class ProductRepository (do not change the name of the class, nor the constructor or method declarations); only write the method bodies.
  • In the repository code use ProductRepository.createDbContext() to instantiate the DbContext (do not use TestConnectionStringHelper here).

SUBMISSION

Upload the changed C# source code.

YOU HAVE NOT FINISHED YET

After pushing your code to the solution branch, create a PR and assign your instructor (github username: akosdudas) to it! (more details: on the assignment submission page)

Exercise 3 optional: Logical Deletion with Entity Framework (0 points)

In the evaluation, you will see the text “imsc” in the exercise title; this is meant for the Hungarian students. Please ignore that.

Deleting data from a database is an operation that can have numerous unintended consequences. Restoring deleted data is much more difficult, and sometimes it is not even possible without repercussions. Deleting data can result in the loss of the entire data history, making it impossible to know the state before deletion or to use it in various statistics. Moreover, there are cases where relationships with other tables and foreign key constraints exist, and deletion affects those tables as well.

To overcome these problems, the most common solution is to implement a non-permanent deletion, known as a soft delete. In this case, a field (typically named IsDeleted) is used to indicate that the data has been deleted. Thus, the data remains in the database, but we can filter it to see whether it has been deleted.

A naive implementation of filtering is not convenient. Imagine having to add a condition to every query or save operation to ensure that deleted items are not affected. To address this, it is advisable to use one of Entity Framework's features, the Global Query Filter. This allows us to define filter conditions that are automatically applied to every query globally by Entity Framework.

Implement soft deletion for the previously created DbProduct class (there are multiple solutions; feel free to choose any of them):

Modifiability

Although the previous task had a restriction against overriding the OnConfiguring method, you are free to do so here if necessary (and you can also override other functions in the DBContext implementation)!

  1. Add an IsDeleted variable that indicates to our application whether the entity is in a deleted state!

  2. Add a QueryFilter that filters out the products that have already been deleted in every query, so they are not returned!

  3. Modify the deletion behavior in the database generally by extending the DbContext save operations (EFCore provides several extension points for this) so that instead of performing a true deletion, it only changes the IsDeleted variable! Do not change the deletion operation in the repository for modification!

SUBMISSION

Upload the modified C# source code.


2024-12-19 Contributors