Skip to content

Commit

Permalink
QE-12356 Allow multiple screenshots (#417)
Browse files Browse the repository at this point in the history
Previously, each step could be associated with only one screenshot.
Now, a step can take as many screenshots as it feels like.
QE-12356
  • Loading branch information
ddl-kgarton authored Jan 4, 2024
1 parent 9b6cfca commit f8d6820
Show file tree
Hide file tree
Showing 11 changed files with 159 additions and 86 deletions.
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project closely adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).

## 0.183.0
- Change - Added ability to take multiple screenshots per step

## 0.182.0
Chore - Update docker README for ARM64 based CPU
Chore - Add seleniarm_hub bash file
Expand Down
12 changes: 6 additions & 6 deletions features/browser/images.feature
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,10 @@ Feature: Images
And I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/index.html"
When I click the link "Multiple scenarios with browser steps"
And I click the link "Open our test buttons page"
Then I should not see the image with the alt text "Given I start a webserver at directory "data/www" and save the port to the variable "PORT""
And I should not see the image with the alt text "And I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should not see the image with the alt text "Then I should see the button "button with child""
Then I should not see the image with the alt text "After I start a webserver at directory "data/www" and save the port to the variable "PORT""
And I should not see the image with the alt text "After I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should not see the image with the alt text "After I should see the button "button with child""
When I click the button "I should see the button "button with child""
Then I should not see the image with the alt text "Given I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should not see the image with the alt text "And I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should see the image with the alt text "Then I should see the button "button with child""
Then I should not see the image with the alt text "After I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should not see the image with the alt text "After I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/buttons.html""
And I should see the image with the alt text "After I should see the button "button with child""
2 changes: 1 addition & 1 deletion features/browser/mht.feature
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
Feature: Download MHT archives
As a user of web-based testing
I want to download MHT archives
So that I can bebug DOM-based failures more easily
So that I can debug DOM-based failures more easily

Scenario: Download an MHT file during a test
Given I skip this scenario if the current browser is not "chrome"
Expand Down
44 changes: 32 additions & 12 deletions features/cli/report_basics.feature
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ Feature: Report basics
When I run the command "ls "{SCENARIO_RESULTS_DIR}/"" and save stdout to "STDOUT", stderr to "STDERR" and expect exit code "0"
Then I should see "{STDOUT}" matches the following
"""
[\s\S]*\.png
(\d\d\d\d - [\s\S]*)
[\s\S]*
"""

Expand All @@ -50,7 +50,7 @@ Feature: Report basics

* # Can see image for step with secret in name
When I click the button "I should see the text "\{MY_SECRET\}""
Then I should see the image with the alt text "Then I should see the text "\{MY_SECRET\}""
Then I should see the image with the alt text "After I should see the text "\{MY_SECRET\}""

* # Cannot see secrets in the exception message
When I click the button "Then I click the button "\{MY_SECRET\}""
Expand All @@ -66,19 +66,19 @@ Feature: Report basics
# Verify HTML report
When I click the link "Multiple scenarios with browser steps"
And I click the link "Open our test checkboxes page"
And I should not see the image with the alt text "Given I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "And I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html\""
And I should not see the image with the alt text "Then I should see the checkbox \"checkbox with inner label\""
And I should not see the image with the alt text "After I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "After I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html\""
And I should not see the image with the alt text "After I should see the checkbox \"checkbox with inner label\""
Then I click the button "Then I should see the checkbox \"checkbox with inner label\""
And I should see the image with the alt text "Then I should see the checkbox \"checkbox with inner label\""
And I should not see the image with the alt text "Given I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "And I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html""
And I should see the image with the alt text "After I should see the checkbox \"checkbox with inner label\""
And I should not see the image with the alt text "After I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "After I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html""
When I save the current url to the variable "CURRENT_URL"
And I click the link "Index"
Then I navigate to the url "{CURRENT_URL}"
And I wait to see the image with the alt text "Then I should see the checkbox \"checkbox with inner label\""
And I should not see the image with the alt text "Given I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "And I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html\""
And I wait to see the image with the alt text "After I should see the checkbox \"checkbox with inner label\""
And I should not see the image with the alt text "After I start a webserver at directory \"data/www\" and save the port to the variable \"PORT\""
And I should not see the image with the alt text "After I open a browser at the url \"http://{HOST_ADDRESS}:{PORT}/checkboxes.html\""

Scenario: User can run a feature with mixed results and has all results reported correctly without skips
Given I run the command "cucu run data/features/feature_with_mixed_results.feature --results {CUCU_RESULTS_DIR}/mixed-results" and expect exit code "1"
Expand Down Expand Up @@ -279,6 +279,26 @@ Feature: Report basics
And I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/index.html"
Then I should see the text "No data available in table"

Scenario: User can run and generate reports that perform highlighting
Given I create a file at "{CUCU_RESULTS_DIR}/highlights/environment.py" with the following:
"""
from cucu.environment import *
"""
And I create a file at "{CUCU_RESULTS_DIR}/highlights/steps/__init__.py" with the following:
"""
from cucu.steps import *
"""
And I start a webserver at directory "data/www" and save the port to the variable "PORT"
And I create a file at "{CUCU_RESULTS_DIR}/highlights/text.feature" with the following:
"""
Feature: run a test with highlighting
Scenario: See text that should be highlit
When I open a browser at the url "http://{HOST_ADDRESS}:{PORT}/text.html"
Then I should see the text "just some text in a label"
"""
And I run the command "cucu run {CUCU_RESULTS_DIR}/highlights --results {CUCU_RESULTS_DIR}/empty_features_results --env CUCU_INJECT_ELEMENT_BORDER='True' --generate-report --report {CUCU_RESULTS_DIR}/highlights_report" and save stdout to "STDOUT", stderr to "STDERR" and expect exit code "0"

@report-only-failures
Scenario: User can generate a report with only failures
Given I run the command "cucu run data/features --tags @passing,@failing --report-only-failures --results {CUCU_RESULTS_DIR}/report_only_failures --generate-report --report {CUCU_RESULTS_DIR}/report_only_failures_report" and expect exit code "1"
Expand All @@ -295,4 +315,4 @@ Feature: Report basics
| .* | Just a scenario that opens a web page | 3 | failed | .* |
When I click the button "Just a scenario that opens a web page"
And I wait to click the button "show images"
And I should see the image with the alt text "And I should see the text \"inexistent\""
And I should see the image with the alt text "After I should see the text \"inexistent\""
4 changes: 2 additions & 2 deletions features/cli/secrets.feature
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,7 @@ Feature: Secrets
Then I pass the secret "\{MY_SECRET\}" to a substep
"""
When I run the command "cucu run {CUCU_RESULTS_DIR}/substeps_without_variable_passthru --secrets MY_SECRET --results={CUCU_RESULTS_DIR}/substeps_without_variable_passthru_results" and save stdout to "STDOUT", stderr to "STDERR" and expect exit code "0"
Then I should see a file at "{CUCU_RESULTS_DIR}/substeps_without_variable_passthru_results/Feature that spills the beans/This scenario prints some secrets to the logs/3 - I echo "************".png"
Then I should see a file at "{CUCU_RESULTS_DIR}/substeps_without_variable_passthru_results/Feature that spills the beans/This scenario prints some secrets to the logs/0003 - I echo "************"/0000 - After I echo "************".png"

Scenario: User gets expected behavior when using variable_passthru=True
Given I create a file at "{CUCU_RESULTS_DIR}/substeps_with_variable_passthru/environment.py" with the following:
Expand Down Expand Up @@ -178,4 +178,4 @@ Feature: Secrets
Then I pass the secret "\{MY_SECRET\}" to a substep
"""
When I run the command "cucu run {CUCU_RESULTS_DIR}/substeps_with_variable_passthru --secrets MY_SECRET --results={CUCU_RESULTS_DIR}/substeps_with_variable_passthru_results" and save stdout to "STDOUT", stderr to "STDERR" and expect exit code "0"
Then I should see a file at "{CUCU_RESULTS_DIR}/substeps_with_variable_passthru_results/Feature that spills the beans/This scenario prints some secrets to the logs/3 - I echo "\{MY_SECRET\}".png"
Then I should see a file at "{CUCU_RESULTS_DIR}/substeps_with_variable_passthru_results/Feature that spills the beans/This scenario prints some secrets to the logs/0003 - I echo "\{MY_SECRET\}"/0000 - After I echo "\{MY_SECRET\}".png"
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tool.poetry]
name = "cucu"
version = "0.182.0"
version = "0.183.0"
license = "MIT"
description = "Easy BDD web testing"
authors = ["Domino Data Lab <open-source@dominodatalab.com>"]
Expand Down
59 changes: 5 additions & 54 deletions src/cucu/environment.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,14 @@
import hashlib
import json
import os
import shutil
import sys
import time
from functools import partial

from cucu import config, init_scenario_hook_variables, logger
from cucu.config import CONFIG
from cucu.page_checks import init_page_checks
from cucu.utils import take_screenshot

CONFIG.define(
"FEATURE_RESULTS_DIR",
Expand All @@ -31,30 +31,6 @@
init_page_checks()


def generate_image_filename(step_index, step_name):
"""
generate .png image file name that meets these criteria:
- hides secrets
- escaped
- filename does not exceed 255 chars (OS limitation)
- uniqueness comes from step number
"""
max_filename = 255 - len(".png")
escaped_step_name = CONFIG.hide_secrets(step_name).replace("/", "_")
filename = f"{step_index} - {escaped_step_name}"

if len(filename) > max_filename:
ellipsis = "..."
# save the last chars as the ending often important
end_count = 100
front_count = max_filename - (len(ellipsis) + end_count)
filename = (
filename[:front_count] + ellipsis + filename[-1 * end_count :]
)

return f"{filename}.png"


def check_browser_initialized(ctx):
"""
check browser session initialized otherwise throw an exception indicating
Expand Down Expand Up @@ -218,6 +194,8 @@ def before_step(ctx, step):
ctx.current_step.has_substeps = False
ctx.start_time = time.monotonic()

CONFIG["__STEP_SCREENSHOT_COUNT"] = 0

# run before all step hooks
for hook in CONFIG["__CUCU_BEFORE_STEP_HOOKS"]:
hook(ctx)
Expand All @@ -240,40 +218,13 @@ def after_step(ctx, step):
# and this step has no substeps as in the reporting the substeps that
# may actually do something on the browser take their own screenshots
if ctx.browser is not None and ctx.current_step.has_substeps is False:
filepath = os.path.join(
ctx.scenario_dir, generate_image_filename(ctx.step_index, step.name)
)

# If we've marked an element as the one we're interacting with,
# inject a border to highlight that element
if (
not CONFIG["CUCU_INJECT_ELEMENT_BORDER"]
or not CONFIG["__PERTINENT_ELEMENT"]
):
ctx.browser.screenshot(filepath)
else:
border_style = "solid magenta 4px"
border_radius = "4px"
highlighter = (
f'arguments[0].style["border"] = "{border_style}";'
f'arguments[0].style["border-radius"] = "{border_radius}";'
)
ctx.browser.execute(highlighter, CONFIG["__PERTINENT_ELEMENT"])
ctx.browser.screenshot(filepath)
clear_highlight = (
'arguments[0].style["border"] = "";'
'arguments[0].style["border-radius"] = "";'
)
ctx.browser.execute(clear_highlight, CONFIG["__PERTINENT_ELEMENT"])
CONFIG["__PERTINENT_ELEMENT"] = None

if CONFIG["CUCU_MONITOR_PNG"] is not None:
shutil.copyfile(filepath, CONFIG["CUCU_MONITOR_PNG"])
take_screenshot(ctx, step.name, label=f"After {step.name}")

# if the step has substeps from using `run_steps` then we already moved
# the step index in the run_steps method and shouldn't do it here
if not step.has_substeps:
ctx.step_index += 1
CONFIG["__STEP_SCREENSHOT_COUNT"] = 0

if CONFIG.bool("CUCU_IPDB_ON_FAILURE") and step.status == "failed":
ctx._runner.stop_capture()
Expand Down
8 changes: 7 additions & 1 deletion src/cucu/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

from cucu import logger, retry, run_steps
from cucu.config import CONFIG
from cucu.utils import take_screenshot


class step(object):
Expand Down Expand Up @@ -120,7 +121,12 @@ def base_should_see_the(ctx, thing, name, index=0):
if element is None:
raise RuntimeError(f'unable to find the {prefix}{thing} "{name}"')
logger.debug(f'Success: saw {prefix}{thing} "{name}"')
CONFIG["__PERTINENT_ELEMENT"] = element
take_screenshot(
ctx,
ctx.current_step.name,
label=f"saw {thing} {name}",
highlight_element=element,
)

@step(f'I should immediately see the {thing} "{{name}}"')
def should_immediately_see_the(ctx, thing, name):
Expand Down
36 changes: 29 additions & 7 deletions src/cucu/reporter/html.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
from cucu import format_gherkin_table, logger
from cucu.ansi_parser import parse_log_to_html
from cucu.config import CONFIG
from cucu.environment import generate_image_filename
from cucu.utils import get_step_image_dir


def escape(data):
Expand Down Expand Up @@ -184,16 +184,38 @@ def generate(results, basepath, only_failures=False):
if show_status:
print("s", end="", flush=True)
total_steps += 1
image_filename = generate_image_filename(
step_index, step["name"]
)
image_filepath = os.path.join(scenario_filepath, image_filename)
image_dir = get_step_image_dir(step_index, step["name"])
image_dirpath = os.path.join(scenario_filepath, image_dir)

if step["name"].startswith("#"):
step["heading_level"] = "h4"

if os.path.exists(image_filepath):
step["image"] = urllib.parse.quote(image_filename)
if os.path.exists(image_dirpath):
_, _, image_names = next(os.walk(image_dirpath))
images = []
for image_name in image_names:
words = image_name.split("-", 1)
index = words[0].strip()
try:
# Images with label should have a name in the form:
# 0000 - This is the image label.png
label, _ = os.path.splitext(words[1].strip())
except IndexError:
# Images with no label should instead look like:
# 0000.png
# so we default to the step name in this case.
label = step["name"]

images.append(
{
"src": urllib.parse.quote(
os.path.join(image_dir, image_name)
),
"index": index,
"label": label,
}
)
step["images"] = sorted(images, key=lambda x: x["index"])

if "result" in step:
if step["result"]["status"] in ["failed", "passed"]:
Expand Down
6 changes: 4 additions & 2 deletions src/cucu/reporter/templates/scenario.html
Original file line number Diff line number Diff line change
Expand Up @@ -134,11 +134,13 @@
<pre style="color: darkgray; margin: 0;">{{ escape("\n".join(step['result']['stdout'])) }}</pre>
{% endif %}

{% if step['image'] is defined %}
{% if step['images'] is defined %}
{% if step['result']['stdout'] %}
<br/>
{% endif %}
<img class="mx-auto d-block img-fluid shadow bg-white rounded" alt='{{ step_name }}' title='{{ step_name }}' src='{{ step["image"] }}'></img>
{% for image in step['images'] %}
<img class="mx-auto d-block img-fluid shadow bg-white rounded" style="margin-bottom:15px" alt='{{ image["label"] }}' title='{{ image["label"] }}' src='{{ image["src"] }}'></img>
{% endfor %}
{% endif %}

{% if step['result']['error_message'] is defined %}
Expand Down
Loading

0 comments on commit f8d6820

Please sign in to comment.