我写了一种无法测试的方法吗?由于我正在使用的库有一个重要的方法,可以实现为扩展方法,所以我似乎无法伪造它。因此,无法测试我的方法。
首先,我将列出要测试的方法的截短版本。然后,我将阐述使用FakeItEasy进行伪造的尝试。
该方法使用缓存,它是对缓存库中静态方法的调用,LazyCache
而我一直在努力进行伪造:
public async Task<BassRuleEditModel> GetBassRuleEditModel(
int facilityId,
int criteriaId,
int bassRuleId,
BassRuleEditDto bassRuleEditDto)
{
var url = _bassRuleService.GetServiceConnectionForFacility(facilityId).Url;
var dto = bassRuleEditDto ?? _bassRuleService.GetBassRuleEditDto(bassRuleId);
var bassRuleEditModel = new BassRuleEditModel
{
...
LocationList = await GetLocations(url),
...
};
...
return bassRuleEditModel;
}
private async Task<IEnumerable<SelectListItem>> GetLocations(string url)
{
var cacheKey = string.Concat(CacheKeys.Location, url);
var selectList = await _appCache.GetOrAddAsync(cacheKey, async () =>
{
return new SelectList(await _tasksAndPrioritiesService.ReturnLocationsAsync(url), NameProperty, NameProperty);
}
, CacheKeys.DefaultCacheLifetime);
return selectList;
}
该GetOrAddAsync
方法是扩展方法。
我只想让假货从缓存中退回一个空的SelectList
。
请注意,AppCache
和所有依赖项都是使用构造函数注入注入的。
我编写的单元测试(在这里我试图伪造)AppCache
是:
[Fact]
public async Task Un_Named_Test_Does_Stuff()
{
var url = "http://somesite.com";
var referrer = new Uri(url);
var facilityId = GetRandom.Id();
var serviceConnectionDto = new ServiceConnectionDto
{
Url = "http://google.com" // this url does not matter
};
var cacheKey = string.Concat(CacheKeys.Location, serviceConnectionDto.Url);
A.CallTo(() => _bassRuleService.GetServiceConnectionForFacility(facilityId)).Returns(serviceConnectionDto);
A.CallTo(() => _urlHelper.Content("~/ServiceSpec/ListView")).Returns(url);
A.CallTo(() => _appViewService.GetReferrer(url)).Returns(referrer);
A.CallTo(() => _appCache.GetOrAddAsync(cacheKey, A<Func<Task<SelectList>>>.Ignored))
.Returns(Task.FromResult(new SelectList(Enumerable.Empty<SelectListItem>().ToList())));
var editModel = await
_bassRuleService.GetBassRuleEditModel(GetRandom.Int32(),
GetRandom.Int32(),
GetRandom.Int32(),
null
);
var path = editModel.Referrer.AbsolutePath;
editModel.Referrer.AbsolutePath.ShouldBe(referrer.AbsolutePath);
}
我在测试的构造函数中创建伪造品(使用xUnit):
public BassRuleQueryServiceTests()
{
_currentUser = A.Fake<ICurrentUser>();
_bassRuleService = A.Fake<IBassRuleService>();
_tasksAndPrioritiesService = A.Fake<ITasksAndPrioritiesService>();
_appViewService = A.Fake<IAppViewService>();
_urlHelper = A.Fake<IUrlHelper>();
_applicationDateTime = A.Fake<IApplicationDateTime>();
_appCache = new MockCacheService();
}
运行测试的错误是:
消息:FakeItEasy.Configuration.FakeConfigurationException:当前代理生成器无法为以下方法拦截方法LazyCache.AppCacheExtenions.GetOrAddAsync
1[Microsoft.AspNetCore.Mvc.Rendering.SelectList](LazyCache.IAppCache cache, System.String key, System.Func
1 [System.Threading.Tasks.Task`1 [Microsoft.AspNetCore.Mvc.Rendering.SelectList]] addItemFactory)原因:-扩展方法是静态的,因此无法被拦截。
我得到的事实是,伪造静态方法未启用。我正在寻找解决方案。
我是否需要向库作者施加压力,使其不要使用扩展方法?(令人发指的问题)
干杯
正如您已经正确指出的那样,扩展是静态方法,并且不能伪造静态方法。
扩展方法通常只是包装器,以简化对其扩展类型的操作。似乎是这种情况。GetOrAddAsync
您要调用的扩展方法最终将调用IAppCache.GetOrAddAsync
方法。因此,您应该改为伪造该方法。
A.CallTo(() => _appCache.GetOrAddAsync(cacheKey, A<Func<ICacheEntry, Task<SelectList>>>.Ignored))
.Returns(new SelectList(Enumerable.Empty<SelectListItem>().ToList()));
这不是很方便,因为这意味着您需要知道扩展方法的作用,但是没有办法解决(除了在库周围创建抽象层之外,但是LazyCache已经是Microsoft.Extensions.Caching.Memory的抽象。 ..)
(顺便说一句,您不需要Task.FromResult
;该Returns
方法具有重载,T
当您配置返回a的方法时,它会接受a Task<T>
)
同样,如果您仍然要返回空序列,则根本不需要配置该方法。FakeItEasy的默认行为是返回一个IEnumerable<SelectListItem>
为空的虚拟对象。
本文收集自互联网,转载请注明来源。
如有侵权,请联系[email protected] 删除。
我来说两句