DynamicMock и Expected # 1, Actual # 0 - PullRequest
       29

DynamicMock и Expected # 1, Actual # 0

4 голосов
/ 01 апреля 2011

Я думаю, что что-то не понимаю в Mock, если я использую DynamicMock, он должен только проверить вызов, который я ожидаю, верно?Почему я получаю исключение в моем тесте ниже?Я не хочу проверять, установлен ли сеанс, я только хочу убедиться, что _authService.EmailIsUnique - это вызов с правильными параметрами.Моя проблема состоит в том, что в моем AdminController я устанавливаю _authService.Session ...

 private MockRepository _mock;
 private ISession _session;
 private IAuthenticationService _authService;
 private AdminController _controller;
 private TestControllerBuilder _builder;

 [SetUp]
 public void

 Setup()
 {
        _mock = new MockRepository();
        _session = _mock.DynamicMock<ISession>();
        _authService = _mock.DynamicMock<IAuthenticationService>();
        _controller = new AdminController(_authService, _session);
        _builder = new TestControllerBuilder();
        _builder.InitializeController(_controller);
 }

 [Test]
 public void Register_Post_AddModelErrorWhenEmailNotUnique()
 {
                var userInfo = new RegisterModel();
                userInfo.Email = "the_email@domain.com";

                //_authService.Expect(x => x.Session = _session).Repeat.Once();
                Expect.Call(_authService.EmailIsUnique(userInfo.Email)).Repeat.Once().Return(false);

                _mock.ReplayAll();
                var result = _controller.Register(userInfo);
                var viewResult = (ViewResult)result;

                _authService.VerifyAllExpectations();
                result.AssertViewRendered().ForView("").WithViewData<RegisterModel>();
                Assert.That(viewResult.ViewData.Model, Is.EqualTo(userInfo));
 }

Rhino.Mocks.Exceptions.ExpectationViolationException: IAuthenticationService.set_Session (ISessionProxy2f2f623898f34cbeacf2385bc9ec1;Ожидаемый # 1, фактический # 0.

Спасибо за помощь!

Обновление

Вот часть моего контроллера и моего AuthService ... Я использую Ninject в качестве DIТак как мой AuthService находится в моем Домене, а Ninject находится в моем WebApp, я не знаю, как я мог бы использовать DI в моем AuthService для разрешения моего Сеанса.Еще раз спасибо!

    public partial class AdminController : Controller
    {
        private IAuthenticationService _authService;
        private ISession _session;

        public AdminController(IAuthenticationService authService, ISession session)
        {
            _authService = authService;
            _authService.Session = session;
            _session = session;
        }
[HttpPost]
        [Authorize(Roles = "Admin")]
        [ValidateAntiForgeryToken]
        public virtual ActionResult Register(RegisterModel userInfo)
        {
            if (!_authService.EmailIsUnique(userInfo.Email)) ModelState.AddModelError("Email", Strings.EmailMustBeUnique);
            if (ModelState.IsValid)
            {
                return RegisterUser(userInfo);
            }

            return View(userInfo);
        }
private RedirectToRouteResult RegisterUser(RegisterModel userInfo)
        {
            _authService.RegisterAdmin(userInfo.Email, userInfo.Password);
            var authToken = _authService.ForceLogin(userInfo.Email);
            SetAuthCookie(userInfo.Email, authToken);
            return RedirectToAction(MVC.Auction.Index());
        }
}

public class AuthenticationService : IAuthenticationService
    {
        public ISession Session { get; set; }

public bool EmailIsUnique(string email)
        {
            var user = Session.Single<User>(u => u.Email == email);
            return user == null;
        }
}

1 Ответ

2 голосов
/ 01 апреля 2011

Моя проблема в том, что в моем AdminController я установил _authService.Session

Да, это действительно проблема, так как это ответственность вашей структуры DI, а не кода вашего контроллера. Поэтому, если ISession не используется непосредственно контроллером, его следует удалить из него. Контроллер не должен выполнять какие-либо операции между службой и какими-либо зависимостями этой службы.

Итак, вот пример:

public class AdminController : Controller
{
    private readonly IAuthenticationService _authService;
    public AdminController(IAuthenticationService authService)
    {
        _authService = authService;
    }

    public ActionResult Register(RegisterModel userInfo)
    {
        if (!_authService.EmailIsUnique(userInfo.Email))
        {
            ModelState.AddModelError("Email", Strings.EmailMustBeUnique);
            return View(userInfo);
        }
        return RedirectToAction("Success");
    }
}

Обратите внимание, что контроллер администратора не должен полагаться на какой-либо сеанс. Это уже опирается на IAuthenticationService. Способ реализации этого сервиса не важен. И в этом случае надлежащим модульным тестом будет:

private IAuthenticationService _authService;
private AdminController _controller;
private TestControllerBuilder _builder;

Setup()
{
    _authService = MockRepository.GenerateStub<IAuthenticationService>();
    _controller = new AdminController(_authService);
    _builder = new TestControllerBuilder();
    _builder.InitializeController(_controller);
}

[Test]
public void Register_Post_AddModelErrorWhenEmailNotUnique()
{
    // arrange
    var userInfo = new RegisterModel();
    userInfo.Email = "the_email@domain.com";
    _authService
        .Stub(x => x.EmailIsUnique(userInfo.Email))
        .Return(false);

    // act
    var actual = _controller.Register(userInfo);

    // assert
    actual
        .AssertViewRendered()
        .WithViewData<RegisterModel>()
        .ShouldEqual(userInfo, "");
    Assert.IsFalse(_controller.ModelState.IsValid);
}

UPDATE:

Теперь, когда вы показали свой код, я подтверждаю:

Удалите зависимость ISession из вашего контроллера, поскольку она не нужна, и оставьте это задание платформе DI.

Теперь я вижу, что ваш AuthenticationService сильно зависит от ISession, поэтому внедрение конструктора будет более адаптированным, чем внедрение свойства. Использовать внедрение свойств только для необязательных зависимостей:

public class AuthenticationService : IAuthenticationService
{
    private readonly ISession _session;
    public AuthenticationService(ISession session)
    {
        _session = session;
    }

    public bool EmailIsUnique(string email)
    {
        var user = _session.Single<User>(u => u.Email == email);
        return user == null;
    }
}

и последняя оставшаяся часть - это сантехника. Это делается в приложении ASP.NET MVC, которое имеет ссылки на все остальные уровни. И поскольку вы упомянули Ninject, вы можете установить Ninject.MVC3 NuGet и в сгенерированном ~/App_Start/NinjectMVC3 просто настроить ядро:

/// <summary>
/// Load your modules or register your services here!
/// </summary>
/// <param name="kernel">The kernel.</param>
private static void RegisterServices(IKernel kernel)
{
    kernel.Bind<ISession>().To<SessionImpl>();
    kernel.Bind<IAuthenticationService>().To<AuthenticationService>();
}
...