graphql.parser.MultiSourceReader Maven / Gradle / Ivy
package graphql.parser;
import graphql.Assert;
import graphql.PublicApi;
import graphql.util.LockKit;
import java.io.IOException;
import java.io.LineNumberReader;
import java.io.Reader;
import java.io.StringReader;
import java.io.UncheckedIOException;
import java.util.ArrayList;
import java.util.List;
/**
* This reader allows you to read N number readers and combine them as one logical reader
* however you can then map back to the underlying readers in terms of their source name
* and the relative lines numbers.
*
* It can also track all data in memory if you want to have all of the previous read data in
* place at some point in time.
*/
@PublicApi
public class MultiSourceReader extends Reader {
// In Java version 16+, LineNumberReader.read considers end-of-stream to be a line terminator
// and will increment the line number, whereas in previous versions it doesn't.
private static final boolean LINE_NUMBER_READER_EOS_IS_TERMINATOR;
private final List sourceParts;
private final StringBuilder data = new StringBuilder();
private int currentIndex = 0;
private int overallLineNumber = 0;
private final boolean trackData;
private final LockKit.ReentrantLock readerLock = new LockKit.ReentrantLock();
static {
LINE_NUMBER_READER_EOS_IS_TERMINATOR = lineNumberReaderEOSIsTerminator();
}
private static boolean lineNumberReaderEOSIsTerminator() {
LineNumberReader reader = new LineNumberReader(new StringReader("a"));
try {
reader.read();
reader.read();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
return reader.getLineNumber() > 0;
}
private MultiSourceReader(Builder builder) {
this.sourceParts = builder.sourceParts;
this.trackData = builder.trackData;
}
@Override
public int read(char[] cbuf, int off, int len) throws IOException {
while (true) {
readerLock.lock();
try {
if (currentIndex >= sourceParts.size()) {
return -1;
}
SourcePart sourcePart = sourceParts.get(currentIndex);
int read = sourcePart.lineReader.read(cbuf, off, len);
if (read == -1) {
currentIndex++;
sourcePart.reachedEndOfStream = true;
} else if (read > 0) {
sourcePart.lastRead = cbuf[off + read - 1];
}
// note: calcLineNumber() must be called after updating sourcePart.reachedEndOfStream
// and sourcePart.lastRead
overallLineNumber = calcLineNumber();
if (read != -1) {
trackData(cbuf, off, read);
return read;
}
} finally {
readerLock.unlock();
}
}
}
private void trackData(char[] cbuf, int off, int len) {
if (trackData) {
data.append(cbuf, off, len);
}
}
private int calcLineNumber() {
int linenumber = 0;
for (SourcePart sourcePart : sourceParts) {
linenumber += sourcePart.getLineNumber();
}
return linenumber;
}
public static class SourceAndLine {
private String sourceName = null;
private int line = 0;
public String getSourceName() {
return sourceName;
}
public int getLine() {
return line;
}
@Override
public String toString() {
return "SourceAndLine{" +
"sourceName='" + sourceName + '\'' +
", line=" + line +
'}';
}
}
/**
* This returns the source name and line number given an overall line number
*
* This is zeroes based like {@link java.io.LineNumberReader#getLineNumber()}
*
* @param overallLineNumber the over all line number
*
* @return the source name and relative line number to that source
*/
public SourceAndLine getSourceAndLineFromOverallLine(int overallLineNumber) {
SourceAndLine sourceAndLine = new SourceAndLine();
if (sourceParts.isEmpty()) {
return sourceAndLine;
}
if (overallLineNumber == 0) {
sourceAndLine.sourceName = sourceParts.get(0).sourceName;
sourceAndLine.line = 0;
return sourceAndLine;
}
SourcePart currentPart;
if (currentIndex >= sourceParts.size()) {
currentPart = sourceParts.get(sourceParts.size() - 1);
} else {
currentPart = sourceParts.get(currentIndex);
}
int page = 0;
int previousPage;
for (SourcePart sourcePart : sourceParts) {
sourceAndLine.sourceName = sourcePart.sourceName;
if (sourcePart == currentPart) {
// we cant go any further
int partLineNumber = currentPart.getLineNumber();
previousPage = page;
page += partLineNumber;
if (page > overallLineNumber) {
sourceAndLine.line = overallLineNumber - previousPage;
} else {
sourceAndLine.line = page;
}
return sourceAndLine;
} else {
previousPage = page;
int partLineNumber = sourcePart.getLineNumber();
page += partLineNumber;
if (page > overallLineNumber) {
sourceAndLine.line = overallLineNumber - previousPage;
return sourceAndLine;
}
}
}
sourceAndLine.line = overallLineNumber - page;
return sourceAndLine;
}
/**
* @return the line number of the current source. This is zeroes based like {@link java.io.LineNumberReader#getLineNumber()}
*/
public int getLineNumber() {
return readerLock.callLocked(() -> {
if (sourceParts.isEmpty()) {
return 0;
}
if (currentIndex >= sourceParts.size()) {
return sourceParts.get(sourceParts.size() - 1).getLineNumber();
}
return sourceParts.get(currentIndex).getLineNumber();
});
}
/**
* @return The name of the current source
*/
public String getSourceName() {
return readerLock.callLocked(() -> {
if (sourceParts.isEmpty()) {
return null;
}
if (currentIndex >= sourceParts.size()) {
return sourceParts.get(sourceParts.size() - 1).sourceName;
}
return sourceParts.get(currentIndex).sourceName;
});
}
/**
* @return the overall line number of the all the sources. This is zeroes based like {@link java.io.LineNumberReader#getLineNumber()}
*/
public int getOverallLineNumber() {
return overallLineNumber;
}
public List getData() {
LineNumberReader reader = new LineNumberReader(new StringReader(data.toString()));
List lines = new ArrayList<>();
while (true) {
try {
String line = reader.readLine();
if (line == null) {
return lines;
}
lines.add(line);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
}
@Override
public void close() throws IOException {
readerLock.lock();
try {
for (SourcePart sourcePart : sourceParts) {
if (!sourcePart.closed) {
sourcePart.lineReader.close();
sourcePart.closed = true;
}
}
} finally {
readerLock.unlock();
}
}
private static class SourcePart {
String sourceName;
LineNumberReader lineReader;
boolean closed;
char lastRead;
boolean reachedEndOfStream = false;
/**
* This handles the discrepancy between LineNumberReader.getLineNumber() for Java versions
* 16+ vs below. Use this instead of lineReader.getLineNumber() directly.
* @return The current line number. EOS is not considered a line terminator.
*/
int getLineNumber() {
int lineNumber = lineReader.getLineNumber();
if (reachedEndOfStream
&& LINE_NUMBER_READER_EOS_IS_TERMINATOR
&& lastRead != '\r'
&& lastRead != '\n') {
return Math.max(lineNumber - 1, 0);
}
return lineNumber;
}
}
public static Builder newMultiSourceReader() {
return new Builder();
}
public static class Builder {
List sourceParts = new ArrayList<>();
boolean trackData = true;
private Builder() {
}
public Builder reader(Reader reader, String sourceName) {
SourcePart sourcePart = new SourcePart();
sourcePart.lineReader = new LineNumberReader(Assert.assertNotNull(reader));
sourcePart.sourceName = sourceName;
sourcePart.closed = false;
sourceParts.add(sourcePart);
return this;
}
public Builder string(String input, String sourceName) {
SourcePart sourcePart = new SourcePart();
sourcePart.lineReader = new LineNumberReader(new StringReader(input));
sourcePart.sourceName = sourceName;
sourcePart.closed = false;
sourceParts.add(sourcePart);
return this;
}
public Builder trackData(boolean trackData) {
this.trackData = trackData;
return this;
}
public MultiSourceReader build() {
return new MultiSourceReader(this);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy