Lines.java
// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.text;
import static de.larssh.utils.collection.Iterators.peekableIterator;
import static java.util.stream.Collectors.toList;
import java.io.BufferedReader;
import java.io.Reader;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.function.BiFunction;
import java.util.function.BiPredicate;
import java.util.function.Function;
import java.util.stream.Stream;
import de.larssh.utils.collection.Iterators;
import de.larssh.utils.collection.Maps;
import de.larssh.utils.collection.PeekableIterator;
/**
* This class contains helper methods for line based processing.
*
* <p>
* <b>Usage example 1:</b> The following shows how to merge lines to logical log
* lines.
*
* <pre>
* // Some lines of example log files
* final List<String> fileLines = lines("[20:57:30] Thread 0: Start request\n"
* + "[20:57:31] Thread 0: Request ID: Example 1\n"
* + "[20:57:31] Thread 0: Request failed: SomeException\n"
* + " on line 12\n"
* + " on line 34\n"
* + "[20:57:32] Thread 0: Stop request");
*
* // Calculating logical log lines based on consecutive lines
* final Stream<List<String>> logicalLogLines = consecutive(
* fileLines,
* (lines, line) -> !line.startsWith("["));
*
* // Output logical log lines with three dashes in between each of them
* System.out.println(logicalLogLines //
* .map(lines -> lines.stream().collect(joining("\n")))
* .collect(joining("\n---\n")));
* </pre>
*
* <p>
* Console output for usage example 1:
*
* <pre>
* [20:57:30] Thread 1: Start request
* ---
* [20:57:31] Thread 1: Request ID: Example 1
* ---
* [20:57:31] Thread 1: Request failed: SomeException
* on line 12
* on line 34
* ---
* [20:57:32] Thread 1: Stop request
* </pre>
*
* <p>
* <b>Usage example 2:</b> The following shows how to group log lines based on
* their thread name.
*
* <pre>
* // Some lines of example log files
* final List<String> fileLines = lines(
* "[20:57:30] Thread 1: Start request\n"
* + "[20:57:30] Thread 2: Stop request\n"
* + "[20:57:30] Thread 1: Request ID: Example 2.1\n"
* + "[20:57:31] Thread 2: Start request\n"
* + "[20:57:31] Thread 1: Request processed\n"
* + "[20:57:31] Thread 2: Request ID: Example 2.2\n"
* + "[20:57:32] Thread 1: Stop request\n"
* + "[20:57:32] Thread 2: Request processed");
*
* // Grouping log lines based on thread names
* final Stream<Entry<String, List<String>>> groupedLogLines = grouped(
* fileLines,
* line -> line.substring(11, 19),
* (lines, line) -> {
* if (line.endsWith("Start request")) {
* return GroupedLineType.START;
* }
* if (line.endsWith("Stop request")) {
* return GroupedLineType.END;
* }
* return GroupedLineType.MIDDLE;
* });
*
* // Output grouped lines with three dashes between each group
* System.out.println(groupedLogLines
* .map(entry -> entry.getKey() + ":\n" + entry.getValue().stream().collect(joining("\n")))
* .collect(joining("\n---\n")));
* </pre>
*
* <p>
* Console output for usage example 2:
*
* <pre>
* Thread 2:
* [20:57:30] Thread 2: Stop request
* ---
* Thread 1:
* [20:57:30] Thread 1: Start request
* [20:57:30] Thread 1: Request ID: Example 2.1
* [20:57:31] Thread 1: Request processed
* [20:57:32] Thread 1: Stop request
* ---
* Thread 2:
* [20:57:31] Thread 2: Start request
* [20:57:31] Thread 2: Request ID: Example 2.2
* [20:57:32] Thread 2: Request processed
* </pre>
*
* <p>
* Based on your your needs you might need to combine the methods
* {@code lines(...)}, {@code consecutive(...)} and {@code grouped(...)}. In
* addition {@link de.larssh.utils.io.MultiReader} might be helpful to parse
* data, which is spread over multiple files.
*/
public final class Lines {
/**
* Creates lists of consecutive lines based on {@code lines}. An input line is
* added to an output list whenever {@code isNextConsecutive} returns
* {@code true}. On {@code false} a new list gets created.
*
* <p>
* Check out usage example 1 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <T> type of line (most probably {@link String})
* @param lines the lines
* @param isNextConsecutive when returning {@code true} the input line is added
* to an output list, else a new list gets created. The
* first argument is the previous list of lines,
* whereas the second argument is the current line.
* @return stream of lists containing consecutive lines
*/
public static <T> Stream<List<T>> consecutive(final Iterable<T> lines, final BiPredicate<List<T>, T> isNextConsecutive) {
return consecutive(lines.iterator(), isNextConsecutive);
}
/**
* Creates lists of consecutive lines based on {@code lines}. An input line is
* added to an output list whenever {@code isNextConsecutive} returns
* {@code true}. On {@code false} a new list gets created.
*
* <p>
* Check out usage example 1 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <T> type of line (most probably {@link String})
* @param linesIterator the lines
* @param isNextConsecutive when returning {@code true} the input line is added
* to an output list, else a new list gets created. The
* first argument is the previous list of lines,
* whereas the second argument is the current line.
* @return stream of lists containing consecutive lines
*/
public static <T> Stream<List<T>> consecutive(final Iterator<T> linesIterator, final BiPredicate<List<T>, T> isNextConsecutive) {
final PeekableIterator<T> peekableLinesIterator = peekableIterator(linesIterator);
return Iterators.stream(state -> {
final List<T> lines = new ArrayList<>();
while (peekableLinesIterator.hasNext()) {
lines.add(peekableLinesIterator.next());
// Fail fast if the peeked line is not consecutive
if (peekableLinesIterator.hasNext() && !isNextConsecutive.test(lines, peekableLinesIterator.peek())) {
return lines;
}
}
return lines.isEmpty() ? state.endOfData() : lines;
});
}
/**
* Creates lists of consecutive lines based on {@code lines}. An input line is
* added to an output list whenever {@code isNextConsecutive} returns
* {@code true}. On {@code false} a new list gets created.
*
* <p>
* Check out usage example 1 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <T> type of line (most probably {@link String})
* @param lines the lines
* @param isNextConsecutive when returning {@code true} the input line is added
* to an output list, else a new list gets created. The
* first argument is the previous list of lines,
* whereas the second argument is the current line.
* @return stream of lists containing consecutive lines
*/
public static <T> Stream<List<T>> consecutive(final Stream<T> lines, final BiPredicate<List<T>, T> isNextConsecutive) {
return consecutive(lines.iterator(), isNextConsecutive);
}
/**
* Groups lines based on a key and line types. While {@code getGroupKey}
* calculates the key that is used for grouping {@code getLineType} calculates
* the line type, which specifies the way groups of lines are created and
* closed.
*
* <p>
* Check out usage example 2 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <K> type of the group key
* @param <V> type of line
* @param lines the lines
* @param getGroupKey calculates the lines group key
* @param getLineType calculates the lines type to specify the way groups of
* lines are created and closed. The first argument is the
* previous list of lines, whereas the second argument is the
* current line.
* @return stream of entries with the group key as key and the grouped lines as
* value
*/
public static <K, V> Stream<Entry<K, List<V>>> grouped(final Iterable<V> lines, final Function<V, K> getGroupKey, final BiFunction<List<V>, V, GroupedLineType> getLineType) {
return grouped(lines.iterator(), getGroupKey, getLineType);
}
/**
* Groups lines based on a key and line types. While {@code getGroupKey}
* calculates the key that is used for grouping {@code getLineType} calculates
* the line type, which specifies the way groups of lines are created and
* closed.
*
* <p>
* Check out usage example 2 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <K> type of the group key
* @param <V> type of line
* @param lines the lines
* @param getGroupKey calculates the lines group key
* @param getLineType calculates the lines type to specify the way groups of
* lines are created and closed. The first argument is the
* previous list of lines, whereas the second argument is the
* current line.
* @return stream of entries with the group key as key and the grouped lines as
* value
*/
@SuppressWarnings("PMD.AvoidInstantiatingObjectsInLoops")
public static <K, V> Stream<Entry<K, List<V>>> grouped(final Iterator<V> lines, final Function<V, K> getGroupKey, final BiFunction<List<V>, V, GroupedLineType> getLineType) {
// Buffer of so-far not-closed groups of lines
final Map<K, List<V>> groups = new LinkedHashMap<>();
return Iterators.stream(state -> {
while (lines.hasNext()) {
final V line = lines.next();
final K groupKey = getGroupKey.apply(line);
List<V> groupOfCurrentLine = groups.computeIfAbsent(groupKey, key -> new ArrayList<>());
final GroupedLineType lineType = getLineType.apply(groupOfCurrentLine, line);
if (lineType == GroupedLineType.END) {
groupOfCurrentLine.add(line);
return Maps.entry(groupKey, groups.remove(groupKey));
}
if (lineType == GroupedLineType.START && !groupOfCurrentLine.isEmpty()) {
groupOfCurrentLine = new ArrayList<>();
groupOfCurrentLine.add(line);
return Maps.entry(groupKey, groups.put(groupKey, groupOfCurrentLine));
}
groupOfCurrentLine.add(line);
}
if (groups.isEmpty()) {
return state.endOfData();
}
final K groupKey = groups.keySet().iterator().next();
return Maps.entry(groupKey, groups.remove(groupKey));
});
}
/**
* Groups lines based on a key and line types. While {@code getGroupKey}
* calculates the key that is used for grouping {@code getLineType} calculates
* the line type, which specifies the way groups of lines are created and
* closed.
*
* <p>
* Check out usage example 2 at {@link Lines}.
*
* <p>
* <b>Tip:</b> {@code lines} must not necessarily consist of elements of type
* {@link String}.
*
* @param <K> type of the group key
* @param <V> type of line
* @param lines the lines
* @param getGroupKey calculates the lines group key
* @param getLineType calculates the lines type to specify the way groups of
* lines are created and closed. The first argument is the
* previous list of lines, whereas the second argument is the
* current line.
* @return stream of entries with the group key as key and the grouped lines as
* value
*/
public static <K, V> Stream<Entry<K, List<V>>> grouped(final Stream<V> lines, final Function<V, K> getGroupKey, final BiFunction<List<V>, V, GroupedLineType> getLineType) {
return grouped(lines.iterator(), getGroupKey, getLineType);
}
/**
* Reads all characters of {@code reader} and splits them into lines using
* {@link BufferedReader#lines()}.
*
* @param reader character stream to split into lines
* @return list of lines
*/
public static Stream<String> lines(final Reader reader) {
final BufferedReader bufferedReader = reader instanceof BufferedReader ? (BufferedReader) reader : new BufferedReader(reader);
return bufferedReader.lines();
}
/**
* Splits {@code value} into lines using {@link BufferedReader#lines()}.
*
* @param value value to split
* @return list of lines
*/
public static List<String> lines(final String value) {
return lines(new StringReader(value)).collect(toList());
}
/**
* Specifies the way groups of lines are created and closed.
*/
@SuppressWarnings("PMD.UnnecessaryModifier")
public static enum GroupedLineType {
/**
* Start of a group of lines
*
* <p>
* Creates a new group of lines. Closes a previously not-closed group, if any.
*/
START, /**
* A grouped line, placed in a groups middle
*
* <p>
* Adds the current line to an existing group of lines. If no group was open, a
* new group is created.
*/
MIDDLE, /**
* End of a group of lines
*
* <p>
* Closes an open group of lines, if any.
*/
END;
}
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
private Lines() {
throw new java.lang.UnsupportedOperationException("This is a utility class and cannot be instantiated");
}
}