225 lines
14 KiB
Markdown
225 lines
14 KiB
Markdown
|
---
|
||
|
title: Razor Pages with EF Core - CRUD - 2 of 10
|
||
|
author: rick-anderson
|
||
|
description: Shows how to create,read,update,delete with EF Core
|
||
|
keywords: ASP.NET Core,Entity Framework Core,CRUD,create,read,update,delete
|
||
|
ms.author: riande
|
||
|
manager: wpickett
|
||
|
ms.date: 10/15/2017
|
||
|
ms.topic: get-started-article
|
||
|
ms.technology: aspnet
|
||
|
ms.prod: asp.net-core
|
||
|
uid: data/ef-rp/crud
|
||
|
---
|
||
|
# Create, Read, Update, and Delete - EF Core with Razor Pages (2 of 10)
|
||
|
|
||
|
By [Tom Dykstra](https://github.com/tdykstra), [Jon P Smith](https://twitter.com/thereformedprog), and [Rick Anderson](https://twitter.com/RickAndMSFT)
|
||
|
|
||
|
[!INCLUDE[validation](../../includes/RP-EF/intro.md)]
|
||
|
|
||
|
In this tutorial, the scaffolded CRUD (create, read, update, delete) code is reviewed and customized.
|
||
|
|
||
|
Note: To minimize complexity and keep these tutorials focused on EF, EF code is used in the Razor Pages code-behind files. Some developers use a service layer or repository pattern in to create an abstraction layer between the UI (Razor Pages) and the data access layer.
|
||
|
|
||
|
In this tutorial, the Create, Edit, Delete, and Details Razor Pages in the *Student* folder are modified.
|
||
|
|
||
|
The scaffolded code uses the following pattern for Create, Edit, and Delete pages:
|
||
|
|
||
|
* Get and display the requested data with the HTTP GET method `OnGetAsync`.
|
||
|
* Save changes to the data with the HTTP POST method `OnPostAsync`.
|
||
|
|
||
|
The Index and Details pages get and display the requested data with the HTTP GET method `OnGetAsync`
|
||
|
|
||
|
## Replace SingleOrDefaultAsync with FirstOrDefaultAsync
|
||
|
|
||
|
The generated code uses [SingleOrDefaultAsync](https://docs.microsoft.com/en-us/dotnet/api/microsoft.entityframeworkcore.entityframeworkqueryableextensions.singleordefaultasync?view=efcore-2.0#Microsoft_EntityFrameworkCore_EntityFrameworkQueryableExtensions_SingleOrDefaultAsync__1_System_Linq_IQueryable___0__System_Linq_Expressions_Expression_System_Func___0_System_Boolean___System_Threading_CancellationToken_) to fetch the requested entity. [FirstOrDefaultAsync](https://docs.microsoft.com/en-us/dotnet/api/microsoft.entityframeworkcore.entityframeworkqueryableextensions.firstordefaultasync?view=efcore-2.0#Microsoft_EntityFrameworkCore_EntityFrameworkQueryableExtensions_FirstOrDefaultAsync__1_System_Linq_IQueryable___0__System_Threading_CancellationToken_) is more efficient at fetching one entity:
|
||
|
|
||
|
* Unless the code needs to verify that there is not more than one entity returned from the query.
|
||
|
* `SingleOrDefaultAsync` fetches more data and does unnecessary work.
|
||
|
|
||
|
Globally replace `SingleOrDefaultAsync` with `FirstOrDefaultAsync`. `SingleOrDefaultAsync` is used in 5 places:
|
||
|
|
||
|
* `OnGetAsync` in the Details page.
|
||
|
* `OnGetAsync` and `OnPostAsync` in the Edit and Delete pages.
|
||
|
|
||
|
## Customize the Details page
|
||
|
|
||
|
Browse to `Pages/Students` page. The **Edit**, **Details**, and **Delete** links are generated by the [Anchor Tag Helper](xref:mvc/views/tag-helpers/builtin-th/anchor-tag-helper)
|
||
|
in the *Pages/Students/Index.cshtml* file.
|
||
|
|
||
|
[!code-cshtml[Main](intro/samples/cu/Pages/Students/Index1.cshtml?range=40-44)]
|
||
|
|
||
|
Select a Details link. The URL is of the form `http://localhost:5000/Students/Details?id=2`. The Student ID is passed using a query string (`?id=2`).
|
||
|
|
||
|
Update the Edit, Details, and Delete Razor Pages to use the `"{id:int}"` route template. Change the page directive for each of these pages from `@page` to `@page "{id:int}"`.
|
||
|
|
||
|
A request to the page with the "{id:int}" route template that does **not** include a integer route value returns an HTTP 404 (not found) error. For example, `http://localhost:5000/Students/Details` returns a 404 error. To make the ID optional, append `?` to the route constraint:
|
||
|
|
||
|
```cshtml
|
||
|
@page "{id:int?}"
|
||
|
```
|
||
|
|
||
|
Run the app, click on a Details link, and verify the URL is passing the ID as route data (`http://localhost:5000/Students/Details/2`).
|
||
|
|
||
|
Don't globally change `@page` to `@page "{id:int}"`, doing so breaks the links to the Home and Create pages.
|
||
|
|
||
|
<!-- See https://github.com/aspnet/Scaffolding/issues/590 -->
|
||
|
|
||
|
### Add related data
|
||
|
|
||
|
The scaffolded code for the Students Index page does not include the `Enrollments` property. In this section, the contents of the `Enrollments` collection is displayed in the Details page.
|
||
|
|
||
|
The `OnGetAsync` method of *Pages/Students/Details.cshtml.cs* uses the `FirstOrDefaultAsync` method to retrieve a single `Student` entity. Add the following highlighted code:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Details.cshtml.cs?name=snippet_Details&highlight=8-12)]
|
||
|
|
||
|
The `Include` and `ThenInclude` methods cause the context to load the `Student.Enrollments` navigation property, and within each enrollment the `Enrollment.Course` navigation property. These methods are examinied in detail in the reading-related data tutorial.
|
||
|
|
||
|
The `AsNoTracking` method improves performance in scenarios when the entities returned are not updated in the current context. `AsNoTracking` is discussed later in this tutorial.
|
||
|
|
||
|
### Display related enrollments on the Details page
|
||
|
|
||
|
Open *Pages/Students/Details.cshtml*. Add the following highlighted code to display a list of enrollments:
|
||
|
|
||
|
<!--2do ricka. if doesn't change, remove dup -->
|
||
|
[!code-cshtml[Main](intro/samples/cu/Pages/Students/Details1.cshtml?highlight=35-53)]
|
||
|
|
||
|
If code indentation is wrong after the code is pasted, press CTRL-K-D to correct it.
|
||
|
|
||
|
The preceding code loops through the entities in the `Enrollments` navigation property. For each enrollment, it displays the course title and the grade. The course title is retrieved from the Course entity that's stored in the `Course` navigation property of the Enrollments entity.
|
||
|
|
||
|
Run the app, select the **Students** tab, and click the **Details** link for a student. The list of courses and grades for the selected student is displayed.
|
||
|
|
||
|
## Update the Create page
|
||
|
|
||
|
Update the `OnPostAsync` method in *Pages/Students/Create.cshtml.cs* with the following code:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Create.cshtml.cs?name=snippet_OnPostAsync)]
|
||
|
|
||
|
<a name="TryUpdateModelAsync"></a>
|
||
|
### TryUpdateModelAsync
|
||
|
|
||
|
Examine the [TryUpdateModelAsync](https://docs.microsoft.com/ dotnet/api/microsoft.aspnetcore.mvc.controllerbase.tryupdatemodelasync?view=aspnetcore-2.0#Microsoft_AspNetCore_Mvc_ControllerBase_TryUpdateModelAsync_System_Object_System_Type_System_String_) code:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Create.cshtml.cs?name=snippet_TryUpdateModelAsync)]
|
||
|
|
||
|
In the preceding code, `TryUpdateModelAsync<Student>` tries to update the `emptyStudent` object using the posted form values from the [PageContext](https://docs.microsoft.com/dotnet/api/microsoft.aspnetcore.mvc.razorpages.pagemodel.pagecontext?view=aspnetcore-2.0#Microsoft_AspNetCore_Mvc_RazorPages_PageModel_PageContext) property in the [PageModel](https://docs.microsoft.com/dotnet/api/microsoft.aspnetcore.mvc.razorpages.pagemodel?view=aspnetcore-2.0). `TryUpdateModelAsync` only updates the properties listed (`s => s.FirstMidName, s => s.LastName, s => s.EnrollmentDate`).
|
||
|
|
||
|
In the preceding sample:
|
||
|
|
||
|
* The second argument (` "student", // Prefix`) is the prefix uses to look up values. It's not case sensitive.
|
||
|
* The posted form values are converted to the types in the `Student` model using [model binding](xref:mvc/models/model-binding#how-model-binding-works).
|
||
|
|
||
|
<a id="overpost"></a>
|
||
|
### Overposting
|
||
|
|
||
|
Using `TryUpdateModel` to update fields with posted values is a security best practice because it prevents overposting. For example, suppose the Student entity includes a `Secret` property that this web page should not update or add:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Models/StudentZsecret.cs?name=snippet_Intro&highlight=7)]
|
||
|
|
||
|
Even if the app doesn't have a `Secret` field on the create/update Razor Page, a hacker could set the `Secret` value by overposting. A hacker could use a tool such as Fiddler, or write some JavaScript, to post a `Secret` form value. The original code doesn't limit the fields that the model binder uses when it creates a Student instance.
|
||
|
|
||
|
Whatever value the hacker specified for the `Secret` form field is updated in the DB. The following image shows the Fiddler tool adding the `Secret` field (with the value "OverPost") to the posted form values.
|
||
|
|
||
|
![Fiddler adding Secret field](../ef-mvc/crud/_static/fiddler.png)
|
||
|
|
||
|
The value "OverPost" is successfully added to the `Secret` property of the inserted row. The app designer never intended the `Secret` property to be set with the Create page.
|
||
|
|
||
|
<a name="vm"></a>
|
||
|
### View model
|
||
|
|
||
|
A view model typically contains a subset of the properties included in the model used by the application. The application model is often called the domain model. The domain model typically contains all the properties required by the corresponding entity in the DB. The view model contains only the properties needed for the UI layer (for example, the Create page). In addition to the view model, some apps use a binding model or input model to pass data between the Razor Pages code-behind class and the browser. Consider the following `Student` view model:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Models/StudentVM.cs)]
|
||
|
|
||
|
View models provide an alternative way to prevent overposting. The view model contains only the properties to view (display) or update.
|
||
|
|
||
|
The following code uses the `StudentVM` view model to create a new student:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/CreateVM.cshtml.cs?name=snippet_OnPostAsync)]
|
||
|
|
||
|
The [SetValues](https://docs.microsoft.com/ dotnet/api/microsoft.entityframeworkcore.changetracking.propertyvalues.setvalues?view=efcore-2.0#Microsoft_EntityFrameworkCore_ChangeTracking_PropertyValues_SetValues_System_Object_) method sets the values of this object by reading values from another [PropertyValues](https://docs.microsoft.com/ dotnet/api/microsoft.entityframeworkcore.changetracking.propertyvalues) object. `SetValues` uses property name matching. The view model type doesn't need to be related to the model type, it just needs to have properties that match.
|
||
|
|
||
|
Using `StudentVM` requires [CreateVM.cshtml](https://github.com/aspnet/Docs/tree/master/aspnetcore/data/ef-rp/intro/samples/cu/Pages/Students/CreateVM.cshtml) be updated to use `StudentVM` rather than `Student`.
|
||
|
|
||
|
In Razor Pages, the `PageModel` derived class is the view model.
|
||
|
|
||
|
## Update the Edit page
|
||
|
|
||
|
Update the `OnPostAsync` method in the Edit page code-behind file:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Edit.cshtml.cs?name=snippet_OnPostAsync)]
|
||
|
|
||
|
The code changes are similar to the Create page with a few exceptions:
|
||
|
|
||
|
* `OnPostAsync` has an optional `id` parameter.
|
||
|
* The current student is fetched from the DB, rather than creating an empty student.
|
||
|
|
||
|
### Test the Edit and Create pages
|
||
|
|
||
|
Create and edit a few student entities.
|
||
|
|
||
|
## Entity States
|
||
|
|
||
|
The DB context keeps track of whether entities in memory are in sync with their corresponding rows in the DB. The DB context sync information determines what happens when `SaveChanges` is called. For example, when a new entity is passed to the `Add` method, that entity's state is set to `Added`. When `SaveChanges` is called, the DB context issues a SQL INSERT command.
|
||
|
|
||
|
An entity may be in one of the following states:
|
||
|
|
||
|
* `Added`: The entity does not yet exist in the DB. The `SaveChanges` method issues an INSERT statement.
|
||
|
|
||
|
* `Unchanged`: No changes need to be saved with this entity. An entity has this status when it is read from the DB.
|
||
|
|
||
|
* `Modified`: Some or all of the entity's property values have been modified. The `SaveChanges` method issues an UPDATE statement.
|
||
|
|
||
|
* `Deleted`: The entity has been marked for deletion. The `SaveChanges` method issues a DELETE statement.
|
||
|
|
||
|
* `Detached`: The entity isn't being tracked by the DB context.
|
||
|
|
||
|
In a desktop app, state changes are typically set automatically. An entity is read, changes are made, and the entity state to automatically be changed to `Modified`. Calling `SaveChanges` generates a SQL UPDATE statement that updates only the changed properties.
|
||
|
|
||
|
In a web app, the `DbContext` that reads an entity and displays the data is disposed after a page is rendered. When a pages `OnPostAsync` method is called, a new web request is made and with a new instance of the `DbContext`. Re-reading the entity in that new context simulates desktop processing.
|
||
|
|
||
|
## Update the Delete page
|
||
|
|
||
|
In this section, code is added to implement a custom error message when the call to `SaveChanges` fails.
|
||
|
|
||
|
Replace the `OnGetAsync` method with the following code:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Delete.cshtml.cs?name=snippet_OnGetAsync&highlight=1,9,17-20)]
|
||
|
|
||
|
The preceding code contains the optional parameter `saveChangesError`. `saveChangesError` indicates whether the method was called after a failure to delete the student object. The delete operation might fail because of transient network problems. Transient network errors are more likely in the cloud. `saveChangesError`is false when the Delete page `OnGetAsync` is called from the UI. When `OnGetAsync` is called by `OnPostAsync` (because the delete operation failed), the `saveChangesError` parameter is true.
|
||
|
|
||
|
### The Delete pages OnPostAsync method
|
||
|
|
||
|
Replace the `OnPostAsync` with the following code:
|
||
|
|
||
|
[!code-csharp[Main](intro/samples/cu/Pages/Students/Delete.cshtml.cs?name=snippet_OnPostAsync)]
|
||
|
|
||
|
The preceding code retrieves the selected entity, then calls the `Remove` method to set the entity's status to `Deleted`. When `SaveChanges` is called, a SQL DELETE command is generated. If `Remove` fails:
|
||
|
|
||
|
* The DB exception is caught.
|
||
|
* The Delete pages `OnGetAsync` method is called with `saveChangesError=true`.
|
||
|
|
||
|
### Update the Delete Razor Page
|
||
|
|
||
|
Add the following highlighted error message to the Delete Razor Page.
|
||
|
|
||
|
[!code-cshtml[Main](intro/samples/cu/Pages/Students/Delete.cshtml?range=1-13&highlight=10)]
|
||
|
|
||
|
Test Delete.
|
||
|
|
||
|
## Common errors
|
||
|
|
||
|
Student/Home or other links don't work:
|
||
|
|
||
|
Verify the Razor Page contains the correct `@page` directive. For example, The Student/Home Razor Page should **not** contain a route template:
|
||
|
|
||
|
```cshtml
|
||
|
@page "{id:int}"
|
||
|
```
|
||
|
|
||
|
Each Razor Page must include the `@page` directive.
|
||
|
|
||
|
>[!div class="step-by-step"]
|
||
|
[Previous](xref:data/ef-rp/intro)
|
||
|
[Next](xref:data/ef-rp/sort-filter-page)
|