ASP NET Core ---POST, PUT, PATCH, DELETE,Model 验证
参照 草根专栏- ASP.NET Core + Ng6 实战:https://v.qq.com/x/page/u0765jbwc6f.html
一、POST
安全性和幂等性
- 安全性是指方法执行后并不会改变资源的表述
- 幂等性是指方法无论执行多少次都会得到同样的结果

POST添加资源:
- 不安全, 不幂等
- 参数 [FromBody]
- 返回 201 Create : CreatedAtRoute(): 它允许响应里带着Location Header,在这个Location Header里包含着一个uri,通过这个uri就可以GET到我们刚刚创建好的资源
- HATEOAS
(1)添加 PostAddResource.cs 类
namespace BlogDemo.Infrastructure.Resources
{
public class PostAddResource
{
public string Title { get; set; }
public string Body { get; set; } public string Remark { get; set; }
}
}
(2)添加映射
namespace BlogDemo.Api.Extensions
{
public class MappingProfile:Profile
{
public MappingProfile()
{
CreateMap<Post, PostDTO>().ForMember(dest=>dest.Updatetime,opt=>opt.MapFrom(src=>src.LastModified));
CreateMap<PostDTO, Post>();
CreateMap<PostAddResource, Post>();
}
}
}
(3)添加post方法:
[HttpPost(Name = "CreatePost")]
public async Task<IActionResult> Post([FromBody] PostAddResource postAddResource)
{
if (postAddResource == null)
{
return BadRequest();
} var newPost = _mapper.Map<PostAddResource, Post>(postAddResource); newPost.Author = "admin";
newPost.LastModified = DateTime.Now; _postRepository.AddPost(newPost); if (!await _unitOfWork.SaveAsync())
{
throw new Exception("Save Failed!");
} var resultResource = _mapper.Map<Post, PostDTO>(newPost); var links = CreateLinksForPost(newPost.Id);
var linkedPostResource = resultResource.ToDynamic() as IDictionary<string, object>;
linkedPostResource.Add("links", links); return CreatedAtRoute("GetPost", new { id = linkedPostResource["Id"] }, linkedPostResource);
}
(4) 测试:

二、Model验证:
定义验证规则
检查验证规则
把验证错误信息发送给API的消费者
1、验证方式:
内置验证:
DataAnnotation
ValidationAttribute
IValidatebleObject
第三方: FluentValidation
2、使用FluentValidation组件(关注点分离)
(1) 安装:
FluentValidation.AspNetCore
FluentValidation
(2) 为每一个Resource建立验证器:
继承AbstractValidator<T>
namespace BlogDemo.Infrastructure.Resources
{
public class PostAddResourceValidator:AbstractValidator<PostAddResource>
{
public PostAddResourceValidator()
{
RuleFor(x => x.Title).NotEmpty()
.WithName("标题").WithMessage("{PropertyName}是必须填写的")
.MaximumLength().WithMessage("{PropertyName}的最大长度是{MaxLength}"); RuleFor(x => x.Body).NotEmpty()
.WithName("正文").WithMessage("{PropertyName}是必须填写的")
.MinimumLength().WithMessage("{PropertyName}的最大长度是{MinLength}");
}
}
}
(3)配置:
services.AddMvc(……).AddFluentValidation();
services.AddTransient<IValidator<PostAddResource>, PostAddResourceValidator>();
(4)Action添加验证:
ModelState.IsValid
ModelState
它是一个字典,包含了Model的状态以及Model所绑定的验证
对于提交的每个属性,它都包含了一个错误信息的集合
返回: 422 Unprocessable Entity
验证错误信息在响应的body里面带回去
if (!ModelState.IsValid)
{
return UnprocessableEntity(ModelState);
}
(5)测试

(6)Action添加Accpet和Content-Type 的自定义hateoas
[RequestHeaderMatchingMediaType("Content-Type", new[] { "application/vnd.cgzl.post.create+json" })]
[RequestHeaderMatchingMediaType("Accept", new[] { "application/vnd.cgzl.hateoas+json" })]
(7) staupDevelopment 注册hateoas
services.AddMvc(option => {
option.ReturnHttpNotAcceptable = true;
// option.OutputFormatters.Add(new XmlDataContractSerializerOutputFormatter());
var outputFormatter = option.OutputFormatters.OfType<JsonOutputFormatter>().FirstOrDefault();
if (outputFormatter != null)
{
outputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.hateoas+json");
}
var intputFormatter = option.InputFormatters.OfType<JsonInputFormatter>().FirstOrDefault();
if (intputFormatter != null)
{
intputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.post.create+json");
intputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.post.update+json");
}
})
(8)测试:

3、自定义错误验证返回结果:
(1) 添加MyUnprocessableEntityObjectResult.cs ResourceValidationError.cs ResourceValidationResult.cs 类
namespace BlogDemo.Api.Helpers
{
public class MyUnprocessableEntityObjectResult : UnprocessableEntityObjectResult
{
public MyUnprocessableEntityObjectResult(ModelStateDictionary modelState) : base(new ResourceValidationResult(modelState))
{
if (modelState == null)
{
throw new ArgumentNullException(nameof(modelState));
}
StatusCode = ;
}
}
}
namespace BlogDemo.Api.Helpers
{
public class ResourceValidationError
{
public string ValidatorKey { get; private set; }
public string Message { get; private set; } public ResourceValidationError(string message, string validatorKey = "")
{
ValidatorKey = validatorKey;
Message = message;
}
}
}
namespace BlogDemo.Api.Helpers
{
public class ResourceValidationResult : Dictionary<string, IEnumerable<ResourceValidationError>>
{
public ResourceValidationResult() : base(StringComparer.OrdinalIgnoreCase)
{ } public ResourceValidationResult(ModelStateDictionary modelState)
: this()
{
if (modelState == null)
{
throw new ArgumentNullException(nameof(modelState));
} foreach (var keyModelStatePair in modelState)
{
var key = keyModelStatePair.Key;
var errors = keyModelStatePair.Value.Errors;
if (errors != null && errors.Count > )
{
var errorsToAdd = new List<ResourceValidationError>();
foreach (var error in errors)
{
var keyAndMessage = error.ErrorMessage.Split('|'); if (keyAndMessage.Length > )
{
errorsToAdd.Add(new ResourceValidationError(keyAndMessage[], keyAndMessage[]));
}
else
{
errorsToAdd.Add(new ResourceValidationError(keyAndMessage[]));
}
}
Add(key, errorsToAdd);
}
}
}
}
}
(2)Action中添加自定义验证:
public async Task<IActionResult> Post([FromBody] PostAddResource postAddResource)
{ if (!ModelState.IsValid)
{
return new MyUnprocessableEntityObjectResult(ModelState);
}
}
(3)测试

三、Delete
1、 在PostRepository 添加Delete方法
public void Delete(Post post)
{
_myContext.Posts.Remove(post);
}
2、Action添加Delete方法:
[HttpDelete("{id}", Name = "DeletePost")]
public async Task<IActionResult> DeletePost(int id)
{
var post = await _postRepository.GetPostId(id);
if (post == null)
{
return NotFound();
}
_postRepository.Delete(post);
if (!await _unitOfWork.SaveAsync())
{
throw new Exception($"Deleting post {id} failed when saving.");
}
return NoContent();
}
四、PUT(整体更新)
1、添加PostUpdateResource.cs 类;
public class PostUpdateResource : PostAddOrUpdateResource
{ }
2、重构 PostAddOrUpdateResourceValidator.cs ,改成泛型
namespace BlogDemo.Infrastructure.Resources
{
public class PostAddOrUpdateResourceValidator<T> : AbstractValidator<T> where T: PostAddOrUpdateResource
{
public PostAddOrUpdateResourceValidator()
{
RuleFor(x => x.Title).NotEmpty()
.WithName("标题").WithMessage("required|{PropertyName}是必须填写的")
.MaximumLength().WithMessage("maxLength|{PropertyName}的最大长度是{MaxLength}"); RuleFor(x => x.Body).NotEmpty()
.WithName("正文").WithMessage("required|{PropertyName}是必须填写的")
.MinimumLength().WithMessage("minLength|{PropertyName}的最小长度是{MinLength}");
}
}
}
3、资源验证注册:
services.AddTransient<IValidator<PostUpdateResource>, PostAddOrUpdateResourceValidator<PostUpdateResource>>();
4、添加Map
namespace BlogDemo.Api.Extensions
{
public class MappingProfile:Profile
{
public MappingProfile()
{
CreateMap<Post, PostDTO>().ForMember(dest=>dest.Updatetime,opt=>opt.MapFrom(src=>src.LastModified));
CreateMap<PostDTO, Post>();
CreateMap<PostAddResource, Post>();
CreateMap<PostUpdateResource, Post>();
}
}
}
5、添加自定义hateoas
public void ConfigureServices(IServiceCollection services)
{
services.AddMvc(option => {
option.ReturnHttpNotAcceptable = true;
// option.OutputFormatters.Add(new XmlDataContractSerializerOutputFormatter());
var outputFormatter = option.OutputFormatters.OfType<JsonOutputFormatter>().FirstOrDefault();
if (outputFormatter != null)
{
outputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.hateoas+json");
} var intputFormatter = option.InputFormatters.OfType<JsonInputFormatter>().FirstOrDefault();
if (intputFormatter != null)
{
intputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.post.create+json");
intputFormatter.SupportedMediaTypes.Add("application/vnd.cgzl.post.update+json");
} })
}
6、Action添加PUT方法:
[HttpPut("{id}", Name = "UpdatePost")]
[RequestHeaderMatchingMediaType("Content-Type", new[] { "application/vnd.cgzl.post.update+json" })]
public async Task<IActionResult> UpdatePost(int id, [FromBody] PostUpdateResource postUpdate)
{
if (postUpdate == null)
{
return BadRequest();
}
if (!ModelState.IsValid)
{
return new MyUnprocessableEntityObjectResult(ModelState);
}
var post = await _postRepository.GetPostId(id);
if (post == null)
{
return NotFound();
}
post.LastModified = DateTime.Now;
_mapper.Map(postUpdate, post);
if (!await _unitOfWork.SaveAsync())
{
throw new Exception($"Updating post {id} failed when saving.");
}
return NoContent();
}
7、测试



五、Patch(局部更新)
1、 PostRepository 添加Update方法:
public void Update(Post post)
{
_myContext.Entry(post).State = EntityState.Modified;
}
2、Action添加Patch方法:
[HttpPatch("{id}", Name = "PartiallyUpdatePost")]
public async Task<IActionResult> PartiallyUpdateCityForCountry(int id,
[FromBody] JsonPatchDocument<PostUpdateResource> patchDoc)
{
if (patchDoc == null)
{
return BadRequest();
}
var post = await _postRepository.GetPostByIdAsync(id);
if (post == null)
{
return NotFound();
}
var postToPatch = _mapper.Map<PostUpdateResource>(post);
patchDoc.ApplyTo(postToPatch, ModelState);
TryValidateModel(postToPatch);
if (!ModelState.IsValid)
{
return new MyUnprocessableEntityObjectResult(ModelState);
}
_mapper.Map(postToPatch, post);
post.LastModified = DateTime.Now;
_postRepository.Update(post);
if (!await _unitOfWork.SaveAsync())
{
throw new Exception($"Patching city {id} failed when saving.");
}
return NoContent();
}
3、测试



六、HTTP常用方法总结

ASP NET Core ---POST, PUT, PATCH, DELETE,Model 验证的更多相关文章
- 008.Adding a model to an ASP.NET Core MVC app --【在 asp.net core mvc 中添加一个model (模型)】
Adding a model to an ASP.NET Core MVC app在 asp.net core mvc 中添加一个model (模型)2017-3-30 8 分钟阅读时长 本文内容1. ...
- 创建ASP.NET Core MVC应用程序(6)-添加验证
创建ASP.NET Core MVC应用程序(6)-添加验证 DRY原则 DRY("Don't Repeat Yourself")是MVC的设计原则之一.ASP.NET MVC鼓励 ...
- ASP.NET MVC基于标注特性的Model验证:将ValidationAttribute应用到参数上
原文:ASP.NET MVC基于标注特性的Model验证:将ValidationAttribute应用到参数上 ASP.NET MVC默认采用基于标准特性的Model验证机制,但是只有应用在Model ...
- ASP.NET MVC基于标注特性的Model验证:一个Model,多种验证规则
原文:ASP.NET MVC基于标注特性的Model验证:一个Model,多种验证规则 对于Model验证,理想的设计应该是场景驱动的,而不是Model(类型)驱动的,也就是对于同一个Model对象, ...
- 如何为ASP.NET Core设置客户端IP白名单验证
原文链接:Client IP safelist for ASP.NET Core 作者:Damien Bowden and Tom Dykstra 译者:Lamond Lu 本篇博文中展示了如何在AS ...
- Asp.net Core, 基于 claims 实现权限验证 - 引导篇
什么是Claims? 这个直接阅读其他大神些的文章吧,解释得更好. 相关文章阅读: http://www.cnblogs.com/JustRun1983/p/4708176.html http://w ...
- 【ASP.NET Core】JSON Patch 使用简述
JSON Patch 是啥玩意儿?不知道,直接翻译吧,就叫它“Json 补丁”吧.干吗用的呢?当然是用来修改 JSON 文档的了.那咋修改呢?比较常见有四大操作:AMRR. 咋解释呢? A—— Add ...
- Asp.Net Core 入门(四)—— Model、View、Controller
和我们学习Asp.Net MVC一样,Asp.Net Core MVC的Model.View.Controller也和我们熟悉的Asp.Net MVC中的相似.不同的是我们在使用Asp.Net Cor ...
- ASP.NET Core MVC 之模型(Model)
1.模型绑定 ASP.NET Core MVC 中的模型绑定将数据从HTTP请求映射到操作方法参数.参数既可以是简单类型,也可以是复杂类型.MVC 通过抽象绑定解决了这个问题. 2.使用模型绑定 当 ...
随机推荐
- 【洛谷P1966】火柴排队
火柴排队 题目链接 ∑(ai−bi)^2=∑ai^2-2*∑ai*bi+∑bi^2 显然∑ai^2+∑bi^2是不变的,我们要让 2*∑ai*bi最大,才能使原式最小 然后我们一眼就可以看出来, ...
- 封装方法到对象(javascript)
/*! * artDialog 5 * Date: 2012-03-21 * http://code.google.com/p/artdialog/ * (c) 2009-2012 TangBin, ...
- 【题解】洛谷P1002过河卒
首先,一道入门DP 然而对于蒟蒻的我已经难到爆了好吗 第一点:动态转移方程 用DP的关键! 这题我们可以发现每一步的方案数由上面的那步加上左边的那步得到 所以自然而然的方程就出来了: f[i][k]= ...
- 关于Echarts的原生js获取DOM元素与动态加载DOM元素的冲突问题
1.前言: 最近在做的看板项目,因为需要循环加载后台数据,并且用Echarts做数据呈现,所以jQuery和angular等库统统靠边站,Echarts用的是原生js获取DOM元素,至于诸多不兼容等深 ...
- katalon安装 appium with mac 遇到的坑
1. Install Homebrew from Terminal: /usr/bin/ruby -e "$(curl -fsSL https://raw.gi ...
- 纯 HTML5 APP与原生APP的差距在哪?
纯 HTML5 APP与原生APP的差距在哪? 写过一些纯H5的APP,虽然开发起来的确很快很舒服,但和原生比起来纯H5APP还是有很多问题,主要聚集在以下几个方面: 1.动画 动画有很多种,比如侧边 ...
- span没有name属性
<span id="test" name="测试数据">测试咯</span> 在eclipse中这么写发现会有警告提示.百度发现原来sp ...
- Qt数据库编程1
Qt中数据编程主要分为以下两点:1.利用qt提供类 访问数据库或者成为简单的数据库编程2.数据库编程中引入model/view编程模型 qt中数据库编程的步骤: 1.加载数据库驱动 QSqlDatab ...
- ETO的公开赛T4《对抗水滴》 题解(BY 萌萌哒123456 )
题意: 给你一个\(n*n\)的矩阵A,其中有\(T\)个元素不为零.定义矩阵内元素\((x,y)\)的能量值 \(E[x][y]=\sum_{i=1}^{x}\sum_{j=1}^{y}[A[i][ ...
- 独木舟(51NOD 1432 )
n个人,已知每个人体重.独木舟承重固定,每只独木舟最多坐两个人,可以坐一个人或者两个人.显然要求总重量不超过独木舟承重,假设每个人体重也不超过独木舟承重,问最少需要几只独木舟? Input 第一行包含 ...
