PACT.io:当 运行ning npm 运行 pactTest 时出现缺少请求错误
PACT.io: Getting Missing requests error, when running npm run pactTest
已创建测试回购:https://github.com/leongaban/pact-io-js-test
预计
运行 npm run pactTest
将为我的 TotalPayout.test.pact.ts
文件创建一个 Pact 文件。
结果
D, [#38238] DEBUG -- : {
"description": "a GET request with a user id",
"request": {
"method": "GET",
"path": "/frontoffice/api/liquidity-pool/get-total-payout",
"headers": {
"Accept": "application/json"
}
},
"response": {
"status": 200,
"headers": {
"Content-Type": "application/json"
}
}
}
W, [#38238] WARN -- : Verifying - actual interactions do not match expected interactions.
Missing requests:
GET /frontoffice/api/liquidity-pool/get-total-payout
W, [#38238] WARN -- : Missing requests:
GET /frontoffice/api/liquidity-pool/get-total-payout
这是我的契约文件
// @ts-ignore
import path from 'path';
// @ts-ignore
import { Pact } from '@pact-foundation/pact';
import { getTotalPayout } from './LiquidityPool';
// const port = 12345;
const endpoint = '/frontoffice/api/liquidity-pool/get-total-payout';
const EXPECTED_BODY = {
total_payout: 100.21,
};
const userId = 'foo';
describe('The API', () => {
// Copy this block once per interaction under test
describe('getUsersTotalPayout', () => {
beforeEach(() => {
const interaction = {
uponReceiving: 'a GET request with a user id',
withRequest: {
method: 'GET',
path: endpoint,
headers: {
Accept: 'application/json',
},
},
willRespondWith: {
status: 200,
headers: {
'Content-Type': 'application/json'
},
data: EXPECTED_BODY
},
};
// @ts-ignore
return provider.addInteraction(interaction);
});
// add expectations
it('Should call getUsersTotalPayout and return an object with the total_payout', done => {
getTotalPayout(userId)
.then((response: any) => {
console.log('response', response);
console.log('EXPECTED_BODY', EXPECTED_BODY);
expect(response).toEqual(EXPECTED_BODY);
})
.then(done);
});
});
});
这是包含 getTotalPayout
功能的服务文件:
这个端点还不存在,但我的理解是这个 Pact 测试应该仍然有效。
// @TODO Note, this is the placeholder for LiquidityPool API endpoints
// @ts-ignore
import axios, * as others from 'axios';
const endpoint = '/frontoffice/api/liquidity-pool/';
export const getTotalPayout = async (userId: string) => {
const response = await axios.get(`${endpoint}get-total-payout`, { params: userId });
return response.data;
};
还有我的 axios
mock in src/__mocks__/axios.ts
// tslint:disable-next-line:no-empty
const mockNoop = () => new Promise(() => {});
export default {
get: jest.fn(() => Promise.resolve({ data: { total_payout: 100.21 }})),
default: mockNoop,
post: mockNoop,
put: mockNoop,
delete: mockNoop,
patch: mockNoop
};
很简单 - 您的测试 未 在 Pact 模拟服务器上命中路径 /frontoffice/api/liquidity-pool/get-total-payout
。
您已在 http://localhost:1234
上将 Pact 设置为 运行,因此您的实际代码需要配置为访问此服务器,而不是真实服务器。
在您的 axios 配置中,您正在模拟 http 请求库,因此它什么都不做。因此,当您的真实代码使用它时,它不会进行 http 调用并且您的 Pact 测试会失败,因为它期待具有特定形状的调用但没有得到它。
以下是您需要更改的内容:
- Axios 好像用过,但是你的package.json里没有。我猜是因为你在嘲笑它 (2)。你需要一个实际的 http 请求库,所以你应该只安装 axios
- Axios 被模拟以提供固定响应,请勿在 Pact 测试期间模拟 axios(如果需要,本地开发人员可以这样做),因为这将阻止对 Pact 进行真正的调用。 Pact 需要对它进行真正的 http 调用,以便它可以检查您的代码是否在做正确的事情,然后将它发现的内容写入 'contract'
- 在测试中配置 axios 以命中 Pact 模拟服务。假设您使用 axios:
pactSetup.ts:
// Configure axios to use the Pact mock server for Pact tests
import axios from "axios";
axios.defaults.baseURL = "http://localhost:1234";
- 阅读 Pact 吐出的日志文件。它告诉你发生了什么。重申一下:您已经告诉 Pact,应该在
/frontoffice/api/liquidity-pool/get-total-payout
上进行交互,但它从未收到过。让你的实际代码命中这个然后你就没事了。
最后,顺便说一句,一旦你开始生成 Pact(在 1-4 被修复之后),你可能想使用 Pact 作为本地存根服务器进行本地开发。该二进制文件实际上已经安装在您的 node_modules 中,有关其操作方式的文档位于 https://github.com/pact-foundation/pact-ruby-standalone/releases
我通常在 package.json 中有一个类似于以下内容的脚本:
"stubs": "$(find . -name pact-stub-service | head -n 1) pacts/* --port 4000"
然后您可以 运行 npm run stubs
并在端口 4000 上拥有提供程序 运行ning 的本地存根,以及您放入测试中的所有 request/responses .
已创建测试回购:https://github.com/leongaban/pact-io-js-test
预计
运行 npm run pactTest
将为我的 TotalPayout.test.pact.ts
文件创建一个 Pact 文件。
结果
D, [#38238] DEBUG -- : {
"description": "a GET request with a user id",
"request": {
"method": "GET",
"path": "/frontoffice/api/liquidity-pool/get-total-payout",
"headers": {
"Accept": "application/json"
}
},
"response": {
"status": 200,
"headers": {
"Content-Type": "application/json"
}
}
}
W, [#38238] WARN -- : Verifying - actual interactions do not match expected interactions.
Missing requests:
GET /frontoffice/api/liquidity-pool/get-total-payout
W, [#38238] WARN -- : Missing requests:
GET /frontoffice/api/liquidity-pool/get-total-payout
这是我的契约文件
// @ts-ignore
import path from 'path';
// @ts-ignore
import { Pact } from '@pact-foundation/pact';
import { getTotalPayout } from './LiquidityPool';
// const port = 12345;
const endpoint = '/frontoffice/api/liquidity-pool/get-total-payout';
const EXPECTED_BODY = {
total_payout: 100.21,
};
const userId = 'foo';
describe('The API', () => {
// Copy this block once per interaction under test
describe('getUsersTotalPayout', () => {
beforeEach(() => {
const interaction = {
uponReceiving: 'a GET request with a user id',
withRequest: {
method: 'GET',
path: endpoint,
headers: {
Accept: 'application/json',
},
},
willRespondWith: {
status: 200,
headers: {
'Content-Type': 'application/json'
},
data: EXPECTED_BODY
},
};
// @ts-ignore
return provider.addInteraction(interaction);
});
// add expectations
it('Should call getUsersTotalPayout and return an object with the total_payout', done => {
getTotalPayout(userId)
.then((response: any) => {
console.log('response', response);
console.log('EXPECTED_BODY', EXPECTED_BODY);
expect(response).toEqual(EXPECTED_BODY);
})
.then(done);
});
});
});
这是包含 getTotalPayout
功能的服务文件:
这个端点还不存在,但我的理解是这个 Pact 测试应该仍然有效。
// @TODO Note, this is the placeholder for LiquidityPool API endpoints
// @ts-ignore
import axios, * as others from 'axios';
const endpoint = '/frontoffice/api/liquidity-pool/';
export const getTotalPayout = async (userId: string) => {
const response = await axios.get(`${endpoint}get-total-payout`, { params: userId });
return response.data;
};
还有我的 axios
mock in src/__mocks__/axios.ts
// tslint:disable-next-line:no-empty
const mockNoop = () => new Promise(() => {});
export default {
get: jest.fn(() => Promise.resolve({ data: { total_payout: 100.21 }})),
default: mockNoop,
post: mockNoop,
put: mockNoop,
delete: mockNoop,
patch: mockNoop
};
很简单 - 您的测试 未 在 Pact 模拟服务器上命中路径 /frontoffice/api/liquidity-pool/get-total-payout
。
您已在 http://localhost:1234
上将 Pact 设置为 运行,因此您的实际代码需要配置为访问此服务器,而不是真实服务器。
在您的 axios 配置中,您正在模拟 http 请求库,因此它什么都不做。因此,当您的真实代码使用它时,它不会进行 http 调用并且您的 Pact 测试会失败,因为它期待具有特定形状的调用但没有得到它。
以下是您需要更改的内容:
- Axios 好像用过,但是你的package.json里没有。我猜是因为你在嘲笑它 (2)。你需要一个实际的 http 请求库,所以你应该只安装 axios
- Axios 被模拟以提供固定响应,请勿在 Pact 测试期间模拟 axios(如果需要,本地开发人员可以这样做),因为这将阻止对 Pact 进行真正的调用。 Pact 需要对它进行真正的 http 调用,以便它可以检查您的代码是否在做正确的事情,然后将它发现的内容写入 'contract'
- 在测试中配置 axios 以命中 Pact 模拟服务。假设您使用 axios:
pactSetup.ts:
// Configure axios to use the Pact mock server for Pact tests
import axios from "axios";
axios.defaults.baseURL = "http://localhost:1234";
- 阅读 Pact 吐出的日志文件。它告诉你发生了什么。重申一下:您已经告诉 Pact,应该在
/frontoffice/api/liquidity-pool/get-total-payout
上进行交互,但它从未收到过。让你的实际代码命中这个然后你就没事了。
最后,顺便说一句,一旦你开始生成 Pact(在 1-4 被修复之后),你可能想使用 Pact 作为本地存根服务器进行本地开发。该二进制文件实际上已经安装在您的 node_modules 中,有关其操作方式的文档位于 https://github.com/pact-foundation/pact-ruby-standalone/releases
我通常在 package.json 中有一个类似于以下内容的脚本:
"stubs": "$(find . -name pact-stub-service | head -n 1) pacts/* --port 4000"
然后您可以 运行 npm run stubs
并在端口 4000 上拥有提供程序 运行ning 的本地存根,以及您放入测试中的所有 request/responses .