ASP.NET Core 集成测试中结合 WebApplicationFactory 使用 SQLite 内存数据库
SQLite 内存数据库(in-memory database)的连接字符串是 Data Source=:memory: ,它的特点是数据库连接一关闭,数据库就会被删除。而使用 services.AddDbContext 通过连接字符串配置 EF Core 时,EF Core 会在每次查询或 SaveChanges 后立即关闭数据库连接。在这样的情况下,集成测试中就无法在向 SQLite 内存数据库写入数据库后进行查询测试。
为了解决上述问题,我们就不能让 EF Core 自己自动维护数据库连接,而只能改为手动模式,手工创建并打开 SqliteConnection 给 EF Core 使用,在用完之后的适当时候关闭连接。
除此之外,由于在每次打开数据库连接都会创建新的数据库,所以还要解决在什么写入数据之前完成数据库的初始化。
结合 WebApplicationFactory ,我们用下面继承自 WebApplicationFactory 的实现代码解决了问题。
public class BlogWebAppFactory<TStartup> : WebApplicationFactory<TStartup> where TStartup : class
{
private DbConnection _dbConnection; public BlogWebAppFactory()
{ } protected override void ConfigureWebHost(IWebHostBuilder builder)
{
base.ConfigureWebHost(builder); builder.ConfigureServices(services =>
{
_dbConnection = new SqliteConnection("Data Source=:memory:");
_dbConnection.Open();
services.AddDbContext<EfUnitOfWork>(options =>
{
options.UseSqlite(_dbConnection);
});
});
} protected override TestServer CreateServer(IWebHostBuilder builder)
{
var server = base.CreateServer(builder); using (var scope = server.Host.Services.CreateScope())
{
var dbContext = scope.ServiceProvider.GetRequiredService<EfUnitOfWork>();
dbContext.Database.EnsureCreated();
} return server;
} protected override void Dispose(bool disposing)
{
base.Dispose(disposing);
_dbConnection?.Dispose();
}
}
集成测试中的示例代码如下
public class PostsWebApiTests : IClassFixture<BlogWebAppFactory<Startup>>
{
private readonly BlogWebAppFactory<Startup> _factory;
private readonly HttpClient _httpClient; public PostsWebApiTests(BlogWebAppFactory<Startup> factory)
{
_factory = factory;
_httpClient = factory.CreateClient();
} [Fact]
public async Task GetPostsByBlogIdsTest()
{
var fakePosts = SeedData();
var blogIds = fakePosts.Select(p => p.BlogID).Distinct();
var response = await _httpClient.PostAsJsonAsync($"/blogposts/blogIds", blogIds);
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
} private IList<BlogPost> SeedData()
{
using (var scope = _factory.Server.Host.Services.CreateScope())
{
var efUnitOfWork = scope.ServiceProvider.GetRequiredService<EfUnitOfWork>(); var blogSites = Builder<BlogSite>.CreateListOfSize().All()
.Do(b => b.BlogID = )
.Build(); var fakePosts = Builder<BlogPost>.CreateListOfSize().All()
.Do(x => x.Id = )
.TheFirst().With(x => x.BlogSite = blogSites[])
.TheNext().With(x => x.BlogSite = blogSites[])
.TheNext().With(x => x.BlogSite = blogSites[])
.Build(); efUnitOfWork.AddRange(fakePosts);
efUnitOfWork.SaveChanges(); return fakePosts;
}
}
}
ASP.NET Core 集成测试中结合 WebApplicationFactory 使用 SQLite 内存数据库的更多相关文章
- ASP.NET Core 集成测试中通过 Serilog 向控制台输出日志
日志是程序员的雷达,不仅在生产环境中需要,在集成测试环境中也需要,可以在持续集成失败后帮助定位问题.与生产环境不同,在集成测试环境中使用控制台输出日志更方便,这样可以通过持续集成 runner 执行 ...
- ASP.NET Core 集成测试中模拟登录用户的一种姿势
不管哪种用户验证方式,最终都是在验证成功后设置 HttpContext.User ,后续处理环节通过 HttpContext.User 获取用户信息.如果能直接修改 HttpContext.User ...
- 如何在 ASP.NET Core 测试中操纵时间?
有时候,我们会遇到一些跟系统当前时间相关的需求,例如: 只有开学季才允许录入学生信息 只有到了晚上或者周六才允许备份博客 注册满 3 天的用户才允许进行一些操作 某用户在 24 小时内被禁止发言 很显 ...
- 通过重建Hosting系统理解HTTP请求在ASP.NET Core管道中的处理流程[下]:管道是如何构建起来的?
在<中篇>中,我们对管道的构成以及它对请求的处理流程进行了详细介绍,接下来我们需要了解的是这样一个管道是如何被构建起来的.总的来说,管道由一个服务器和一个HttpApplication构成 ...
- ASP.NET Core MVC 中的 [Controller] 和 [NonController]
前言 我们知道,在 MVC 应用程序中,有一部分约定的内容.其中关于 Controller 的约定是这样的. 每个 Controller 类的名字以 Controller 结尾,并且放置在 Contr ...
- 如何在ASP.NET Core应用中实现与第三方IoC/DI框架的整合?
我们知道整个ASP.NET Core建立在以ServiceCollection/ServiceProvider为核心的DI框架上,它甚至提供了扩展点使我们可以与第三方DI框架进行整合.对此比较了解的读 ...
- ASP.NET Core MVC 中设置全局异常处理方式
在asp.net core mvc中,如果有未处理的异常发生后,会返回http500错误,对于最终用户来说,显然不是特别友好.那如何对于这些未处理的异常显示统一的错误提示页面呢? 在asp.net c ...
- ASP.NET Core中的缓存[1]:如何在一个ASP.NET Core应用中使用缓存
.NET Core针对缓存提供了很好的支持 ,我们不仅可以选择将数据缓存在应用进程自身的内存中,还可以采用分布式的形式将缓存数据存储在一个“中心数据库”中.对于分布式缓存,.NET Core提供了针对 ...
- 006.Adding a controller to a ASP.NET Core MVC app with Visual Studio -- 【在asp.net core mvc 中添加一个控制器】
Adding a controller to a ASP.NET Core MVC app with Visual Studio 在asp.net core mvc 中添加一个控制器 2017-2-2 ...
随机推荐
- Java基础9-死锁;String;编码
昨日内容回顾 死锁案例 class DeadLock{ public static void main(String[] args){ Pool pool = new Pool(); Producer ...
- $ Django 调API的几种方式
API调用方式 下面是python中会用到的库.urllib2httplib2pycurlrequestsurllib2 #request import requests, json github_u ...
- Gin 路由解析树详解
说明: 无意间看到gin 中有trees的属性,好奇想一探究竟,到底gin是怎样生成路由解析树的? 这是一个测试截图,图中大概可以了解到gin是怎样做路由解析的.配合源码的阅读,解析树大致如下: 通过 ...
- Python-Django-Ajax进阶3
1 中间件 -是什么? 中间件顾名思义,是介于request与response处理之间的一道处理过程,相对比较轻量级,并且在全局上改变django的输入与输出.因为改变的是全局,所以需要谨慎实用,用不 ...
- 27)django-form操作示例(动态Select数据,自定义字段验证,全局验证等)
1)普通传递select数据 # -*- coding:utf-8 -*- __author__ = 'shisanjun' from django import forms from django. ...
- java 根据Url下载对应的文件到指定位置,读txt文件获取url
package test; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; im ...
- 论文阅读笔记四十三:DeeperLab: Single-Shot Image Parser(CVPR2019)
论文原址:https://arxiv.org/abs/1902.05093 github:https://github.com/lingtengqiu/Deeperlab-pytorch 摘要 本文提 ...
- Jmeter性能测试之Monitor监控(SSHMon Samples Collector)
前面写的一篇Monitor监控有缺陷, 这篇文章使用Jmeter4.0+的版本, 使用插件SSHMon Samples Collector来做资源监控 1. 官网下载插件: plugins-manag ...
- python设计模式---创建型之工厂模式
# coding = utf-8 from abc import ABCMeta, abstractmethod # 简单工厂模式 class Animal(metaclass=ABCMeta): @ ...
- python---自己实现双向链表常用功能
这个和单向链表有几个功能是同样的代码. 但在add,insert,append,remove时,由于node拥有prev指针, 所以操作不一样.注意看注释. # coding = utf-8 # 双向 ...