Moq: Invalid setup on a non-overridable member: x => x.GetByTitle(“asdf”)

后端 未结 3 641
盖世英雄少女心
盖世英雄少女心 2020-12-09 00:32

Not sure how I can fix this, trying to do a unit test on the method \"GetByTitle\"

Here are my definitions:

public class ArticleDAO :  GenericNHibern         


        
相关标签:
3条回答
  • 2020-12-09 01:07

    Here's how I Mock HttpMessageHandler:

    private HttpRequestMessage requestMessage = new HttpRequestMessage();
    Mock<HttpMessageHandler> handlerMock = 
    GetHttpMessageHandlerMock(HttpStatusCode.OK);  
    
    MyRestService myRestService = new MyRestService();
    myRestService.client = new HttpClient(handlerMock.Object);
    
    var response = myRestService.Get("");
    

    //At this point, the Mock of HttpRequestMessage is called and the Callback fills my class variable requestMessage. I can now look inside the requestMessage.

    var headers = requestMessage?.Headers.ToString();
    var queryBegin = requestMessage.RequestUri.OriginalString.IndexOf('?');
    var queryString = requestMessage.RequestUri.OriginalString.Substring(queryBegin + 1);
            Assert.That(headers.Contains("x-api-key: fakeApiKey"));
    

    //Helper methods below

    private Mock<HttpMessageHandler> GetHttpMessageHandlerMock(HttpStatusCode statusCode)
    {
            var handlerMock = new Mock<HttpMessageHandler>(MockBehavior.Strict);
            handlerMock
               .Protected()
               .Setup<Task<HttpResponseMessage>>(
                  "SendAsync",
                  ItExpr.IsAny<HttpRequestMessage>()
                 , ItExpr.IsAny<CancellationToken>()
               )
               .Returns(Task.FromResult(GetFakeResponse(statusCode)))
               .Callback<HttpRequestMessage, CancellationToken>((p, q) => requestMessage = p)
              .Verifiable();
            return handlerMock;
        }
    
    
        private HttpResponseMessage GetFakeResponse(HttpStatusCode statusCode)
        {
            var s = "{\"data\":{\"status\":\"SUCCESS\",\"errorCode\":\"\",\"errorMessage\":\"9\"}}";
            HttpResponseMessage response = new HttpResponseMessage()
            {
    
                StatusCode = statusCode,
                Content = new StringContent(s),
                ReasonPhrase = "OK",
                RequestMessage = new HttpRequestMessage()
            };
            return response;
        }
    

    I use this for almost all my REST tests, because I can pass in status, content, etc. So, I can test different return values.

    0 讨论(0)
  • 2020-12-09 01:16

    In order to control the behavior of a mock object (in Moq, at least), you either need to mock an interface, or make sure that the behavior you're trying to control is marked virtual. In your comment, I understand it so that the instantiating of _mockArticleDao is done something like this:

    _mockArticleDao = new Mock<ArticleDAO>();
    

    If you want to keep it as so, you need to mark the GetArticle method virtual:

    public class ArticleDAO :  GenericNHibernateDAO(IArticle, int>, IArticleDAO
    {
        public virtual IArticle GetByTitle(string title)
        {
            // ...
        }
    }
    

    Otherwise (and this is what I recommend), mock the interface instead.

    _mockArticleDao = new Mock<IArticleDAO>();
    
    0 讨论(0)
  • 2020-12-09 01:17

    Create an inherited mockable class

    I had the same issue trying to mock a class I have no control over, from a framework. In my specific case I had to mock an HttpResponseMessage setting up the status code to return Ok, but how to do it if that property is not virtual?

    This code does not work because StatusCode is not virtual:

    var httpResponseMessage = new Mock<HttpResponseMessage>();
    httpResponseMessage.SetupGet(x => x.StatusCode).Returns(HttpStatusCode.OK);
    

    Answer:

    1. Create a new class in your test project, inheriting from the class you want to mock
    2. Redefine the same set of constructors calling the base constructors
    3. Redefine the non virtual properties or methods you want to setup as virtual (use the new keyword to explicitly hide the original members)
    4. From the redefined virtual properties or methods, call the non virtual base property or method.

    Done. Now you can mock a derived object that can be used anywhere the original one is used, because it inherits from it. Here is the code for my MockableHttpResponseMessage class:

    public class MockableHttpResponseMessage: HttpResponseMessage
    {
        public MockableHttpResponseMessage() : base() {}
        public MockableHttpResponseMessage(HttpStatusCode code) : base (code) { }
        public new virtual HttpStatusCode StatusCode { 
            get { return base.StatusCode; } 
            set { base.StatusCode = value; }
        }        
    }
    

    Now, this code works:

    var httpResponseMessage = new Mock<MockableHttpResponseMessage>();
    httpResponseMessage.SetupGet(x => x.StatusCode).Returns(HttpStatusCode.OK);
    
    0 讨论(0)
提交回复
热议问题