com.seleniumtests.reporter.logger.TestStep Maven / Gradle / Ivy
/**
* Orignal work: Copyright 2015 www.seleniumtests.com
* Modified work: Copyright 2016 www.infotel.com
* Copyright 2017-2019 B.Hecquet
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.seleniumtests.reporter.logger;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.List;
import java.util.stream.Collectors;
import com.seleniumtests.driver.screenshots.SnapshotCheckType;
import org.apache.commons.io.FileUtils;
import org.apache.commons.lang3.StringUtils;
import org.json.JSONArray;
import org.json.JSONObject;
import org.testng.ITestResult;
import com.seleniumtests.core.Step.RootCause;
import com.seleniumtests.core.TestStepManager;
import com.seleniumtests.util.ExceptionUtility;
/**
* Group of test actions
* Represent each method call inside a PageObject during the test
* TestStep allows to log any action done during the test. See {@link com.seleniumtests.core.aspects.LogAction#logTestStep }
* We get a tree like this:
* root (TestStep)
* +--- action1 (TestAction)
* +--+ sub-step1 (TestStep)
* +--- sub-action1
* +--- message (TestMessage)
* +--- sub-action2
* +--- action2
*
* Each TestStep is then logged
*
* @author behe
*
*/
public class TestStep extends TestAction {
private static final String BEFORE_STEP_PREFIX = "before-";
public static final String COMPOSITE_STEP_PREFIX = "Composite ";
private List stepActions;
private Long duration;
private Date startDate;
private long videoTimeStamp;
private List harCaptures;
private List files;
private List snapshots;
private ITestResult testResult;
private RootCause errorCause;
private String errorCauseDetails;
private Integer stepResultId; // the stepResult if it has been recorded on seleniumRobot-server
private boolean disableBugtracker;
public enum StepStatus {
SUCCESS,
FAILED,
WARNING
}
/**
* Step for test
* @param name
*/
public TestStep(String name) {
this(name, null, new ArrayList<>(), true);
}
/**
*
* @param name action name
* @param testResult associated TestNG result
* @param pwdToReplace list of string to replace when returning actions so that passwords are masked
*/
public TestStep(String name, ITestResult testResult, List pwdToReplace, boolean maskPassword) {
this(name, testResult, pwdToReplace, maskPassword, RootCause.NONE, null, false);
}
public TestStep(String name, ITestResult testResult, List pwdToReplace, boolean maskPassword, RootCause errorCause, String errorCauseDetails, boolean disableBugtracker) {
super(name, false, pwdToReplace);
this.maskPassword = maskPassword;
stepActions = new ArrayList<>();
files = new ArrayList<>();
harCaptures = new ArrayList<>();
snapshots = new ArrayList<>();
duration = 0L;
startDate = new Date();
this.testResult = testResult;
if (errorCause != RootCause.NONE) {
this.errorCause = errorCause;
this.errorCauseDetails = errorCauseDetails;
}
this.disableBugtracker = disableBugtracker;
}
public Long getDuration() {
long consolidatedDuration = duration;
consolidatedDuration -= getDurationToExclude();
return Math.max(0, consolidatedDuration);
}
public boolean isTestEndStep() {
return TestStepManager.LAST_STEP_NAME.equals(name);
}
public void setStartDate() {
startDate = new Date();
}
public void setStartDate(Date date) {
startDate = date;
}
@Override
public long getDurationToExclude() {
long consolidatedDurationToExclude = durationToExclude;
for (Snapshot snapshot: snapshots) {
consolidatedDurationToExclude += snapshot.getDurationToExclude();
}
for (TestAction action: stepActions) {
consolidatedDurationToExclude += action.getDurationToExclude();
}
return consolidatedDurationToExclude;
}
/**
* set duration in milliseconds
* @param duration
*/
public void setDuration(Long duration) {
this.duration = duration;
}
public void updateDuration() {
duration = new Date().getTime() - startDate.getTime();
}
/**
* Add a password to the list of passwords that should be masked / replaced
* This password will also be added to the list of password for sub steps
* Useful for Selenium IDE tests, where we specify a password in the middle of a step
* Only substeps, action, messages created AFTER this call will have their new password masked
* @param password the password to add
*/
public void addPasswordToReplace(String password) {
if (password != null && password.length() > TestStepManager.MIN_PASSWORD_LENGTH) {
pwdToReplace.add(password);
}
}
public List getStepActions() {
return stepActions;
}
/**
* Return
* StepStatus.FAILED if and only if the step is marked as failed
* StepStatus.WARNING if any of the sub actions (or steps) are failed, but not the step itself
* StepStatus.SUCCESS in any other cases
*/
public StepStatus getStepStatus() {
if (Boolean.TRUE.equals(getFailed())) {
return StepStatus.FAILED;
}
for (TestAction action: stepActions) {
if ((action instanceof TestStep && ((TestStep) action).getStepStatus() != StepStatus.SUCCESS)
|| Boolean.TRUE.equals(action.getFailed())) {
return StepStatus.WARNING;
}
}
return StepStatus.SUCCESS;
}
public String getExceptionMessage(String format) {
if (actionException == null) {
return "";
}
StringBuilder stackString = new StringBuilder();
ExceptionUtility.generateTheStackTrace(actionException, actionException.getMessage(), stackString, format);
return stackString.toString();
}
public void addAction(TestAction action) {
action.setPosition(stepActions.size());
stepActions.add(action);
action.setParent(this);
// add replacement of the parent to this action
action.pwdToReplace.addAll(pwdToReplace);
// inherit password masking from step
action.maskPassword = maskPassword;
}
public void addMessage(TestMessage message) {
message.setPosition(stepActions.size());
stepActions.add(message);
message.setParent(this);
// add replacement of the parent to this message
message.pwdToReplace.addAll(pwdToReplace);
// inherit password masking from step
message.maskPassword = maskPassword;
}
public void addValue(TestValue value) {
value.setPosition(stepActions.size());
stepActions.add(value);
value.setParent(this);
}
public void addStep(TestStep step) {
step.setPosition(stepActions.size());
stepActions.add(step);
step.setParent(this);
// add replacement of the parent step to this step
step.pwdToReplace.addAll(pwdToReplace);
// inherit password masking from step
step.maskPassword = maskPassword;
}
public void addNetworkCapture(HarCapture har) {
har.setPosition(stepActions.size());
harCaptures.add(har);
har.setParent(this);
}
public void addFile(GenericFile file) {
file.setPosition(stepActions.size());
files.add(file);
file.setParent(this);
}
/**
* Add snapshot to this step
* @param snapshot the snapshot
* @param stepIdx the index of this step in the test
* @param userGivenName name of the snapshot, user wants to display
*/
public void addSnapshot(Snapshot snapshot, int stepIdx, String userGivenName) {
snapshot.setPosition(snapshots.size());
// rename file so that user can easily consult it
snapshot.rename(this, position, snapshot.position + getSnapshotIndex(), userGivenName);
snapshots.add(snapshot);
snapshot.setParent(this);
}
private int getSnapshotIndex() {
return getSnapshotIndex(1) - position;
}
private int getSnapshotIndex(int multiplier) {
int idx = (position + 1) * multiplier;
if (getParent() instanceof TestStep) {
idx += ((TestStep)getParent()).getSnapshotIndex(multiplier * 10);
}
return idx;
}
@Override
public String toString() {
return toString(0);
}
@Override
public String toString(int spaces) {
String currentIndent = StringUtils.repeat(" ", spaces);
StringBuilder testStepRepr = new StringBuilder(String.format("Step %s\n", getName()));
for (TestAction testAction: getStepActions()) {
testStepRepr.append(currentIndent + " - " + testAction.toString(spaces + 2) + "\n");
}
return testStepRepr.toString().trim();
}
@Override
public JSONObject toJson() {
JSONObject stepJSon = super.toJson();
stepJSon.put("name", getName());
stepJSon.put("duration", getDuration());
stepJSon.put("date", startDate);
stepJSon.put("videoTimeStamp", videoTimeStamp);
stepJSon.put("type", "step");
stepJSon.put("status", getStepStatus().toString());
stepJSon.put("harCaptures", new JSONArray());
for (HarCapture harCapture: getHarCaptures()) {
stepJSon.getJSONArray("harCaptures").put(harCapture.toJson());
}
stepJSon.put("actions", new JSONArray());
for (TestAction testAction: getStepActions()) {
stepJSon.getJSONArray("actions").put(testAction.toJson());
}
stepJSon.put("files", new JSONArray());
for (GenericFile file: getFiles()) {
stepJSon.getJSONArray("files").put(file.toJson());
}
stepJSon.put("snapshots", new JSONArray());
for (Snapshot snapshot: getSnapshots()) {
stepJSon.getJSONArray("snapshots").put(snapshot.toJson());
}
return stepJSon;
}
public Date getStartDate() {
return startDate;
}
public void removeSnapshot(Snapshot snapshotToRemove) {
snapshots.remove(snapshotToRemove);
}
public List getSnapshots() {
return getSnapshots(false);
}
public List getSnapshots(boolean fromSubSteps) {
List snaps = new ArrayList<>(snapshots);
if (fromSubSteps) {
for (TestAction subStep : stepActions.stream().filter(a -> a instanceof TestStep).collect(Collectors.toList())) {
snaps.addAll(((TestStep) subStep).getSnapshots(fromSubSteps));
}
}
return snaps;
}
public ITestResult getTestResult() {
return testResult;
}
/**
* Returns the list of files referenced by this test step and sub-steps
* @return
*/
public List getAllAttachments() {
return getAllAttachments(false);
}
/**
* Returns the list of files referenced by this test step and sub-steps
* @param onlyPictureOfSnapshots If true, the HTML file linked to snapshot won't be returned
* @param requestedSnapshotCheckTypes If not empty, only returns Snapshot files whose SnapshotCheckType correspond to this value
* @return
*/
public List getAllAttachments(boolean onlyPictureOfSnapshots, SnapshotCheckType ... requestedSnapshotCheckTypes) {
List usedFiles = new ArrayList<>();
for (Snapshot snapshot: snapshots) {
if (snapshot == null || snapshot.getScreenshot() == null ||
(requestedSnapshotCheckTypes.length != 0 && !Arrays.asList(requestedSnapshotCheckTypes).contains(snapshot.getCheckSnapshot()))) {
continue;
}
if (snapshot.getScreenshot().getHtml() != null && !onlyPictureOfSnapshots) {
usedFiles.add(snapshot.getScreenshot().getHtml());
}
if (snapshot.getScreenshot().getImage() != null) {
usedFiles.add(snapshot.getScreenshot().getImage());
}
}
usedFiles.addAll(getFiles().stream().map(GenericFile::getFileContent).collect(Collectors.toList()));
usedFiles.addAll(getHarCaptures().stream().map(HarCapture::getFileContent).collect(Collectors.toList()));
// add attachments declared in sub-steps
for (TestAction subStep: stepActions.stream().filter(a -> a instanceof TestStep).collect(Collectors.toList())) {
usedFiles.addAll(((TestStep)subStep).getAllAttachments(onlyPictureOfSnapshots, requestedSnapshotCheckTypes));
usedFiles.addAll(((TestStep)subStep).getFiles().stream().map(GenericFile::getFileContent).collect(Collectors.toList()));
}
return usedFiles;
}
/**
* move attachments from "before-" folder to outputDirectory folder
* @param outputDirectory
* @throws IOException
*/
public void moveAttachments(String outputDirectory) throws IOException {
// create the output directory if it does not exist
new File(outputDirectory).mkdirs();
for (Snapshot snapshot: snapshots) {
if (snapshot != null
&& snapshot.getScreenshot() != null
&& ((snapshot.getScreenshot().getHtml() != null && snapshot.getScreenshot().getHtml().getFile().getAbsolutePath().contains(BEFORE_STEP_PREFIX))
|| (snapshot.getScreenshot().getImage() != null && snapshot.getScreenshot().getImage().getFile().getAbsolutePath().contains(BEFORE_STEP_PREFIX)))) {
try {
snapshot.relocate(outputDirectory);
} catch (IOException e) {
logger.error("Cannot relocate snapshot: " + e.getMessage());
}
}
}
// move attachments in sub-steps
for (TestAction subStep: stepActions.stream().filter(a -> a instanceof TestStep).collect(Collectors.toList())) {
((TestStep)subStep).moveAttachments(outputDirectory);
}
for (HarCapture harCapture: harCaptures) {
if (harCapture.getFile().toString().contains(BEFORE_STEP_PREFIX)) {
harCapture.getFileContent().relocate(outputDirectory);
}
}
for (GenericFile genericFile: files) {
if (genericFile.getFile().toString().contains(BEFORE_STEP_PREFIX)) {
genericFile.getFileContent().relocate(outputDirectory);
}
}
}
public List getHarCaptures() {
return harCaptures;
}
public void setHarCaptures(List harCaptures) {
this.harCaptures = harCaptures;
}
/**
* @deprecated use 'getRootCause' instead
* @return
*/
@Deprecated
public RootCause getErrorCause() {
return getRootCause();
}
public RootCause getRootCause() {
if (getStepStatus() == StepStatus.SUCCESS) {
return null;
}
if (errorCause != null) {
return errorCause;
}
for (TestAction action: stepActions) {
if ((action instanceof TestStep && ((TestStep) action).getRootCause() != null)) {
return ((TestStep) action).getRootCause();
}
}
return errorCause;
}
/**
* @deprecated use getRootCauseDetails instead
* @return
*/
@Deprecated
public String getErrorCauseDetails() {
return getRootCauseDetails();
}
/**
* Returns the error cause detail of the step or any sub step
* @return
*/
public String getRootCauseDetails() {
if (getStepStatus() == StepStatus.SUCCESS) {
return null;
}
if (errorCause != null) {
return errorCauseDetails;
}
for (TestAction action: stepActions) {
if ((action instanceof TestStep && ((TestStep) action).getRootCause() != null)) {
return ((TestStep) action).getRootCauseDetails();
}
}
return errorCauseDetails;
}
public boolean isDisableBugtracker() {
return disableBugtracker;
}
public List getFiles() {
return files;
}
/**
* Creates an new TestStep, encoding text of sub-steps and elements.
*/
@Override
public TestStep encode(String format) {
List encodedPasswords = encodePasswords(pwdToReplace, format);
TestStep step = new TestStep(encodeString(name, format), testResult, encodedPasswords, maskPassword);
step.stepActions = new ArrayList<>();
for (TestAction testAction: stepActions) {
step.stepActions.add(testAction.encode(format));
}
step.failed = failed;
step.snapshots = new ArrayList<>(snapshots);
step.files = new ArrayList<>();
for (GenericFile file: files) {
step.files.add(file.encode(format));
}
if (format == null) {
step.encoded = encoded;
} else {
step.encoded = true;
}
step.duration = duration;
step.startDate = startDate;
step.videoTimeStamp = videoTimeStamp;
step.harCaptures = new ArrayList<>();
for (HarCapture har: harCaptures) {
step.harCaptures.add(har);
}
step.actionException = actionException;
if (actionException != null) {
step.actionExceptionMessage = encodeString(ExceptionUtility.getExceptionMessage(actionException), format);
}
step.errorCause = errorCause;
step.errorCauseDetails = encodeString(errorCauseDetails, format);
step.disableBugtracker = disableBugtracker;
step.timestamp = timestamp;
return step;
}
@Override
public TestStep deepCopy() {
return encode(null);
}
public long getVideoTimeStamp() {
return videoTimeStamp;
}
public void setVideoTimeStamp(long videoTimeStamp) {
this.videoTimeStamp = videoTimeStamp;
}
public Integer getStepResultId() {
return stepResultId;
}
public void setStepResultId(Integer stepResultId) {
this.stepResultId = stepResultId;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy