7 minute read

Python and requests and header

Logos in header image sources: Python, Requests, JSON, HTTP, Cerberus ReportPortal

This is the eighth post in a series on how to build an API framework using python.

You can read previous parts below:


Ah, Test results reporting is a favorite topic of discussion among test automation engineers.

While everyone wants to create their own reporting dashboard/site because it sure is fun to mess around with JS, HTML, CSS, Store test results in a relational or document database and maybe create some APIs routes using a web framework like Flask, Django.

It’s also true that all this is a lot of work to set up in the correct way. I won’t discourage you if you want to build the next-gen reporting solution that the community can use.

However, Should we really invest our valuable time in reinventing the wheel? Probably not, (at least, if we can avoid it 😉)

The test results reporting space has some great tools and libraries already available that get the job done however, of late one of my absolute favorites is the ReportPortal because of the ton of features it packs out of the box

In this post, We will see how to integrate it into our framework that we’ve been building so far:

Setup ReportPortal using docker

I had previously written a blog on setting up a local ReportPortal instance on Docker and demonstrated a basic Java test on Gradle for reporting. You can read that post here: How to setup ReportPortal on a local docker instance

Essentially, in a nutshell, the setup is quite easy to do using docker as it pulls all the required dependencies and spins up an environment for your use very quickly

Below are the steps to follow to set up your containers

# Step 1: Download docker-compose YAML file
curl https://raw.githubusercontent.com/reportportal/reportportal/master/docker-compose.yml -o docker-compose.yml

# Step 2: Spin up containers
# This will download the images (one time) over the internet so ensure you are on a good connection
docker-compose -p reportportal up -d --force-recreate

Once the above command finishes, verify if all the containers are spun up by executing:

docker ps -a

Once done you should be able to open up the local site at http://localhost:8080/ui/ with default id and password in ReportPortals website (at the time of writing this blog the credentials are user: superadmin, password: erebus)

(🙏🏼 : Please change this in any production instance that you wish to use in your organizations unless you want those pesky hackers to have access to your projects test results)

Bug workaround: elastic search keeps on restarting

You might notice that once you log in, you get pop-ups like “An error occurred” and your session is terminated.

To identify the root cause and fix this issue, follow the below steps.

If this works perfectly for you, then feel free to skip these steps and move on to the next section that talks about setting up our framework

docker ps -a

We can observe that part of the problem is that the elasticsearch container keeps on restarting.

7b6052df5d30   docker.elastic.co/elasticsearch/elasticsearch:7.3.0   "/usr/local/bin/dock…"   11 minutes ago   Restarting (1) 53 seconds ago                                                         reportportal_elasticsearch_1

Yes, I know libraries have their own quirks 🤷🏼‍♂️, you can fix this quite easily by following the below steps:


# Stop all running containers
docker stop $(docker ps -aq)

# Create /data/elasticsearch directory
mkdir -p data/elasticsearch

# Give permissions
chmod 777 data/elasticsearch

# Change ownership of this directory
sudo chown 1000:1000 data/elasticsearch/

# Finally bring up the containers again
docker-compose -p reportportal up -d

You can even quit docker for mac and restart it once again.

This workaround is from a Github issue on the ReportPortal website. You can see more details here. I even started a thread on the ReportPortal slack channel that you can follow for further discussions.

Note: If the above steps still result in elastic search restarting, you can manually reset docker by going to troubleshoot and click on Clean / Purge data and Reset to factory defaults

Docker reset

Integrate with pytest

Let’s understand how we can integrate our test suite (run using pytest) to report results directly into ReportPortal

You can find more details about different types of test framework integrations supported by ReportPortal here

Also, see the complete documentation for the pytest plugin here

Step 1: Create a project in ReportPortal and update pytest.ini file

It is recommended to create a new project in ReportPortal where you want to report test results to:

Tap on user profile settings

Tap on create a new project and enter a name, we’ll name our project people-tests

Open Profile

You can access the project settings which would be updated in our framework project.

Copy the required fields into a pytest.ini file in the project root

The pytest.ini file allows you to specify some default values that are auto loaded by pytest. You can also pass these manually via the command line but it can become quite tedious quite fast to pass these on every execution.

At a minimum, ensure you have the below properties present in the pytest.ini file

rp_uuid = 28e75c0f-2b29-49a1-9e2e-c45c9650dff0 (unique id of the project)
rp_endpoint = http://localhost:8080 (where ReportPortal is hosted)
rp_project = people-tests (name of the project)
rp_launch = people-tests (what the launch will be called)

Below is a complete sample file

[pytest]
rp_uuid = 28e75c0f-2b29-49a1-9e2e-c45c9650dff0
rp_endpoint = http://localhost:8080
rp_project = people-tests
rp_launch = people-tests
rp_launch_attributes = 'PyTest' 'Smoke'
rp_launch_description = 'Smoke test'
rp_ignore_errors = True
rp_ignore_attributes = 'xfail' 'usefixture'

Step 2: Add session-scoped logger

Pytest agent needs the plugin to be installed and registered.

Execute below:

pipenv install pytest-reportportal

We will add a logger method in the module scoped conftest.py file, which would be used in all the test methods to send different types of logs to ReportPortal’s elasticsearch database.

tests/conftest.py

import pytest
from pytest_reportportal import RPLogger, RPLogHandler


@pytest.fixture(scope="session")
def logger(request):
   logger = logging.getLogger(__name__)
   logger.setLevel(logging.DEBUG)

   # Create a handler for ReportPortal if the service has been
   # configured and started.
   if hasattr(request.node.config, 'py_test_service'):
       # Import ReportPortal logger and handler to the test module.
       logging.setLoggerClass(RPLogger)
       rp_handler = RPLogHandler(request.node.config.py_test_service)

       # Add additional handlers if it is necessary
       console_handler = logging.StreamHandler(sys.stdout)
       console_handler.setLevel(logging.INFO)
       logger.addHandler(console_handler)
   else:
       rp_handler = logging.StreamHandler(sys.stdout)

   # Set INFO level for ReportPortal handler.
   rp_handler.setLevel(logging.INFO)
   return logger

Here we make use of the standard python logger class and set RPLogger as the LoggerClass, along with even a console handler if needed. Notice that we’ve marked this fixture as session scope, This ensures the logger is reused and keeps on recording results to the appropriate project in the report portal throughout one test suites run

Step 3: Add logger to tests

We’ll add the newly created logger as a dependency into the test method and pytest would auto-resolve this for us.

def test_read_all_has_kent(logger):
   """
   Test on hitting People GET API, we get a user named kent in the list of people
   """
   response = client.read_all_persons()

   assert_that(response.status_code).is_equal_to(requests.codes.ok)
   logger.info("User successfully read")
   assert_people_have_person_with_first_name(response, first_name='Kent')

Notice, we’ve added a docstring to the test method

"""
Test on hitting People GET API, we get a user named kent in the list of people
"""

ReportPortal pytest plugin updates all method docstrings as an actual test case description

Also, once the status code check passes, we can log the message via the below line:

logger.info("User successfully read")

Step 4: Execute tests

Execute the below command to push logs to ReportPortal. Notice that you would need to add –reportportal flag

python -m pytest ./tests --reportportal

Don’t forget to ensure the people-api and covid-tracker services are up, Refer to Post 2 and Post 4, In case you need a refresher on how to start these on the local machine.

Step 5: Analysis in ReportPortal

Once run, let’s see how the results would look like in ReportPortal

Below you can see the overall results for the test run. Notice the rp_launch_attributes and rp_launch_description are displayed on the UI.

You can select Failed to see the test failure details

You can also see a detailed stack trace for any failures that might have occurred

Also, the docstring is displayed as the test method description

And finally, if you open the passing test test_read_all_has_kent, you can see that our log message is printed

Conclusion

While the ReportPortal might not be the perfect reporting solution out there, it does take care of a lot of common use cases when it comes to Test reporting and easily integrates with your test framework of choice.

I hope this post helps you understand how to set this up for your own python-based framework. Do check out GitHub readme for agent-python-pytest to understand some more features that you can make use of.

You can see this entire code on this Github repo and branch

If you found this post useful, Do share it with a friend or colleague and if you have thoughts, I’d be more than happy to chat over at twitter or comments. Until next time. Happy Testing.

Comments