2016-10-25 10 views
2

Я работаю над проектом Asp.net MVC 5, и я пытаюсь настроить макет для возврата пользовательского принципала в контроллер. У меня есть поиск и попытка использования другого подхода, но никто из них не работает.Как издеваться над HttpContext.User

У меня есть BaseController, из которого все мои контроллеры унаследованы. У BaseController есть свойство User, которое возвращает HttpContext.User в getter. HttpContext.user возвращает значение при вызове внутри проекта, но возвращает нуль при вызове из проекта модульного тестирования.

BaseController

public class BaseController : Controller 
{ 
    protected virtual new CustomPrincipal User 
    { 
     get { return HttpContext.User as CustomPrincipal; } ***<== Line with issue*** 
    } 
} 

Пользовательские Principal

public class CustomPrincipal : IPrincipal, ICustomPrincipal 
{ 
    public IIdentity Identity { get; private set; } 

    public string UserId { get; set; } 
    public string FirstName { get; set; } 
    public string LastName { get; set; } 
    public bool IsStoreUser { get; set; } 

    public CustomPrincipal(string username) 
    { 
     this.Identity = new GenericIdentity(username); 
    } 
} 

Контроллер

public class DocumentsController : BaseController 
    { 
     public ViewResult ViewDocuments() 
     { 
      var userType = User.IsStoreUser ? UserType.StoreUser : UserType.Corporate; ***<== User is null when calling from a unit test.*** 
     } 
    } 

Пример теста

[Test] 
public void ViewDocuments_WhenCalled_ShouldReturnViewModel() 
{ 
    // Arrange 
    var principal = new CustomPrincipal("2038786"); 
    principal.UserId = "2038786"; 
    principal.FirstName = "Test"; 
    principal.LastName = "User"; 
    principal.IsStoreUser = true; 

    var _mockController = new Mock<DocumentsController>(new UnitOfWork(_context)) { CallBase = true }; 
     _mockController.Setup(u => u.User).Returns(principal); ***<== Error - "Invalid setup on a non-virtual (overridable in VB) member: u => u.User"*** 

    // Act 
    var result = _controller.ViewDocuments(); 
} 

Я использую NUnit и Moq для создания фиктивного объекта, но я не уверен, что я делаю неправильно. Мне нужно высмеять возврат User.IsStore в DocumentControl, чтобы вернуть значение IsStore в пользовательский главный объект i, созданный в тесте.

ответ

1

Сделать макет Context HTTP

private class MockHttpContext : HttpContextBase { 
    private readonly IPrincipal user; 

    public MockHttpContext(IPrincipal principal) { 
     this.user = principal; 
    } 

    public override IPrincipal User { 
     get { 
      return user; 
     } 
     set { 
      base.User = value; 
     } 
    } 
} 

Устройте тест соответственно.

[Test] 
public void ViewDocuments_WhenCalled_ShouldReturnViewModel() { 
    // Arrange 
    var principal = new CustomPrincipal("2038786"); 
    principal.UserId = "2038786"; 
    principal.FirstName = "Test"; 
    principal.LastName = "User"; 
    principal.IsStoreUser = true; 

    var mockUoW = new Mock<IUnitOfWork>(); 
    //...setup UoW dependency if needed 
    var controller = new DocumentsController(mockUoW.Object); 
    controller.ControllerContext = new ControllerContext { 
     Controller = controller, 
     HttpContext = new MockHttpContext(principal) 
    }; 

    // Act 
    var result = controller.ViewDocuments(); 

    //Assert 
    //...assertions 
} 

Не тестируйте систему под контролем. Обманите его зависимости.

+0

Спасибо Nkosi, это отлично работает. Спасибо за вашу помощь и предложения. –

0

Это становится намного проще, если вы не зависите от HttpContext напрямую. Создайте интерфейс IUserProvider и реализацию, которая зависит от HttpContext (например, HttpContextUserProvider), а затем в ваших тестах заглушите IUserProvider.

IUserProvider должен быть передан вашему контроллеру через инъекцию зависимости.

+0

Stijn, спасибо за ваш ответ на мой вопрос. У вас есть пример того, что вы предлагаете. Я всегда заинтересован в упрощении/рефакторинг моего кода, где это возможно. Благодарю. –

+0

@EdmandLooi Я попытаюсь включить пример позже на этой неделе. Если я забуду, не стесняйтесь пинговать меня через неделю или около того :) – Stijn

Смежные вопросы