【问题标题】:PactNet contract test failing with HttpRequestException and SocketExceptionPactNet 合同测试失败并出现 HttpRequestException 和 SocketException
【发布时间】:2021-04-16 09:38:44
【问题描述】:

我正在尝试使用PactNet 为以下方法编写合同测试:

public async Task<IEnumerable<Models.RefData.Instrument> GetInstruments(string issuerCountry, string instrumentType)
{
    ValidateNotNullOrWhiteSpaceParameter(issuerCountry, nameof(issuerCountry));
    ValidateNotNullOrWhiteSpaceParameter(instrumentType, nameof(instrumentType)); ;

    var queryString = $"instruments?issuerCountry={HttpUtility.UrlEncode(issuerCountry)}&instrumentType={HttpUtility.UrlEncode(instrumentType)}";

    int pageNo = 0;
    int pageSize = 20;

    _logger.LogDebug($"GetInstruments Request:{queryString}");

    var httpResponseMessage = await _client.GetAsync(queryString + $"&page={pageNo}&size={pageSize}");

    _logger.LogDebug($"GetInstruments Response Status Code:{httpResponseMessage.StatusCode}");

    switch (httpResponseMessage.StatusCode)
    {
        case HttpStatusCode.OK:
        var content = await httpResponseMessage.Content.ReadAsStringAsync();
        var result = JsonConvert.DeserializeObject<GetInstrumentsResponse>(content);
                    
        // if there are no results, return Empty
        if (result.Metadata.TotalElements == 0)
        {
            return Enumerable.Empty<Models.RefData.Instrument>();
        }

        var instruments = new List<Models.RefData.Instrument>();
        instruments.AddRange(result.Embedded.Instruments);
        for (pageNo = 1; pageNo < result.Metadata.TotalPages; pageNo++)
        {
            var innerQueryString = queryString + $"&page={pageNo}&size={pageSize}";

            _logger.LogDebug($"GetInstruments Request Additional Page:{innerQueryString}");

            var httpResponseMessage2 = await _client.GetAsync(innerQueryString);
            if (httpResponseMessage2.StatusCode != HttpStatusCode.OK)
            {
                _logger.LogError($"The requested page number {pageNo} gets response error {httpResponseMessage2.StatusCode.ToString()}.");
                throw new UnexpectedResponseException(httpResponseMessage.StatusCode);
            }
                var content2 = await httpResponseMessage2.Content.ReadAsStringAsync();
                var result2 = JsonConvert.DeserializeObject<GetInstrumentsResponse>(content2);

                if (result2.Embedded.Instruments != null && result2.Embedded.Instruments.Any())
                {
                    instruments.AddRange(result2.Embedded.Instruments);
                }
            }

            if (instruments.Count != result.Metadata.TotalElements)
            {
                _logger.LogWarning($"Total number of instruments is different from MetaData.  MetaData states {result.Metadata.TotalElements}, however only {instruments.Count} instruments retrieved.");
        }

        _logger.LogDebug($"GetInstruments Result:{instruments.ToJson()}");

        return instruments;
        default:
            throw new UnexpectedResponseException(httpResponseMessage.StatusCode);
    }
}

我使用this 作为指导创建了以下ConsumerPactTests.csConsumerPactClassFixture.cs

public class ConsumerPactTests : IClassFixture<ConsumerPactClassFixture>
{
    private IMockProviderService _mockProviderService;
    private string _mockProviderServiceBaseUri;

    public ConsumerPactTests(ConsumerPactClassFixture fixture)
    {
        _mockProviderService = fixture.MockProviderService;
        _mockProviderService.ClearInteractions(); //NOTE: Clears any previously registered interactions before the test is run
        _mockProviderServiceBaseUri = fixture.MockProviderServiceBaseUri;
    }

    [Fact]
    public void ItHandlesInvalidDateParam()
    {
        // Arange
        var invalidRequestMessage = "issuerCountry or instrumentType is not valid";
        _mockProviderService.Given("There is data")
            .UponReceiving("A invalid GET request for Date Validation with invalid date parameter")
            .With(new ProviderServiceRequest
            {
                Method = HttpVerb.Get,
                Path = "/api/v2",
                Query = "issuerCountry=USA&instrumentType=foo"
            })
            .WillRespondWith(new ProviderServiceResponse
            {
                Status = 400,
                Headers = new Dictionary<string, object>
                {
                    { "Content-Type", "application/json; charset=utf-8" }
                },
                Body = new
                {
                    message = invalidRequestMessage
                }
            });

        // Act
        RefDataHttpService sut = new RefDataHttpServiceBuilder().Build();
        var result = sut.GetInstruments("USA", "foo").GetAwaiter().GetResult();
        var resultBodyText = result.GetEnumerator();

        // Assert
        Assert.NotNull(resultBodyText);
    }
}



public class ConsumerPactClassFixture : IDisposable
{
    public IPactBuilder PactBuilder { get; private set; }
    public IMockProviderService MockProviderService { get; private set; }

    public int MockServerPort { get { return 9222; } }
    public string MockProviderServiceBaseUri { get { return String.Format("http://localhost:{0}", MockServerPort); } }

   public ConsumerPactClassFixture()
   {
       var pactConfig = new PactConfig
       {
           SpecificationVersion = "2.0.0",
           PactDir = @"..\..\..\..\..\pacts",
           LogDir = @".\pact_logs"
       };

       PactBuilder = new PactBuilder(pactConfig);

       PactBuilder.ServiceConsumer("Consumer")
           .HasPactWith("Provider");

       MockProviderService = PactBuilder.MockService(MockServerPort);
    }

    #region IDisposable Support
    private bool disposedValue = false; // To detect redundant calls

    protected virtual void Dispose(bool disposing)
    {
        if (!disposedValue)
        {
            if (disposing)
            {
                // This will save the pact file once finished.
                PactBuilder.Build();
            }

            disposedValue = true;
        }
    }

    // This code added to correctly implement the disposable pattern.
    public void Dispose()
    {
        // Do not change this code. Put cleanup code in Dispose(bool disposing) above.
        Dispose(true);
    }
    #endregion
}

当我运行我的测试时,我得到了这个错误:

dotnet test --filter "FullyQualifiedName=Bond.Publisher.Tests.ContractTest.ConsumerPactTests.ItHandlesInvalidDateParam"

测试运行 c:\Workspace\prod\test\Bond.Publisher.Tests\bin\Debug\netcoreapp3.1\Bond.Publisher.Tests.dll(.NETCoreApp,Version=v3.1) Microsoft (R) 测试执行命令行工具版本 16.7.0 版权所有 (c) 微软公司。保留所有权利。

开始测试执行,请稍候...

共有 1 个测试文件与指定的模式匹配。 [xUnit.net 00:00:10.95] Bond.Publisher.Tests.ContractTest.ConsumerPactTests.ItHandlesInvalidDateParam [失败] X Bond.Publisher.Tests.ContractTest.ConsumerPactTests.ItHandlesInvalidDateParam [4s 196ms] 错误信息: System.Net.Http.HttpRequestException :无法建立连接,因为目标机器主动拒绝了它。 ---- System.Net.Sockets.SocketException : 由于目标机器主动拒绝,无法建立连接。 堆栈跟踪: 在 System.Net.Http.ConnectHelper.ConnectAsync(字符串主机,Int32 端口,CancellationToken 取消令牌) 在 System.Net.Http.HttpConnectionPool.ConnectAsync(HttpRequestMessage 请求,布尔 allowHttp2,CancellationToken 取消令牌) 在 System.Net.Http.HttpConnectionPool.CreateHttp11ConnectionAsync(HttpRequestMessage 请求,CancellationToken 取消令牌) 在 System.Net.Http.HttpConnectionPool.GetHttpConnectionAsync(HttpRequestMessage 请求,CancellationToken cancelToken) 在 System.Net.Http.HttpConnectionPool.SendWithRetryAsync(HttpRequestMessage 请求,布尔 doRequestAuth,CancellationToken 取消令牌) 在 System.Net.Http.RedirectHandler.SendAsync(HttpRequestMessage 请求,CancellationToken cancelToken) 在 c:\Workspace\usprod\src\Bond.Publisher\HttpMessageHandlers\UnoAuthorisationHeaderMessageHandler.cs:line 37 中的 Bond.Publisher.HttpMessageHandlers.UnoAuthorisationHeaderMessageHandler.SendAsync(HttpRequestMessage request, CancellationToken cancelToken) 在 System.Net.Http.HttpClient.FinishSendAsyncBuffered(Task`1 sendTask, HttpRequestMessage 请求, CancellationTokenSource cts, Boolean disposeCts) 在 c:\Workspace\prod\src\Bond.Publisher\Services\RefDataHttpService.cs:line 52 中的 Bond.Publisher.Services.RefDataHttpService.GetInstruments(String issuerCountry, String instrumentType) 在 Bond.Publisher.Tests.ContractTest.ConsumerPactTests.ItHandlesInvalidDateParam() 在 c:\Workspace\prod\test\Bond.Publisher.Tests\ContractTest\ConsumerPactTests.cs:line 52 ----- 内部堆栈跟踪 ----- 在 System.Net.Http.ConnectHelper.ConnectAsync(字符串主机,Int32 端口,CancellationToken 取消令牌)

测试运行失败。

我怀疑这可能是某种身份验证问题,因为UnoAuthorisationHeaderMessageHandler.cs 正在处理该问题。我做错了什么?

【问题讨论】:

    标签: c# .net testing pact-net


    【解决方案1】:

    对我来说,这条路太长了。当我将项目移动到更靠近 C:\ 的文件夹时,测试运行了。

    【讨论】:

      【解决方案2】:

      System.Net.Sockets.SocketException : 无法建立连接,因为目标机器主动拒绝。通常发生在没有服务器监听您发送的 url 时。

      检查 ruby​​ 服务是否启动并运行(测试运行程序启动它),您应该在任务管理器中的Visual Studio 下看到它

      或者,在您调用 pactBuilder.Build() 之前,您应该能够通过 PostMan 向 http://localhost:9222/instruments...发出 HTTP 请求...

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2022-08-10
        • 2015-05-28
        • 2014-12-02
        • 2020-05-25
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多