- Inspired by jest-html-reporter
- Using project dominate to generate the page
- https://fedoramagazine.org/automation-through-accessibility/
- Full technical solution of our team's (Red Hat DesktopQE) automation stack https://modehnal.github.io/
python3 -m pip install behave-html-pretty-formatter
To use it with Behave create a behave.ini
file in your project folder
(or in home) with the following content:
# -- FILE: behave.ini
# Define ALIAS for PrettyHTMLFormatter.
[behave.formatters]
html-pretty = behave_html_pretty_formatter:PrettyHTMLFormatter
# Optional configuration of PrettyHTMLFormatter
# also possible to use "behave ... -D behave.formatter.html-pretty.{setting}={value}".
[behave.userdata]
behave.formatter.html-pretty.title_string = Test Suite Reporter
# Example usecase, print {before/after}_scenarios as steps with attached data.
behave.formatter.html-pretty.pseudo_steps = false
# Structure of the result html page readable(pretty) or condensed.
behave.formatter.html-pretty.pretty_output = true
# The '%' must be escaped in ini format.
behave.formatter.html-pretty.date_format = %%d-%%m-%%Y %%H:%%M:%%S
# Defines if the summary is expanded upon start.
behave.formatter.html-pretty.show_summary = false
# Defines if the user is interested in what steps are not executed.
behave.formatter.html-pretty.show_unexecuted_steps = true
# Define what to collapse by default, possible values:
# "auto" - show everything except embeds (default)
# "all" - hide everything
# comma separated list - specify subset of "scenario,embed,table,text"
# "none" - show everything, even embeds
behave.formatter.html-pretty.collapse = auto
# Defines if the user wants to see previous attempts when using auto retry.
# Auto retry https://github.com/behave/behave/blob/main/behave/contrib/scenario_autoretry.py
behave.formatter.html-pretty.show_retry_attempts = true
# Override global summary visibility
# "auto" - show global summary if more than one feature executed (default)
# "true" - show global summary
# "false" - hide global summary
behave.formatter.html-pretty.global_summary = auto
# Following will be formatted in summary section as "tester: worker1".
behave.additional-info.tester = super_worker
# Can be used multiple times.
behave.additional-info.location = super_awesome_lab
Alternatively, with behave >= v1.2.7.dev3, you can put the same configuration in
pyproject.toml
, like so:
[tool.behave.formatters]
"html-pretty" = "behave_html_pretty_formatter:PrettyHTMLFormatter"
# Optional configuration of PrettyHTMLFormatter
[tool.behave.userdata]
"behave.formatter.html-pretty.title_string" = "Test Suite Reporter"
"behave.formatter.html-pretty.pseudo_steps" = false
"behave.formatter.html-pretty.pretty_output" = true
"behave.formatter.html-pretty.date_format" = "%%d-%%m-%%Y %%H:%%M:%%S"
"behave.formatter.html-pretty.show_summary" = false
"behave.formatter.html-pretty.collapse" = "auto"
"behave.formatter.html-pretty.show_unexecuted_steps" = true
"behave.formatter.html-pretty.global_summary" = "auto"
"behave.additional-info.tester" = "super_worker"
"behave.additional-info.location" = "super_awesome_lab"
Then use the formatter by running Behave with the -f
/--format
option, e.g.
behave -f help
behave -f html-pretty
behave -f html-pretty -o behave-report.html
You can find information about behave and user-defined formatters in the behave docs.
- Default Static Example:
- High contrast Static Example:
- Colours adjusted.
- Extra information is added before every decorator about the status of the step.
- Text is bigger.
You can switch between the different contrasts with the toggle button.
Stylesheet follows the browser dark theme, so it reverts background to dark and adjusts colors to darker shade.
Summary is hidden by default
- Pretty HTML Formatter With Summary Expanded and Collapsed Steps:
- Pretty HTML Formatter High contrast With Summary Expanded and Collapsed Steps:
To change the setting use the .ini file.
behave.formatter.html-pretty.show_summary = true
Take a look at the Common MIME types, few of them are useful for us.
data_base64 = base64.b64encode(open("/path/to/image", "rb").read())
data_encoded = data_base64.decode("utf-8").replace("\n", "")
# Format
"data:<mime_type>;<encoding>,<data_encoded>"
# Example
f"data:image/png;base64,{data_encoded}"
for formatter in context._runner.formatters:
if formatter.name == "html-pretty":
context.formatter = formatter
icon_data = f"data:image/svg+xml;base64,{data_encoded}"
context.formatter.set_icon(icon=icon_data)
Example use case - an indicator if the test is running under X11 or Wayland:
context.formatter.set_title(title="Test Suite Reporter")
- This is configurable also from the behave.ini file
behave.formatter.html-pretty.title_string = Test Suite Reporter
context.formatter.add_html_head_element('<script src="https://example.js.org/my_js_lib.min.js"></script>')
Example use case - import custom JS library (e.g. plotly.js).
Used as an information panel to describe or provide information about the page contents. You will need to define your own step where you will set flag for commentary step.
@step('Commentary')
def commentary_step(context):
# Get the correct step to override.
scenario = context.formatter.current_scenario
step = scenario.current_step
# Override the step, this will prevent the decorator to be generated and only the text will show.
step.commentary_override = True
Feature file example usage:
@commentary
Scenario: Example of commentary usage.
* Start application "zenity" via command "zenity --about"
* Commentary
"""
This field is generated from decorator 'Commentary'
Where you insert text and override step to not print its decorator.
The text will get printed and will be seen, as you can see, haha.
"""
* Left click "About" "radio button"
Result can be seen in the image examples.
This is to create shortcut context.embed()
. There are multiple ways to achieve this.
If you have Basic setup, you can simply define it like this:
context.embed = context.formatter.embed
If you don't have Basic setup, the proper way is the following code
for formatter in context._runner.formatters:
if formatter.name == "html-pretty":
context.embed = formatter.embed
You can also define custom embed function which does something else when behave is called without html-pretty
formatter (i.e. for debugging purposes), and overwrite it with formatter.embed
function only if the formatter is present.
def _embed(mime_type, data, caption):
if "text" in mime_type:
# Do your logging here, for example:
print(f"{caption}: {data}")
context.embed = _embed
# This requires Basic setup
if hasattr(context, "formatter"):
context.embed = context.formatter.embed
context.embed(mime_type="image/png", data="/path/to/image.png", caption="Screenshot")
context.embed(mime_type="video/webm", data="/path/to/video.webm", caption="Video")
# This have to be done once per report, e.g. in `before_all()`.
context.formatter.add_html_head_element('<script src="https://cdn.plot.ly/plotly-2.35.2.min.js" charset="utf-8"></script>')
# Example graph
graph_js = \
"""
TESTER = document.getElementById('tester');
TESTER.innerHTML = '';
Plotly.newPlot( TESTER, [{
x: [1, 2, 3, 4, 5],
y: [1, 2, 4, 8, 16] }], {
margin: { t: 0 } } );
"""
# Embed the wrapping `<div>` element together with JS code.
context.formatter.embed(
data=f'<div id="tester" style="width:600px;height:250px;"></div><script>{graph_js}</script>',
mime_type="text/html",
caption="Example Graph",
compress=False,
)
# Embed the same JS code gzip compressed (large graphs).
graph_js_min = graph_js.replace("\n", "").replace("tester", "compressed_tester").strip()
context.formatter.embed(
data=f'<div id="compressed_tester" style="width:600px;height:250px;" onclick="{graph_js_min}">Click me to load the graph.</div>',
mime_type="text/html",
caption="Example Compressed Graph",
compress=True,
)
Note: with compress=True
the decompression is done by javascript, and decompressed script is not executed by browser (additional onclick
callback is required).
These are examples we use on daily basis, we can define more if required.
mime_type="video/webm", data="/path/to/video.webm" or data="<base64_encoded_video>"
mime_type="image/png", data="/path/to/image.png" or data="<base64_encoded_image>"
mime_type="text/plain", data="<string>"
mime_type="text/html", data="<string>" # data string is pasted as raw HTML (not escaped)
mime_type="text/markdown", data="<string>" # data string is converted using markdown pip module
mime_type="link", data="list(<link>, <label>)"
You can simply set data=data_encoded
generated as described in Encoding to base64 section and the formatter will generate the proper Format based on MIME type, or you can just use the data="/path/to/file"
and formatter will attempt to convert it.
Function embed()
returns object, which can be saved and modified later via set_data()
and set_fail_only()
methods. This is if you want to embed some data which are still being processes (output of a background process started in a step, etc.).
If the testsuite uses before_scenario()
and after_scenario()
and you would like to see them as steps in HTML report (for example to have embeds separated from the standard steps), configuration switch in behave.ini file behave.formatter.html-pretty.pseudo_steps = true
will do the trick, together with calling context.html_formatter.before_scenario_finish(status)
at the end of before_scenario()
(analogously for after_scenario()
). The status is one of "passed", "failed", "skipped"
. Function will set color class of the pseudo step and also record pseudo step duration.
# Example use in features.environment.py
def before_scenario(context, scenario):
...
# This requires to have html_formatter set by code above.
if error_found:
context.embed("text", str(error_found), "Error Message")
context.html_formatter.before_scenario_finish("failed")
raise error_found
else:
context.html_formatter.before_scenario_finish("passed")
def after_scenario(context, scenario):
...
if error_found:
context.embed("text", str(error_found), "Error Message")
context.html_formatter.after_scenario_finish("failed")
raise error_found
else:
context.html_formatter.after_scenario_finish("passed")
You want to help with improving this software? Please create an issue in our open bug tracker, or open a pull request directly.
We use tox for running linting and tests, e.g.
tox
tox -l
tox -e flake8
For code formatting we use black, which you can run using our Tox setup, e.g.
tox -e black
If you need to change CSS or JavaScript code: First edit the regular files, then generate the minified versions like so:
tox -e minify