Skip to content

Commit

Permalink
Merge pull request #92 from reportportal/develop
Browse files Browse the repository at this point in the history
Release
  • Loading branch information
HardNorth authored Nov 20, 2024
2 parents ea4140f + 83eb37c commit 1ee5495
Show file tree
Hide file tree
Showing 5 changed files with 153 additions and 46 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/release.yml
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ jobs:
java-version: '8'

- name: Setup git credentials
uses: oleksiyrudenko/gha-git-credentials@v2.1.1
uses: oleksiyrudenko/gha-git-credentials@v2-latest
with:
name: 'reportportal.io'
email: '[email protected]'
Expand Down
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,11 @@
# Changelog

## [Unreleased]
### Added
- Common Stack Trace frames skip in description and logs, by @HardNorth
- Reporting of Last Error Log in Item description, by @HardNorth and @ArtemOAS
### Changed
- Client version updated on [5.2.21](https://github.com/reportportal/client-java/releases/tag/5.2.21), by @HardNorth

## [5.2.2]
### Changed
Expand Down
6 changes: 3 additions & 3 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ repositories {
}

dependencies {
api 'com.epam.reportportal:client-java:5.2.13'
api 'com.epam.reportportal:client-java:5.2.21'
api "io.cucumber:cucumber-java:${project.cucumber_version}"

implementation 'org.slf4j:slf4j-api:2.0.7'
Expand All @@ -49,7 +49,7 @@ dependencies {
testImplementation 'org.hamcrest:hamcrest-core:2.2'
testImplementation 'org.mockito:mockito-core:3.3.3'
testImplementation 'org.mockito:mockito-junit-jupiter:3.3.3'
testImplementation 'ch.qos.logback:logback-classic:1.3.12'
testImplementation 'ch.qos.logback:logback-classic:1.3.14'
testImplementation "io.cucumber:cucumber-testng:${project.cucumber_version}"
testImplementation 'com.epam.reportportal:logger-java-logback:5.2.2'
testImplementation ("org.junit.platform:junit-platform-runner:${project.junit_runner_version}") {
Expand All @@ -58,7 +58,7 @@ dependencies {
testImplementation "org.junit.jupiter:junit-jupiter-api:${project.junit_version}"
testImplementation "org.junit.jupiter:junit-jupiter-params:${project.junit_version}"
testImplementation "org.junit.jupiter:junit-jupiter-engine:${project.junit_version}"
testImplementation 'org.apache.commons:commons-io:1.3.2'
testImplementation 'commons-io:commons-io:2.16.1'
}

test {
Expand Down
121 changes: 89 additions & 32 deletions src/main/java/com/epam/reportportal/cucumber/AbstractReporter.java
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,8 @@
import com.epam.reportportal.service.tree.TestItemTree;
import com.epam.reportportal.utils.*;
import com.epam.reportportal.utils.files.ByteSource;
import com.epam.reportportal.utils.formatting.MarkdownUtils;
import com.epam.reportportal.utils.http.ContentType;
import com.epam.reportportal.utils.markdown.MarkdownUtils;
import com.epam.reportportal.utils.properties.SystemAttributesExtractor;
import com.epam.ta.reportportal.ws.model.FinishExecutionRQ;
import com.epam.ta.reportportal.ws.model.FinishTestItemRQ;
Expand All @@ -47,6 +47,7 @@
import gherkin.pickles.PickleTable;
import gherkin.pickles.PickleTag;
import io.reactivex.Maybe;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.Pair;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand All @@ -66,10 +67,11 @@
import static com.epam.reportportal.cucumber.Utils.*;
import static com.epam.reportportal.cucumber.util.ItemTreeUtils.createKey;
import static com.epam.reportportal.cucumber.util.ItemTreeUtils.retrieveLeaf;
import static com.epam.reportportal.utils.formatting.ExceptionUtils.getStackTrace;
import static java.lang.String.format;
import static java.util.Optional.of;
import static java.util.Optional.ofNullable;
import static org.apache.commons.lang3.StringUtils.isNotBlank;
import static org.apache.commons.lang3.exception.ExceptionUtils.getStackTrace;

/**
* Abstract Cucumber 4.x formatter for Report Portal
Expand All @@ -89,6 +91,7 @@ public abstract class AbstractReporter implements ConcurrentEventListener {
private static final String FILE_PREFIX = "file:";
private static final String HOOK_ = "Hook: ";
private static final String DOCSTRING_DECORATOR = "\n\"\"\"\n";
private static final String ERROR_FORMAT = "Error:\n%s";

public static final TestItemTree ITEM_TREE = new TestItemTree();
private static volatile ReportPortal REPORT_PORTAL = ReportPortal.builder().build();
Expand All @@ -108,6 +111,15 @@ public abstract class AbstractReporter implements ConcurrentEventListener {
// End of feature occurs once launch is finished.
private final Map<String, Date> featureEndTime = new ConcurrentHashMap<>();

/**
* This map uses to record the description of the scenario and the step to append the error to the description.
*/
private final Map<Maybe<String>, String> descriptionsMap = new ConcurrentHashMap<>();
/**
* This map uses to record errors to append to the description.
*/
private final Map<Maybe<String>, Throwable> errorMap = new ConcurrentHashMap<>();

public static ReportPortal getReportPortal() {
return REPORT_PORTAL;
}
Expand Down Expand Up @@ -162,7 +174,8 @@ protected TestCaseIdEntry getTestCaseId(@Nonnull TestStep testStep, @Nullable St
if (definitionMatch != null) {
try {
Method method = retrieveMethod(definitionMatch);
return TestCaseIdUtils.getTestCaseId(method.getAnnotation(TestCaseId.class),
return TestCaseIdUtils.getTestCaseId(
method.getAnnotation(TestCaseId.class),
method,
codeRef,
(List<Object>) ARGUMENTS_TRANSFORM.apply(arguments)
Expand Down Expand Up @@ -235,14 +248,20 @@ private void addToTree(RunningContext.FeatureContext featureContext, RunningCont
* @param scenarioContext current scenario context
*/
protected void beforeScenario(RunningContext.FeatureContext featureContext, RunningContext.ScenarioContext scenarioContext) {
String scenarioName = Utils.buildName(scenarioContext.getKeyword(),
String scenarioName = Utils.buildName(
scenarioContext.getKeyword(),
AbstractReporter.COLON_INFIX,
scenarioContext.getTestCase().getName()
);
Maybe<String> id = startScenario(featureContext.getFeatureId(),
buildStartScenarioRequest(scenarioContext.getTestCase(), scenarioName, featureContext.getUri(), scenarioContext.getLine())
StartTestItemRQ startTestItemRQ = buildStartScenarioRequest(
scenarioContext.getTestCase(),
scenarioName,
featureContext.getUri(),
scenarioContext.getLine()
);
Maybe<String> id = startScenario(featureContext.getFeatureId(), startTestItemRQ);
scenarioContext.setId(id);
descriptionsMap.put(id, ofNullable(startTestItemRQ.getDescription()).orElse(StringUtils.EMPTY));
if (launch.get().getParameters().isCallbackReportingEnabled()) {
addToTree(featureContext, scenarioContext);
}
Expand All @@ -260,9 +279,7 @@ protected ItemStatus mapItemStatus(@Nullable Result.Type status) {
return null;
} else {
if (STATUS_MAPPING.get(status) == null) {
LOGGER.error(String.format("Unable to find direct mapping between Cucumber and ReportPortal for TestItem with status: '%s'.",
status
));
LOGGER.error("Unable to find direct mapping between Cucumber and ReportPortal for TestItem with status: '{}'.", status);
return ItemStatus.SKIPPED;
}
return STATUS_MAPPING.get(status);
Expand All @@ -282,11 +299,34 @@ protected ItemStatus mapItemStatus(@Nullable Result.Type status) {
protected FinishTestItemRQ buildFinishTestItemRequest(@Nonnull Maybe<String> itemId, @Nullable Date finishTime,
@Nullable ItemStatus status) {
FinishTestItemRQ rq = new FinishTestItemRQ();
if (status == ItemStatus.FAILED) {
Optional<String> currentDescription = Optional.ofNullable(descriptionsMap.remove(itemId));
Optional<Throwable> currentError = Optional.ofNullable(errorMap.remove(itemId));
currentDescription.flatMap(description -> currentError.map(errorMessage -> resolveDescriptionErrorMessage(
description,
errorMessage
))).ifPresent(rq::setDescription);
}
ofNullable(status).ifPresent(s -> rq.setStatus(s.name()));
rq.setEndTime(ofNullable(finishTime).orElse(Calendar.getInstance().getTime()));
rq.setEndTime(finishTime);
return rq;
}

/**
* Resolve description
*
* @param currentDescription Current description
* @param error Error message
* @return Description with error
*/
private String resolveDescriptionErrorMessage(String currentDescription, Throwable error) {
String errorStr = format(ERROR_FORMAT, getStackTrace(error, new Throwable()));
return Optional.ofNullable(currentDescription)
.filter(StringUtils::isNotBlank)
.map(description -> MarkdownUtils.asTwoParts(currentDescription, errorStr))
.orElse(errorStr);
}

/**
* Finish a test item with specified status
*
Expand All @@ -300,10 +340,11 @@ protected Date finishTestItem(Maybe<String> itemId, Result.Type status) {
return null;
}

FinishTestItemRQ rq = buildFinishTestItemRequest(itemId, null, mapItemStatus(status));
Date endTime = Calendar.getInstance().getTime();
FinishTestItemRQ rq = buildFinishTestItemRequest(itemId, endTime, mapItemStatus(status));
//noinspection ReactiveStreamsUnusedPublisher
launch.get().finishTestItem(itemId, rq);
return rq.getEndTime();
return endTime;
}

/**
Expand All @@ -330,6 +371,9 @@ protected void afterScenario(TestCaseFinished event) {
RunningContext.ScenarioContext context = getCurrentScenarioContext();
String featureUri = context.getFeatureUri();
currentScenarioContextMap.remove(Pair.of(context.getLine(), featureUri));
if (mapItemStatus(event.result.getStatus()) == ItemStatus.FAILED) {
Optional.ofNullable(event.result.getError()).ifPresent(error -> errorMap.put(context.getId(), error));
}
Date endTime = finishTestItem(context.getId(), event.result.getStatus());
featureEndTime.put(featureUri, endTime);
currentScenarioContext.remove();
Expand Down Expand Up @@ -436,7 +480,9 @@ protected void beforeStep(TestStep testStep) {
context.setCurrentStepId(stepId);
String stepText = step.getText();
context.setCurrentText(stepText);

if (rq.isHasStats()) {
descriptionsMap.put(stepId, ofNullable(rq.getDescription()).orElse(StringUtils.EMPTY));
}
if (launch.get().getParameters().isCallbackReportingEnabled()) {
addToTree(context, stepText, stepId);
}
Expand All @@ -450,6 +496,9 @@ protected void beforeStep(TestStep testStep) {
protected void afterStep(Result result) {
reportResult(result, null);
RunningContext.ScenarioContext context = getCurrentScenarioContext();
if (mapItemStatus(result.getStatus()) == ItemStatus.FAILED) {
Optional.ofNullable(result.getError()).ifPresent(error -> errorMap.put(context.getCurrentStepId(), error));
}
finishTestItem(context.getCurrentStepId(), result.getStatus());
context.setCurrentStepId(null);
}
Expand All @@ -472,8 +521,8 @@ protected StartTestItemRQ buildStartHookRequest(HookType hookType) {
/**
* Start before/after-hook item on Report Portal
*
* @param parentId parent item id
* @param rq hook start request
* @param parentId parent item id
* @param rq hook start request
* @return hook item id
*/
@Nonnull
Expand Down Expand Up @@ -552,7 +601,7 @@ protected void reportResult(Result result, String message) {
if (errorMessage != null) {
sendLog(errorMessage, level);
} else if (result.getError() != null) {
sendLog(getStackTrace(result.getError()), level);
sendLog(getStackTrace(result.getError(), new Throwable()), level);
}
}

Expand All @@ -577,7 +626,8 @@ protected void embedding(@Nullable String name, String mimeType, byte[] data) {
String type = ofNullable(mimeType).filter(ContentType::isValidType).orElseGet(() -> getDataType(data, name));
String attachmentName = ofNullable(name).filter(m -> !m.isEmpty())
.orElseGet(() -> ofNullable(type).map(t -> t.substring(0, t.indexOf("/"))).orElse(""));
ReportPortal.emitLog(new ReportPortalMessage(ByteSource.wrap(data), type, attachmentName),
ReportPortal.emitLog(
new ReportPortalMessage(ByteSource.wrap(data), type, attachmentName),
"UNKNOWN",
Calendar.getInstance().getTime()
);
Expand Down Expand Up @@ -620,7 +670,8 @@ protected void finishFeature(Maybe<String> itemId, Date dateTime) {
LOGGER.error("BUG: Trying to finish unspecified test item.");
return;
}
FinishTestItemRQ rq = buildFinishTestItemRequest(itemId, dateTime, null);
Date endTime = ofNullable(dateTime).orElse(Calendar.getInstance().getTime());
FinishTestItemRQ rq = buildFinishTestItemRequest(itemId, endTime, null);
//noinspection ReactiveStreamsUnusedPublisher
launch.get().finishTestItem(itemId, rq);
}
Expand Down Expand Up @@ -746,14 +797,16 @@ protected void handleStartOfTestCase(TestCaseStarted event) {
TestCase testCase = event.testCase;
RunningContext.FeatureContext newFeatureContext = new RunningContext.FeatureContext(testCase);
String featureUri = newFeatureContext.getUri();
RunningContext.FeatureContext featureContext = currentFeatureContextMap.computeIfAbsent(featureUri, u -> {
getRootItemId(); // trigger root item creation
newFeatureContext.setFeatureId(startFeature(buildStartFeatureRequest(newFeatureContext.getFeature(), featureUri)));
if (launch.get().getParameters().isCallbackReportingEnabled()) {
addToTree(newFeatureContext);
}
return newFeatureContext;
});
RunningContext.FeatureContext featureContext = currentFeatureContextMap.computeIfAbsent(
featureUri, u -> {
getRootItemId(); // trigger root item creation
newFeatureContext.setFeatureId(startFeature(buildStartFeatureRequest(newFeatureContext.getFeature(), featureUri)));
if (launch.get().getParameters().isCallbackReportingEnabled()) {
addToTree(newFeatureContext);
}
return newFeatureContext;
}
);

if (!featureContext.getUri().equals(testCase.getUri())) {
throw new IllegalStateException("Scenario URI does not match Feature URI.");
Expand All @@ -762,10 +815,12 @@ protected void handleStartOfTestCase(TestCaseStarted event) {
RunningContext.ScenarioContext newScenarioContext = featureContext.getScenarioContext(testCase);

Pair<Integer, String> scenarioLineFeatureURI = Pair.of(newScenarioContext.getLine(), featureContext.getUri());
RunningContext.ScenarioContext scenarioContext = currentScenarioContextMap.computeIfAbsent(scenarioLineFeatureURI, k -> {
currentScenarioContext.set(newScenarioContext);
return newScenarioContext;
});
RunningContext.ScenarioContext scenarioContext = currentScenarioContextMap.computeIfAbsent(
scenarioLineFeatureURI, k -> {
currentScenarioContext.set(newScenarioContext);
return newScenarioContext;
}
);

beforeScenario(featureContext, scenarioContext);
}
Expand Down Expand Up @@ -793,14 +848,16 @@ protected void handleTestStepFinished(TestStepFinished event) {
}

protected void addToTree(RunningContext.ScenarioContext scenarioContext, String text, Maybe<String> stepId) {
retrieveLeaf(scenarioContext.getFeatureUri(),
retrieveLeaf(
scenarioContext.getFeatureUri(),
scenarioContext.getLine(),
ITEM_TREE
).ifPresent(scenarioLeaf -> scenarioLeaf.getChildItems().put(createKey(text), TestItemTree.createTestItemLeaf(stepId)));
}

protected void removeFromTree(RunningContext.ScenarioContext scenarioContext, String text) {
retrieveLeaf(scenarioContext.getFeatureUri(),
retrieveLeaf(
scenarioContext.getFeatureUri(),
scenarioContext.getLine(),
ITEM_TREE
).ifPresent(scenarioLeaf -> scenarioLeaf.getChildItems().remove(createKey(text)));
Expand Down
Loading

0 comments on commit 1ee5495

Please sign in to comment.