Skip to content

Commit 3dd1d72

Browse files
authored
Add failFast option to allow displaying all errors in build (#1032)
* Refactor LogRecordsProcessors and MemoryLogHandler to reduce code * Add constructor to LogHandler to set defaults for booleans in a consistent way across Maven and tests * Sort files in Mojo to be consistent across different OS's closes #1024
1 parent 7465684 commit 3dd1d72

File tree

18 files changed

+930
-119
lines changed

18 files changed

+930
-119
lines changed

CHANGELOG.adoc

+3
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ For a detailed view of what has changed, refer to the {uri-repo}/commits/main[co
1313

1414
== Unreleased
1515

16+
Improvements (for all modules)::
17+
18+
* Add failFast option to logHandler to allow displaying all errors in build (#1032)
1619

1720
== v3.1.1 (2024-11-14)
1821

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
package org.asciidoctor.maven.log;
2+
3+
import java.io.File;
4+
5+
import org.asciidoctor.ast.Cursor;
6+
import org.asciidoctor.log.LogRecord;
7+
8+
/**
9+
* {@link LogRecord} proxy that allows capturing the source file being
10+
* processed.
11+
* Important: the {@link #sourceFile} and the actual source where an error is present
12+
* may not be the same. For example if the source is being included.
13+
*
14+
* @since 3.2.0
15+
*/
16+
final class CapturedLogRecord extends LogRecord {
17+
18+
private final File sourceFile;
19+
20+
CapturedLogRecord(LogRecord record, File sourceFile) {
21+
super(record.getSeverity(), record.getCursor(), record.getMessage(), record.getSourceFileName(), record.getSourceMethodName());
22+
this.sourceFile = sourceFile;
23+
}
24+
25+
public Cursor getCursor() {
26+
if (super.getCursor() != null) {
27+
return super.getCursor();
28+
}
29+
if (sourceFile != null) {
30+
return new FileCursor(sourceFile);
31+
}
32+
return null;
33+
}
34+
35+
public File getSourceFile() {
36+
return sourceFile;
37+
}
38+
39+
class FileCursor implements Cursor {
40+
41+
private final File file;
42+
43+
public FileCursor(File file) {
44+
this.file = file;
45+
}
46+
47+
@Override
48+
public int getLineNumber() {
49+
return 0;
50+
}
51+
52+
@Override
53+
public String getPath() {
54+
return file.getName();
55+
}
56+
57+
@Override
58+
public String getDir() {
59+
return file.getParent();
60+
}
61+
62+
@Override
63+
public String getFile() {
64+
return file.getAbsolutePath();
65+
}
66+
}
67+
}

asciidoctor-maven-commons/src/main/java/org/asciidoctor/maven/log/LogHandler.java

+14
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,18 @@
77
* POJO for Maven XML mapping.
88
*
99
* @author abelsromero
10+
* @since 1.5.7
1011
*/
1112
public class LogHandler {
1213

1314
private Boolean outputToConsole;
1415
private FailIf failIf;
16+
private Boolean failFast;
17+
18+
public LogHandler() {
19+
outputToConsole = Boolean.TRUE;
20+
failFast = Boolean.TRUE;
21+
}
1522

1623
public Boolean getOutputToConsole() {
1724
return outputToConsole;
@@ -37,4 +44,11 @@ public boolean isContainsTextNotBlank() {
3744
return failIf != null && isNotBlank(failIf.getContainsText());
3845
}
3946

47+
public Boolean getFailFast() {
48+
return failFast;
49+
}
50+
51+
public void setFailFast(Boolean failFast) {
52+
this.failFast = failFast;
53+
}
4054
}

asciidoctor-maven-commons/src/main/java/org/asciidoctor/maven/log/LogRecordsProcessors.java

+14-21
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
import java.io.File;
44
import java.util.List;
5+
import java.util.Optional;
56
import java.util.function.Consumer;
67

78
import org.asciidoctor.log.LogRecord;
@@ -23,35 +24,27 @@ public LogRecordsProcessors(LogHandler logHandler,
2324
}
2425

2526
public void processLogRecords(MemoryLogHandler memoryLogHandler) throws Exception {
26-
if (logHandler.isSeveritySet() && logHandler.isContainsTextNotBlank()) {
27-
final Severity severity = logHandler.getFailIf().getSeverity();
28-
final String textToSearch = logHandler.getFailIf().getContainsText();
27+
if (logHandler.isSeveritySet() || logHandler.isContainsTextNotBlank()) {
28+
final Severity severity = Optional.ofNullable(logHandler.getFailIf()).map(FailIf::getSeverity).orElse(null);
29+
final String textToSearch = Optional.ofNullable(logHandler.getFailIf()).map(FailIf::getContainsText).orElse(null);
2930

3031
final List<LogRecord> records = memoryLogHandler.filter(severity, textToSearch);
3132
if (records.size() > 0) {
3233
for (LogRecord record : records) {
3334
errorMessageConsumer.accept(LogRecordFormatter.format(record, sourceDirectory));
3435
}
35-
throw new Exception(String.format("Found %s issue(s) matching severity %s or higher and text '%s'", records.size(), severity, textToSearch));
36+
throw new Exception(getMessage(records, severity, textToSearch));
3637
}
38+
}
39+
}
40+
41+
private String getMessage(List<LogRecord> records, Severity severity, String textToSearch) {
42+
if (logHandler.isSeveritySet() && logHandler.isContainsTextNotBlank()) {
43+
return String.format("Found %s issue(s) matching severity %s or higher and text '%s'", records.size(), severity, textToSearch);
3744
} else if (logHandler.isSeveritySet()) {
38-
final Severity severity = logHandler.getFailIf().getSeverity();
39-
final List<LogRecord> records = memoryLogHandler.filter(severity);
40-
if (records.size() > 0) {
41-
for (LogRecord record : records) {
42-
errorMessageConsumer.accept(LogRecordFormatter.format(record, sourceDirectory));
43-
}
44-
throw new Exception(String.format("Found %s issue(s) of severity %s or higher during conversion", records.size(), severity));
45-
}
46-
} else if (logHandler.isContainsTextNotBlank()) {
47-
final String textToSearch = logHandler.getFailIf().getContainsText();
48-
final List<LogRecord> records = memoryLogHandler.filter(textToSearch);
49-
if (records.size() > 0) {
50-
for (LogRecord record : records) {
51-
errorMessageConsumer.accept(LogRecordFormatter.format(record, sourceDirectory));
52-
}
53-
throw new Exception(String.format("Found %s issue(s) containing '%s'", records.size(), textToSearch));
54-
}
45+
return String.format("Found %s issue(s) of severity %s or higher during conversion", records.size(), severity);
46+
} else {
47+
return String.format("Found %s issue(s) containing '%s'", records.size(), textToSearch);
5548
}
5649
}
5750

asciidoctor-maven-commons/src/main/java/org/asciidoctor/maven/log/MemoryLogHandler.java

+32-12
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
package org.asciidoctor.maven.log;
22

3+
import java.io.File;
34
import java.util.ArrayList;
45
import java.util.List;
56
import java.util.function.Consumer;
@@ -8,6 +9,7 @@
89
import org.asciidoctor.log.LogHandler;
910
import org.asciidoctor.log.LogRecord;
1011
import org.asciidoctor.log.Severity;
12+
import org.asciidoctor.maven.commons.StringUtils;
1113

1214

1315
/**
@@ -23,16 +25,26 @@ public class MemoryLogHandler implements LogHandler {
2325
private final Boolean outputToConsole;
2426
private final Consumer<LogRecord> recordConsumer;
2527

28+
/**
29+
* Provides simple way to inject the current file being processes.
30+
* Will need re-work in concurrent scenarios.
31+
*
32+
* @since 3.2.0
33+
*/
34+
private File currentFile;
35+
2636
public MemoryLogHandler(Boolean outputToConsole, Consumer<LogRecord> recordConsumer) {
2737
this.outputToConsole = outputToConsole == null ? Boolean.FALSE : outputToConsole;
2838
this.recordConsumer = recordConsumer;
2939
}
3040

3141
@Override
3242
public void log(LogRecord logRecord) {
33-
records.add(logRecord);
43+
final CapturedLogRecord record = new CapturedLogRecord(logRecord, currentFile);
44+
45+
records.add(record);
3446
if (outputToConsole)
35-
recordConsumer.accept(logRecord);
47+
recordConsumer.accept(record);
3648
}
3749

3850
public void clear() {
@@ -46,9 +58,7 @@ public void clear() {
4658
* @return list of filtered logRecords
4759
*/
4860
public List<LogRecord> filter(Severity severity) {
49-
return this.records.stream()
50-
.filter(record -> severityIsHigher(record, severity))
51-
.collect(Collectors.toList());
61+
return filter(severity, null);
5262
}
5363

5464
/**
@@ -58,16 +68,14 @@ public List<LogRecord> filter(Severity severity) {
5868
* @return list of filtered logRecords
5969
*/
6070
public List<LogRecord> filter(String text) {
61-
return this.records.stream()
62-
.filter(record -> messageContains(record, text))
63-
.collect(Collectors.toList());
71+
return filter(null, text);
6472
}
6573

6674
/**
6775
* Returns LogRecords that are equal or above the severity level and whose message contains text.
6876
*
69-
* @param severity Asciidoctor's severity level
70-
* @param text text to search for in the LogRecords
77+
* @param severity Asciidoctor's severity level (no filter applied when null)
78+
* @param text text to search for in the LogRecords (no filter applied when null)
7179
* @return list of filtered logRecords
7280
*/
7381
public List<LogRecord> filter(Severity severity, String text) {
@@ -96,11 +104,23 @@ public void processAll() {
96104
}
97105

98106
private static boolean severityIsHigher(LogRecord record, Severity severity) {
99-
return record.getSeverity().ordinal() >= severity.ordinal();
107+
if (severity == null) {
108+
return true;
109+
} else {
110+
return record.getSeverity().ordinal() >= severity.ordinal();
111+
}
100112
}
101113

102114
private static boolean messageContains(LogRecord record, String text) {
103-
return record.getMessage().contains(text);
115+
if (StringUtils.isBlank(text)) {
116+
return true;
117+
} else {
118+
return record.getMessage().contains(text);
119+
}
120+
}
121+
122+
public void setCurrentFile(File currentFile) {
123+
this.currentFile = currentFile;
104124
}
105125

106126
}

asciidoctor-maven-commons/src/main/java/org/asciidoctor/maven/site/SiteLogHandlerDeserializer.java

+9-12
Original file line numberDiff line numberDiff line change
@@ -20,18 +20,15 @@ public LogHandler deserialize(Xpp3Dom configNode) {
2020
if (logHandlerNode == null || !logHandlerNode.getName().equals(NODE_NAME))
2121
return logHandler;
2222

23-
logHandler.setOutputToConsole(deserializeOutputToConsole(logHandlerNode));
23+
logHandler.setOutputToConsole(getBoolean(logHandlerNode, "outputToConsole"));
24+
logHandler.setFailFast(getBoolean(logHandlerNode, "failFast"));
2425

2526
deserializeFailIf(logHandlerNode.getChild("failIf"))
26-
.ifPresent(logHandler::setFailIf);
27+
.ifPresent(logHandler::setFailIf);
2728

2829
return logHandler;
2930
}
3031

31-
private Boolean deserializeOutputToConsole(Xpp3Dom node) {
32-
return getBoolean(node, "outputToConsole");
33-
}
34-
3532
private Optional<FailIf> deserializeFailIf(Xpp3Dom node) {
3633
if (node == null)
3734
return Optional.empty();
@@ -58,18 +55,18 @@ private Optional<FailIf> deserializeFailIf(Xpp3Dom node) {
5855
return Optional.ofNullable(failIf);
5956
}
6057

61-
private String sanitizeString(Xpp3Dom severity) {
62-
String value = severity.getValue();
63-
return value == null ? "" : value.trim();
64-
}
65-
6658
private LogHandler defaultLogHandler() {
6759
LogHandler logHandler = new LogHandler();
6860
logHandler.setOutputToConsole(Boolean.TRUE);
6961
return logHandler;
7062
}
7163

72-
private Boolean getBoolean(Xpp3Dom node, String name) {
64+
private static String sanitizeString(Xpp3Dom node) {
65+
final String value = node.getValue();
66+
return value == null ? "" : value.trim();
67+
}
68+
69+
private static Boolean getBoolean(Xpp3Dom node, String name) {
7370
final Xpp3Dom child = node.getChild(name);
7471
if (child == null) {
7572
return Boolean.TRUE;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
package org.asciidoctor.maven.log;
2+
3+
import java.io.File;
4+
5+
import static org.assertj.core.api.Assertions.assertThat;
6+
7+
import org.asciidoctor.ast.Cursor;
8+
import org.asciidoctor.log.LogRecord;
9+
import org.asciidoctor.log.Severity;
10+
import org.junit.jupiter.api.Nested;
11+
import org.junit.jupiter.api.Test;
12+
13+
class CapturedLogRecordTest {
14+
15+
@Nested
16+
class WhenLogRecordsContainsCursor {
17+
18+
@Test
19+
void shouldReturnCursorFile() {
20+
final var cursorFile = new File("uno", "dos");
21+
final var recordFile = new File("tres", "quatre");
22+
final var cursor = new TestCursor(cursorFile.getAbsolutePath(), 0, null, null);
23+
24+
final var logRecord = new CapturedLogRecord(getLogRecord(cursor), recordFile);
25+
final var capturedLogRecord = new CapturedLogRecord(logRecord, cursorFile);
26+
27+
assertThat(capturedLogRecord.getCursor().getFile()).isEqualTo(cursorFile.getAbsolutePath());
28+
}
29+
30+
}
31+
32+
@Nested
33+
class WhenLogRecordsDoesNotContainsCursor {
34+
35+
@Test
36+
void shouldReturnLogRecordFileWhenCursorFileIsSet() {
37+
final var cursorFile = new File("uno", "dos");
38+
final var recordFile = new File("tres", "quatre");
39+
40+
final var logRecord = new CapturedLogRecord(getLogRecord(null), recordFile);
41+
final var capturedLogRecord = new CapturedLogRecord(logRecord, cursorFile);
42+
43+
assertThat(capturedLogRecord.getCursor().getFile()).isEqualTo(recordFile.getAbsolutePath());
44+
}
45+
46+
@Test
47+
void shouldReturnNullWhenCursorFileIsNoSet() {
48+
final var logRecord = new CapturedLogRecord(getLogRecord(null), null);
49+
final var capturedLogRecord = new CapturedLogRecord(logRecord, null);
50+
51+
assertThat(capturedLogRecord.getCursor()).isNull();
52+
}
53+
54+
}
55+
56+
private LogRecord getLogRecord(Cursor cursor) {
57+
return new LogRecord(Severity.INFO, cursor, "a message");
58+
}
59+
60+
}

0 commit comments

Comments
 (0)