io.virtdata.basicsmappers.from_long.to_string.DirectoryLines Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of virtdata-lib-realer Show documentation
Show all versions of virtdata-lib-realer Show documentation
With inspiration from other libraries
package io.virtdata.basicsmappers.from_long.to_string;
import io.virtdata.annotations.Example;
import io.virtdata.annotations.ThreadSafeMapper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes;
import java.util.*;
import java.util.function.LongFunction;
import java.util.regex.Pattern;
/**
* Read each line in each matching file in a directory structure, providing one
* line for each time this function is called. The files are sorted at the time
* the function is initialized, and each line is read in order.
*
* This function does not produce the same result per cycle value. It is possible
* that different cycle inputs will return different inputs if the cycles are not
* applied in strict order. Still, this function is useful for consuming input
* from a set of files as input to a test or simulation.
*/
@ThreadSafeMapper
public class DirectoryLines implements LongFunction {
private final static Logger logger = LoggerFactory.getLogger(DirectoryLines.class);
private final Pattern namePattern;
private final String basepath;
private final List allFiles;
private Iterator stringIterator;
private Iterator pathIterator;
@Example({"DirectoryLines('/var/tmp/bardata', '.*')","load every line from every file in /var/tmp/bardata"})
public DirectoryLines(String basepath, String namePattern) {
this.basepath = basepath;
this.namePattern = Pattern.compile(namePattern);
allFiles = getAllFiles();
if (allFiles.size() == 0) {
throw new RuntimeException("Loaded zero files from " + basepath + ", full path:" + Paths.get(basepath).getFileName());
}
pathIterator = allFiles.iterator();
try {
stringIterator = Files.readAllLines(pathIterator.next()).iterator();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
@Override
public synchronized String apply(long value) {
while (!stringIterator.hasNext()) {
if (pathIterator.hasNext()) {
Path nextPath = pathIterator.next();
try {
stringIterator = Files.readAllLines(nextPath).iterator();
} catch (IOException e) {
throw new RuntimeException(e);
}
} else {
logger.debug("Resetting path iterator after exhausting input.");
pathIterator = allFiles.iterator();
}
}
return stringIterator.next();
}
private List getAllFiles() {
logger.debug("Loading file paths from " + basepath);
Set options = new HashSet<>();
options.add(FileVisitOption.FOLLOW_LINKS);
FileList fileList = new FileList(namePattern);
try {
Files.walkFileTree(Paths.get(basepath), options, 10, fileList);
} catch (IOException e) {
throw new RuntimeException(e);
}
logger.debug("File reader: " + fileList.toString() + " in path: " + Paths.get(basepath).getFileName());
fileList.paths.sort(Path::compareTo);
return fileList.paths;
}
private static class FileList implements FileVisitor {
public final Pattern namePattern;
public int seen;
public int kept;
public List paths = new ArrayList<>();
private FileList(Pattern namePattern) {
this.namePattern = namePattern;
}
@Override
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
seen++;
if (file.toString().matches(namePattern.pattern())) {
paths.add(file);
kept++;
}
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult visitFileFailed(Path file, IOException exc) throws IOException {
logger.warn("Error traversing file: " + file + ":" + exc);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException {
return FileVisitResult.CONTINUE;
}
public String toString() {
return "" + kept + "/" + seen + " files with pattern '" + namePattern + "'";
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy