Writing a behaviour driven API testing environment within Postman
Behaviour driven test frameworks like mocha and jasmine have certain unalienable advantages. They provide a clean way to organize tests as well as safe and (in this particular case,) isolated assertions to prevent runtime error of the testing spec itself.
For obvious reasons, having the same test specification style in Postman sounds just about right. After all, one may argue that they are testing the behavior of the response! And not to mention the endless benefits of well organized tests, seamless creation and teardown of fixtures, yada yada yada!
I would personally love to write tests within Postman that look like the following. (Never mind the aforementioned xkcd :-/ )
describe("response", function () { var responseJSON; it("must be a valid JSON", function () { responseJSON = JSON.parse(responseBody); expect(typeof responseJSON).toBe("object"); }); describe("json", function () { it("must not have error object", function () { expect(responseJSON.error).toNotBeOk(); }); it("must have a message", function () { expect(responseJSON.message).toBeOk(); }); }); });
Thus began my transit-time project to create a tiny (really tiny) test framework that I could include in my Postman Collections and write tests using it. I personally started using it and thought sharing it might help us BDD aficionados. The framework ended up being a gist that is about 0.7kb when minified. I named the script describe-it (such creativity!)
The “describe-it” test framework script
The following piece of minified code is the complete test framework that provides behavior driven testing functionalities (albeit basic) similar to mocha and jasmine. The script does look cryptic — but that is to ensure that it is as lightweight as possible.
(typeof tests!=='object')&&(tests={});var it=((it=function(k,v){it.d.push(k);it.t[it.d]=1;it.b.forEach(it.c);try{v()}catch(e){it.t[it.d]=0;setTimeout&&setTimeout(function(){throw e;})}it.a.forEach(it.c);it.d.pop()}), it.a=[],it.b=[],it.c=function(x){x()},it.d=[],it.d.toString=function(){return this.join(' ')}, it.t=tests,it.x=function(v){this.v=v},it.xp=it.x.prototype,it.xp.toBe=function(x){(this.v!==x)&&it._()}, it.xp.toNotBe=function(x){(this.v===x)&&it._()},it.xp.toEql=function(x){(this.v!=x)&&it._()}, it.xp.toNotEql=function(x){(this.v==x)&&it._()},it.xp.toBeOk=function(){!this.v&&it._()}, it.xp.toNotBeOk=function(){this.v&&it._()},it),describe=function(k,v){it.d.push(k);v();it.d.pop()}, expect=function(v){return new it.x(v)},beforeEach=it.b.push.bind(it.b),afterEach=it.a.push.bind(it.a);
The not-so-cryptic version of the script is in the gist: https://gist.github.com/shamasis/7b12f451374314500da3 (Although, my colleagues rightfully contradict that it’s still cryptic.)
The script exposes the following functions:
describe (testgroup:string, tests:function);
it (testname:string, test:function);
beforeEach (setup:function);
afterEach (teardown:function);
expect (value); // assertion function
These functions work similar to mocha/jasmine and as such referring to their documentation is an obvious shortcut I would take — https://jasmine.github.io/2.0/introduction.html
The “expect” assertion function supports the following checks:
expect(something).toBeOk(); // allows truthy variables
expect(something).toNotBeOk(); // allows falsey variables
expect(something).toBe(somethingElse); // checks equality
expect(something).toNotBe(somethingElse); // checks inequality
expect(something).toEql(somethingElse); // checks similarity
expect(something).toNotEql(somethingElse); // checks dissimilarity
We might need more of these and especially ones that are related to API testing. (Would love to hear more in comments.)
Using the script within Postman
There are two parts to using this within Postman:
- Save the script as a global variable
- In the test scripts, eval the global variable before using the framework
Saving the script as a global variable
Saving the script as a global variable allows me to re-use the test framework script in multiple requests without having to manually add them. One can simply go and add the contents of the script to a global variable (or an environment variable) with name, say, describe-it-test-script
. The documentation article Our Manage Environments documentation outlines how to set and use these variables.
Personally, I save the test script as part of the first request to ensure that the test script is bundled with Postman Collections while running them in Collection Runner or Newman.
The following collection fetches the test framework from the gist (so that I can keep it updated with latest content) and saves it in global variables. The subsequent requests use this global variable to get the test framework within each request’s test script.
Sample Collection that uses this test framework:
www.getpostman.com/collections/34163da95c4372e36c39Import it in your postman and refer to the tests written in the collection called “describe-it” them.
Using the script in tests
Once you have the script stored in a variable, it is easy to use it in any test by running eval
on the script. The first line of your test would look somewhat like the following:
eval(postman.getGlobalVariable("describe-it-test-script"));
I know — eval is evil! But that is a separate discussion altogether. If you want to avoid “eval” altogether, simply paste the minified script at the beginning of your tests. But that would make the tests look a bit ugly.
Post this, you can start using the describe
, it
, expect
and other functions of the test framework normally.
Working with the test framework
Say, you have already saved the framework in a global variable, and you would like to run a test that ensures that a request responds with a 200 status code and serves HTML content type. The tests for the same would look like:
eval(postman.getGlobalVariable("describe-it-test-script")); describe ("response", function () { it ("must have html content type header", function () { expect(responseHeaders['Content-Type']) .toBe('text/html; charset=utf-8'); }); it ("must return a 200 status code", function () { expect(responseCode.code).toEql(200); }); });
We can write as many tests as we want and nest as many describe
blocks as we need. If there are runtime errors in one it
block, the other tests would still continue to execute sequentially.
The collection shared in this post (named describe-it,) has examples of tests written using this micro framework. It has tests written with failing and passing scenarios. And upon running the collection in Postman Collection Runner, it outputs the test results grouped within their respective describe
blocks.
How it works
The it
function executes the test parameter function within a try-catch
block. If the function executes without an error, it adds to the test description parameter (along with the description of it’s parent describe
blocks) to the Postman tests
variable and sets it to a truthy value. In the event that an expect
assertion fails or some other error occurs, the tests
object key is set to a falsey value.
In the above example, if all tests pass, the tests
object looks like
tests: { "response must have html content type header": 1, "response must return a 200 status code": 1 }
Hello Shamasis, thanks for that!
This is a great addition to testing in Postman. Would be great to see it comes with standard Postman. Another idea is to have it as an extension or plugin.
I would like to know more about why this way of testing is so useful to you. It would help us craft this in right direction.
Hello @shamasis:disqus , it’s mainly because the test and the results are easy to read, this turns out also to be easy to design.
Using Tests how can I compare JSON response to previously saved JSON response from same request?
You could store the JSON in environment variable and when the request is re run, compare the results. @tmack8001’s script is cool.
Itest I have done this before. I have even uses “Tests” in postman to compare between a legacy system and the new system that we are replacing said legacy system with (only works if systems are written to in parity).
The simplest way to answer your question is via the following:
// store the responseBody for future comparisons
var oldResponse = postman.getEnvironmentVariable(“unique-to-this-test”);
if (oldResponse === null) {
tests[‘no previous value to compare’] = true;
postman.setEnvironmentVariable(“unique-to-this-test”, responseBody)
} else {
// compare old response with current response
// depending on what you are doing something like the following will be good enough, but this requests the responseBodies to be exact.
tests[‘response matches previous’] = (oldResponse === responseBody)
}
If you want to do say a deepEquals() where the attributes in the responseBody are the same, but are in a different order then you will need to customize the above a bit to your specific needs. And obviously I prefer to use the behaviour testing that is mentioned in this blog post, so feel free to update the script I gave to be except(responseBody).toEql(oldResponse) instead.
This way allows you to test any type of response and not just JSON, if you need to parse the json and compare it you should then do something different.
I saved the script as global variable but when I run tests, I get ‘describe is not defined’.
Did you run the ‘eval’ code in the first line of your tests? This line ensures that the functions are available in the script.
eval(postman.getGlobalVariable(“describe-it-test-script”));
why don’t let us include external scripts… instead this hacky global var + eval nonsense?
This really need to be integrated into Postman. Atomic test is a great protection but it does not help to test complete scenarii. Kudo on this awesome premise.
I’d like to see it as a Post-success trigger. If all my atomic tests succeed then launch the scenarii.
After one year, did this ever made it into the core product?
If not, are there any plans on incorporating it?
Thanks
Is there a way to get this method to work within newman?
When I try to load an exported environment file with the “describe-it-test-script”, newman won’t load it and make the test suite functions available to my test collection.
Hi there,
Any news about some ability to import and use multiple scripts in test collections.
Tried and used the setGlobalVariable / eval trick. It works like a charm (I extensively use postman-bdd that way), but is quite limitative when having a bunch of scripts.
One of the tighter limit with postman_collection is about source control.
A new easy mechanism to have external libraries made available would be great for that. So would be great to not update collection and tests guids on each collection import (same as above, makes version control quite tedious).
Anyway, thanks for your great job.
It doesn’t seems to work any more. When i run the test collection, I have an error saying :”There was an error in evaluating the test script: Error: it._ is not a function”
Have you any clue to solve the problem ?
Hi Lionel, Please contact our support team at http://www.postman.com/support and they’ll be able to help you.