Sham-It allows you to easily create mock web services that you can use within your tests.
When you need to test a web service which depends on third party web services you are forced to choose between loading the entire platform, creating complicated stub services or mocking out the parts of your codebase that call external services.
As the size of your platform gets bigger, trying to load all of it at once will grind your integration tests to a halt. This inevitably forces you to choose between stub services and mocking out huge chucks of the api.
As the number of tests you write increases, the complexity of your stub services will also have to increase to send your api down the paths you need to test. You will also need to come up with inventive solutions to change your api's responses between tests and check that it was called.
With mocking classes and functions there are plenty of easy to use options for mocking out individual calls within the tests you are writing and writing expectations that they were cakked. Unfortunately with this approach, you aren't testing how your service behaves across the network and the code you've replace with mocks won't get tested.
With sham-it you get the best of both worlds. Each sham that you create is a http server which means you only need to point your code at a different uri. Each sham also provides methods for mocking requests and records a list of all the requests so that you can write expectations about the http calls your code is making.
All of the code to create mock routes and write expectations can be done within your test functions, giving you the convenience and flexibility you get when using mocking libraries as well as the confidence that a network request was made.
First intall the npm module:
npm install sham-it
Then in JavaScript follow the steps below to use sham-it to create a sham, mock out an endpoint, call it then check it was called.
// Step 1: Import sham-it
const shamIt = require("sham-it");
(async function() {
// Step 2: Create a new sham using the defaults
const sham = await shamIt();
// or...
// create a new sham with all options
const shamWithOptions = await shamIt({
// port is the Port that the sham can be access on
port: 9001, // If not specified then one will be chosen automatically
// defaultReply is used when a mocked route isn't matched
defaultReply: {
status: 404, // 404 is the default status of the defaultReply
headers: { "Content-Type": "text/plain" }, // "Content-Type": "text/plain"
// is the default header of the defaultReply
body: "Not Found" // "Not Found" is the default body of the default reply
}
});
// Step 3: Get the uri for the Sham
console.log(`The uri is a property on the Sham Client: ${sham.uri}`); // http://localhost:8000
// Step 4: Mock out an endpoint
const matcher = await sham.addMatcher({
// when is a function for building a matcher. (Required, will throw an error if not supplied)
when: ({ and, equals }) =>
and(equals("method", "GET"), equals("pathname", "/a/b/c")),
// Mocked response that's returned if the matcher function returns true
// (Required, will throw an error if not supplied)
respond: {
status: 200, // Optional: Defaults to 200
headers: { "Content-Type": "application/json" }, // Optional: Defaults to { "Content-Type": "application/json" }
body: { my: "data" } // If an object is supplied it is automatically stringified using JSON.stringify(...)
}
// Optional: You can also pass in a 3rd parameter for the number of times the matcher should match.
// After which it will be deleted, if no value is specified then the matcher will match an unlimited number of times.
});
console.log(matcher); // { id: "...", when: {...}, respond: { status: ..., headers: {...}, body: {...} } }
// Step 5: Send a request to the sham
const fetch = require("node-fetch");
const response = await fetch(`${sham.uri}/a/b/c`);
console.log(response.status === matcher.respond.status); // true
console.log(
response.headers.get("Content-Type") ===
matcher.respond.headers["Content-Type"]
); // true
const body = await response.json();
console.log(body.my === matcher.respond.body.my); // true
// You can also now see the call by calling getRequest();
console.log(await sham.getRequests()); // [{ request: {...}, matcher: {...}, response: {...} }]
// Step 6: In your tests you can expect that the sham received the correct request by doing
expect(
await sham.hasBeenCalledWith(({ and, equals }) =>
and(equals("method", "GET"), equals("pathname", "/a/b/c"))
)
).toBe(true);
// If the expectation fails you will receive full details about the reason why similar to when
// you call expect(jest.fn()).toHaveBeenCalledWith(...)
// Step 7: Reset the mocked routes and calls
await sham.reset();
console.log(await sham.getRequests()); // []
// Step 8: Close the sham to stop it listening
await sham.close();
await shamWithOptions.close();
})();
The sham-it repo contains the following testing examples.
The request-example.js and request-example.test.js are examples of testing a function which is using request to send requests to an api.
Instead of mocking out the request library, you could use sham-it to run integration tests and make sure that the right http calls will actually be sent.
The supertest-example.js and supertest-example.test.js are examples of testing an express api which (using request) calls an external api to validate the incoming requests.
When trying to run integration tests against API's having to mock out calls to external services can be difficult and in doing so undermines the value of the tests. With sham-it you can run integration tests against mock/stub api's, but still get the same convenience and flexibility provided by mocking libraries.
For our future plans and roadmap we are using GitHub Projects. You can see our projects here
If you find any problems with sham-it or have any feature requests please log them here.
Pull requests are more than welcome!
Licensed under MIT.