The Issue
Ensuring that your API calls are working is key to having a functional application. As we covered with our post on best practices for API testing, automation is essential to ensure proper testing coverage. Lest take a look at one way we can automate a basic API request using Ruby – and then validate that we are seeing what we want to see.
For our example, we’ll use The Movie Database (TMDB). We’ll see how we can run a POST function by creating a list within our user settings, and then asserting that we receive the correct response code.
The Code
First, lets take a look at the front end. The TMDB user page has a list section:
Step 1: Setting Up your Ruby Environment.
We want to run a POST command that will create a list. This is a simple enough endeavor – we’ll just need to make sure we are passing through our correct JSON headers and body. First, we will call our rest-client gem by using the below code:
1 |
require 'rest-client' |
This will be followed by including our values like URL, body, etc. using RestClient.get/post/delete. See formatting below:
1 2 3 4 5 |
require 'rest-client' RestClient.get(url, headers={}) RestClient.post(url, payload, headers={}) |
Step 2: Adding JSON Parameters & Building the Ruby Script
In our case, we want to include our URL, request body, and headers. We will grab our URL for creating lists and the body format, and insert it (taken from our API documentation for TMDB). We will then include the headers, which span both the content_type and our authorization. For your authorization, follow the steps here to generate a bearer token, which will be included in this header. Once we have all of these, we will place them per the below code:
1 |
RestClient.post "https://api.themoviedb.org/4/list", '{"name": "NEW_NAME12323","iso_639_1": "en"}', {content_type: "application/json;charset=utf-8", authorization: "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE1MTk4NDYyNzYsInN1YiI6IjVhOGNhNDBmOTI1MTQxMGIwMTA4ZjUzNiIsImp0aSI6IjY5OTkxOCIsImF1ZCI6IjEyMDgyMWY2ZDIxOTE4Y2Y1NzA1MGNmOWU4Y2Q0ZDBhIiwic2NvcGVzIjpbImFwaV9yZWFkIiwiYXBpX3dyaXRlIl0sInZlcnNpb24iOjF9.4B-StWrStNmjIUI7f4BBKej4dGJCIW6oK3lR-W-5o8A"} |
Now that we have our JSON request ready, we can incorporate it into our entire Ruby file, and also include commands to display specific output. In this case, we will choose to display the output body, the headers and the response code:
1 2 3 4 5 6 7 8 9 |
#TMDBTest.rb require 'rest-client' response = RestClient.post "https://api.themoviedb.org/4/list", '{"name": "NEW_NAME12323","iso_639_1": "en"}', {content_type: "application/json;charset=utf-8", authorization: "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE1MTk4NDYyNzYsInN1YiI6IjVhOGNhNDBmOTI1MTQxMGIwMTA4ZjUzNiIsImp0aSI6IjY5OTkxOCIsImF1ZCI6IjEyMDgyMWY2ZDIxOTE4Y2Y1NzA1MGNmOWU4Y2Q0ZDBhIiwic2NvcGVzIjpbImFwaV9yZWFkIiwiYXBpX3dyaXRlIl0sInZlcnNpb24iOjF9.4B-StWrStNmjIUI7f4BBKej4dGJCIW6oK3lR-W-5o8A"} puts response.code puts response.headers puts response.body |
We will put these all into a Ruby file, and save it as TMDBTest.rb.
Step 3: Executing Your Script, HTTP Request
Then, we can actually run the file, using ruby TMDBTest.rb. For our response, we will be returned the following:
So, we now have our foundation – a functioning API POST request that we can run from Ruby. However, we will need to add a few things before we can actually utilize this in a test case. First, we want to make sure the test is reusable – if we dont change our list name, we will not be able to create another list due to the name duplication. So, we can solve this by ensuring we are generating a new random name each time. In order to do this, we will build out a function that will generate a random string of alpha characters. See the below code
1 2 3 4 5 6 7 8 |
def rand_string_alpha(length, save_as = nil) chars = 'abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNOPQRSTUVWXYZ' result = Array.new(length) { chars[rand(chars.length)].chr }.join current_test.test_params[save_as] = result if save_as result end result = rand_string_alpha(10) |
This gives us a function to work with: result. When we call result now, it will generate a string of 10 random alpha characters. So, now we need to figure how to incorporate result into our JSON request. There are two ways we can do this. The first will directly incorporate it as a string into our JSON request. We will first define request as a variable. In it, we include the area where we are calling result between our single quotes and plus signs – this converts it into a string that will be utilized by our JSON request. See below
1 |
request = '{"name": "'+result+'","iso_639_1": "en"}' |
So, when we run this, we will actually be doing the equivalent of the below code, with RANDOMNAME being any random selection of 10 Alpha characters
1 |
request = '{"name": "RANDOMNAME","iso_639_1": "en"}' |
As for inputting this into our JSON request, we can include the our defined request into the body, like so:
1 |
response = RestClient.post "https://api.themoviedb.org/4/list", <strong>request</strong>, {content_type: "application/json;charset=utf-8", authorization: "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE1MTk4NDYyNzYsInN1YiI6IjVhOGNhNDBmOTI1MTQxMGIwMTA4ZjUzNiIsImp0aSI6IjY5OTkxOCIsImF1ZCI6IjEyMDgyMWY2ZDIxOTE4Y2Y1NzA1MGNmOWU4Y2Q0ZDBhIiwic2NvcGVzIjpbImFwaV9yZWFkIiwiYXBpX3dyaXRlIl0sInZlcnNpb24iOjF9.4B-StWrStNmjIUI7f4BBKej4dGJCIW6oK3lR-W-5o8A"} |
Now, if we combine all of these and put them back into our main file, we will see something like this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
require 'rest-client' def rand_string_alpha(length, save_as = nil) chars = 'abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNOPQRSTUVWXYZ' result = Array.new(length) { chars[rand(chars.length)].chr }.join current_test.test_params[save_as] = result if save_as result end result = rand_string_alpha(10) request = '{"name": "'+result+'","iso_639_1": "en"}' response = RestClient.post "https://api.themoviedb.org/4/list", request, {content_type: "application/json;charset=utf-8", authorization: "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE1MTk4NDYyNzYsInN1YiI6IjVhOGNhNDBmOTI1MTQxMGIwMTA4ZjUzNiIsImp0aSI6IjY5OTkxOCIsImF1ZCI6IjEyMDgyMWY2ZDIxOTE4Y2Y1NzA1MGNmOWU4Y2Q0ZDBhIiwic2NvcGVzIjpbImFwaV9yZWFkIiwiYXBpX3dyaXRlIl0sInZlcnNpb24iOjF9.4B-StWrStNmjIUI7f4BBKej4dGJCIW6oK3lR-W-5o8A"} puts response.code puts response.headers puts response.body |
We will put our result variable in a string (see above code) which will allow you insert in the JSON request. Thus, we can utilize our function that we have constructed to generate a random string of characters which will act as our list name in this case.
Altanetively, we can use the .gsub command to replace our given placeholder with whatever we choose. See below
1 2 3 |
request = '{"name": "NAME","iso_639_1": "en"}' request = request.gsub("NAME",result) |
We defined request a bit differently, and include “NAME” where we want our name to be generated. We then include an additional line with our .gsub command, which will replace any given instance of NAME with our desired random string.
Step 4: Assertion
Now, let’s delve into two areas: the assertion, and actual incorporation into a frontend UI test. To validate that we receive the correct response code, we will add some Ruby logic into our file. First we will initialize results to an empty hash – see below
1 |
results = {} |
This empty hash will allow us to assign a key/value pair of our choosing. The key is going be our validation, and the value will be the result. We will use the following logic:
1 |
results["Response Code should be 201"] = response.code == 201 |
In this example, [“Response Code should be 201”] = acts as our validation, and the response.code == 201 acts as our value.
In display our results (and raise an exception on failure), we will add a puts command to print the contents of our variable:
1 |
puts results |
We can then add a step to fail the test if any of the validations in our results variable have failed:
1 |
fail if results.values.include?(false) |
So, when we run the test we will now see a command line output like so:
If the test failed, we would see a message like so:
Lets take a look at what the entire script will now look like:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
require 'rest-client' require 'watir-webdriver' def rand_string_alpha(length, save_as = nil) chars = 'abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNOPQRSTUVWXYZ' result = Array.new(length) { chars[rand(chars.length)].chr }.join current_test.test_params[save_as] = result if save_as result + end result = rand_string_alpha(10) results = {} request = '{"name": "NAME","iso_639_1": "en"}' request = request.gsub("NAME",result) response = RestClient.post "https://api.themoviedb.org/4/list", request, {content_type: "application/json;charset=utf-8", authorization: "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE1MTk4NDYyNzYsInN1YiI6IjVhOGNhNDBmOTI1MTQxMGIwMTA4ZjUzNiIsImp0aSI6IjY5OTkxOCIsImF1ZCI6IjEyMDgyMWY2ZDIxOTE4Y2Y1NzA1MGNmOWU4Y2Q0ZDBhIiwic2NvcGVzIjpbImFwaV9yZWFkIiwiYXBpX3dyaXRlIl0sInZlcnNpb24iOjF9.4B-StWrStNmjIUI7f4BBKej4dGJCIW6oK3lR-W-5o8A"} puts response.code results["Response code should be 201"] = response.code == 201 puts response.headers puts response.body puts results fail("Test Failed.") if results.values.include?(false) |
Takeaway
We now have a functioning Ruby script that will send an API request, display our responses and assert we are seeing the correct response code. You can continue to assert different parts of your responses – such as a specific value in the response body, or a a certain response in the headers. A benefit of testing your API calls through a Ruby script in this way is the versatility of incorporating front end testing (through Watir Webdriver commands) in the same script (which we will be breaking down in a future post).