当用户仅提供部分字段时发布实体 class

Posting entity class when only some fields are provided by user

我很好奇人们是如何处理这个问题的。

我已经使用 Visual Studio 将带有 Entity Framework 的 CRUD 页面添加到我的 Razor Pages 应用程序。

但是,Create 页面无法验证。那是因为该实体具有诸如用户 ID、具有默认值的创建日期以及我想要设置默认值的几个字段等字段。所以 ModelState.IsValid 显然是要 return false。而且似乎很少有用户会为实体的每个字段提供完全正确的数据。

处理这个问题的一个选择是创建一个新的 class,它只包含我需要的用户字段,然后将它们复制到实体中,然后再将其保存到数据库中。

另一种选择是编写代码,在调用 ModelState.IsValid 之前填充不依赖于用户输入的字段。

这对我来说是一个新情况。我很想听听其他人是如何处理这种情况的。

我个人总是通过在前端使用一个单独的 class 来抽象 EF class 来处理这种情况。这提供了以下好处:

  1. 仅允许用户在保存前填写某些字段(例如您的案例),但不允许用户填写 ID、时间戳等其他字段。
  2. 减少耦合 - 允许更轻松地重构前端或后端代码,而无需也重构另一个(前提是前端 class 不变)。
  3. [编辑] 这可以防止任何用户编辑 ID 等不应由用户更新的字段。

这样做的一个缺点是它会导致相似 class 之间的许多映射,但是自动映射器工具可以在一定程度上缓解这种情况。

我使用隐藏输入。

<hidden asp-for="UserId" value=@GetUser() />
<hidden asp-for="CreatedDate" value=@DateTime.Now.ToString("mm/dd/yyyy") />

它们将作为表单的一部分与模型的其余部分一起提交。这是最简单的解决方案,imo。

在我看来,如果您可以使用 AutoMapper 而不是复制值,那么您的第一个选择会更加简单和安全。

参考下面使用asp.net core 3.0 Razor Pages 的演示:

1.Install-Package AutoMapper.Extensions.Microsoft.DependencyInjection

2.Create模型(保存到数据库)和ViewModel(显示在用户页面)

public class Movie
{
    public Movie()
    {
        Genre = "Action";        
    }
    public int ID { get; set; }
    public string Title { get; set; }

    [DataType(DataType.Date)]
    public DateTime ReleaseDate { get; set; }
    public string Genre { get; set; }

}

public class MovieViewModel
{

    public string Title { get; set; }

    [DataType(DataType.Date)]
    public DateTime ReleaseDate { get; set; }

}

3.Create AutoMapper Profile class

public class MovieProfile: Profile    
{
    public MovieProfile()
    {
        CreateMap<MovieViewModel, Movie>().ReverseMap();
    }

}

4.Create.cshtml.cs

public class CreateModel : PageModel
{
    private readonly ApplicationContext _context;
    private readonly IMapper _mapper;
    public CreateModel(ApplicationContext  context, IMapper mapper)
    {
        _context = context;
        _mapper = mapper;
    }

    public IActionResult OnGet()
    {
        return Page();
    }

    [BindProperty]
    public MovieViewModel MovieVM { get; set; }

    public async Task<IActionResult> OnPostAsync()
    {
        if (!ModelState.IsValid)
        {
            return Page();
        }

        Movie movie = _mapper.Map<Movie>(MovieVM);
        _context.Movie.Add(movie);
        await _context.SaveChangesAsync();

        return RedirectToPage("./Index");
    }
}

Create.cshtml

@page
@model RazorpagesCore.Pages.Movies.CreateModel
<div class="row">
    <div class="col-md-4">
        <form method="post">
            <div asp-validation-summary="ModelOnly" class="text-danger"></div>
            <div class="form-group">
                <label asp-for="MovieVM.Title" class="control-label"></label>
                <input asp-for="MovieVM.Title" class="form-control" />
                <span asp-validation-for="MovieVM.Title" class="text-danger"></span>
            </div>
            <div class="form-group">
                <label asp-for="MovieVM.ReleaseDate" class="control-label"></label>
                <input asp-for="MovieVM.ReleaseDate" class="form-control" />
                <span asp-validation-for="MovieVM.ReleaseDate" class="text-danger"></span>
            </div>
            <div class="form-group">
                <input type="submit" value="Create" class="btn btn-primary" />
            </div>
        </form>
    </div>
</div>
@section Scripts {
    @{await Html.RenderPartialAsync("_ValidationScriptsPartial");}
}