org.sonar.api.batch.fs.internal.DefaultInputFile Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of sonar-plugin-api-impl Show documentation
Show all versions of sonar-plugin-api-impl Show documentation
Open source platform for continuous inspection of code quality
/*
* SonarQube
* Copyright (C) 2009-2024 SonarSource SA
* mailto:info AT sonarsource DOT com
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
package org.sonar.api.batch.fs.internal;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Collection;
import java.util.Optional;
import java.util.Set;
import java.util.function.Consumer;
import java.util.stream.Collectors;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.apache.commons.io.ByteOrderMark;
import org.apache.commons.io.input.BOMInputStream;
import org.sonar.api.batch.fs.InputFile;
import org.sonar.api.batch.fs.TextPointer;
import org.sonar.api.batch.fs.TextRange;
import static org.sonar.api.utils.Preconditions.checkArgument;
import static org.sonar.api.utils.Preconditions.checkState;
/**
* @since 4.2
* To create {@link InputFile} in tests, use TestInputFileBuilder.
*/
public class DefaultInputFile extends DefaultInputComponent implements InputFile {
private static final int DEFAULT_BUFFER_SIZE = 1024 * 4;
private final DefaultIndexedFile indexedFile;
private final String contents;
private final Consumer metadataGenerator;
private final Consumer scmStatusGenerator;
private boolean published;
private boolean excludedForCoverage;
private boolean excludedForDuplication;
private boolean ignoreAllIssues;
// Lazy init to save memory
private BitSet noSonarLines;
private Status status;
private Charset charset;
private Metadata metadata;
private Collection ignoreIssuesOnlineRanges;
private BitSet executableLines;
private boolean markedAsUnchanged;
public DefaultInputFile(DefaultIndexedFile indexedFile, Consumer metadataGenerator, Consumer scmStatusGenerator) {
this(indexedFile, metadataGenerator, null, scmStatusGenerator);
}
// For testing
public DefaultInputFile(DefaultIndexedFile indexedFile, Consumer metadataGenerator, @Nullable String contents,
Consumer scmStatusGenerator) {
super(indexedFile.scannerId());
this.indexedFile = indexedFile;
this.metadataGenerator = metadataGenerator;
this.scmStatusGenerator = scmStatusGenerator;
this.metadata = null;
this.markedAsUnchanged = false;
this.published = false;
this.excludedForCoverage = false;
this.contents = contents;
}
public void checkMetadata() {
if (metadata == null) {
metadataGenerator.accept(this);
}
}
private void checkScmStatus() {
if(status == null) {
scmStatusGenerator.accept(this);
}
}
@Override
public InputStream inputStream() throws IOException {
return contents != null ? new ByteArrayInputStream(contents.getBytes(charset()))
: new BOMInputStream(Files.newInputStream(path()),
ByteOrderMark.UTF_8, ByteOrderMark.UTF_16LE, ByteOrderMark.UTF_16BE, ByteOrderMark.UTF_32LE, ByteOrderMark.UTF_32BE);
}
public boolean isMarkedAsUnchanged() {
return markedAsUnchanged;
}
public DefaultInputComponent setMarkedAsUnchanged(boolean markedAsUnchanged) {
this.markedAsUnchanged = markedAsUnchanged;
return this;
}
@Override
public String contents() throws IOException {
if (contents != null) {
return contents;
} else {
ByteArrayOutputStream result = new ByteArrayOutputStream();
try (InputStream inputStream = inputStream()) {
byte[] buffer = new byte[DEFAULT_BUFFER_SIZE];
int length;
while ((length = inputStream.read(buffer)) != -1) {
result.write(buffer, 0, length);
}
}
return result.toString(charset().name());
}
}
public DefaultInputFile setPublished(boolean published) {
this.published = published;
return this;
}
public boolean isPublished() {
return published;
}
public DefaultInputFile setExcludedForCoverage(boolean excludedForCoverage) {
this.excludedForCoverage = excludedForCoverage;
return this;
}
public boolean isExcludedForCoverage() {
return excludedForCoverage;
}
public DefaultInputFile setExcludedForDuplication(boolean excludedForDuplication) {
this.excludedForDuplication = excludedForDuplication;
return this;
}
public boolean isExcludedForDuplication() {
return excludedForDuplication;
}
/**
* @deprecated since 6.6
*/
@Deprecated
@Override
public String relativePath() {
return indexedFile.relativePath();
}
public String getModuleRelativePath() {
return indexedFile.getModuleRelativePath();
}
public String getProjectRelativePath() {
return indexedFile.getProjectRelativePath();
}
@Override
public String absolutePath() {
return indexedFile.absolutePath();
}
@CheckForNull
public String oldRelativePath() {
return indexedFile.oldRelativePath();
}
@Override
public File file() {
return indexedFile.file();
}
@Override
public Path path() {
return indexedFile.path();
}
@CheckForNull
@Override
public String language() {
return indexedFile.language();
}
@Override
public Type type() {
return indexedFile.type();
}
/**
* Component key (without branch).
*/
@Override
public String key() {
return indexedFile.key();
}
@Override
public int hashCode() {
return indexedFile.hashCode();
}
@Override
public String toString() {
return indexedFile.toString();
}
/**
* {@link #setStatus(Status)}
*/
@Override
public Status status() {
checkScmStatus();
if(status == null) {
// scm might not be available, fallback to using hashes in the metadata
checkMetadata();
}
return status;
}
public boolean isStatusSet() {
return status != null;
}
@Override
public int lines() {
checkMetadata();
return metadata.lines();
}
@Override
public boolean isEmpty() {
checkMetadata();
return metadata.isEmpty();
}
@Override
public Charset charset() {
checkMetadata();
return charset;
}
public int lastValidOffset() {
checkMetadata();
return metadata.lastValidOffset();
}
/**
* Digest hash of the file.
*/
@Override
public String md5Hash() {
checkMetadata();
return metadata.hash();
}
public int nonBlankLines() {
checkMetadata();
return metadata.nonBlankLines();
}
public int[] originalLineStartOffsets() {
checkMetadata();
checkState(metadata.originalLineStartOffsets() != null, "InputFile is not properly initialized.");
checkState(metadata.originalLineStartOffsets().length == metadata.lines(),
"InputFile is not properly initialized. 'originalLineStartOffsets' property length should be equal to 'lines'");
return metadata.originalLineStartOffsets();
}
public int[] originalLineEndOffsets() {
checkMetadata();
checkState(metadata.originalLineEndOffsets() != null, "InputFile is not properly initialized.");
checkState(metadata.originalLineEndOffsets().length == metadata.lines(),
"InputFile is not properly initialized. 'originalLineEndOffsets' property length should be equal to 'lines'");
return metadata.originalLineEndOffsets();
}
@Override
public TextPointer newPointer(int line, int lineOffset) {
checkMetadata();
DefaultTextPointer textPointer = new DefaultTextPointer(line, lineOffset);
checkValid(textPointer, "pointer");
return textPointer;
}
@Override
public TextRange newRange(TextPointer start, TextPointer end) {
checkMetadata();
checkValid(start, "start pointer");
checkValid(end, "end pointer");
return newRangeValidPointers(start, end, false);
}
@Override
public TextRange newRange(int startLine, int startLineOffset, int endLine, int endLineOffset) {
checkMetadata();
TextPointer start = newPointer(startLine, startLineOffset);
TextPointer end = newPointer(endLine, endLineOffset);
return newRangeValidPointers(start, end, false);
}
@Override
public TextRange selectLine(int line) {
checkMetadata();
TextPointer startPointer = newPointer(line, 0);
TextPointer endPointer = newPointer(line, lineLength(line));
return newRangeValidPointers(startPointer, endPointer, true);
}
public void validate(TextRange range) {
checkMetadata();
checkValid(range.start(), "start pointer");
checkValid(range.end(), "end pointer");
}
/**
* Create Range from global offsets. Used for backward compatibility with older API.
*/
public TextRange newRange(int startOffset, int endOffset) {
checkMetadata();
return newRangeValidPointers(newPointer(startOffset), newPointer(endOffset), false);
}
public TextPointer newPointer(int globalOffset) {
checkMetadata();
checkArgument(globalOffset >= 0, "%s is not a valid offset for a file", globalOffset);
checkArgument(globalOffset <= lastValidOffset(), "%s is not a valid offset for file %s. Max offset is %s", globalOffset, this, lastValidOffset());
int line = findLine(globalOffset);
int startLineOffset = originalLineStartOffsets()[line - 1];
// In case the global offset is between \r and \n, move the pointer to a valid location
return new DefaultTextPointer(line, Math.min(globalOffset, originalLineEndOffsets()[line - 1]) - startLineOffset);
}
public DefaultInputFile setStatus(Status status) {
this.status = status;
return this;
}
public DefaultInputFile setCharset(Charset charset) {
this.charset = charset;
return this;
}
private void checkValid(TextPointer pointer, String owner) {
checkArgument(pointer.line() >= 1, "%s is not a valid line for a file", pointer.line());
checkArgument(pointer.line() <= this.metadata.lines(), "%s is not a valid line for %s. File %s has %s line(s)", pointer.line(), owner, this, metadata.lines());
checkArgument(pointer.lineOffset() >= 0, "%s is not a valid line offset for a file", pointer.lineOffset());
int lineLength = lineLength(pointer.line());
checkArgument(pointer.lineOffset() <= lineLength,
"%s is not a valid line offset for %s. File %s has %s character(s) at line %s", pointer.lineOffset(), owner, this, lineLength, pointer.line());
}
public int lineLength(int line) {
return originalLineEndOffsets()[line - 1] - originalLineStartOffsets()[line - 1];
}
private static TextRange newRangeValidPointers(TextPointer start, TextPointer end, boolean acceptEmptyRange) {
checkArgument(acceptEmptyRange ? (start.compareTo(end) <= 0) : (start.compareTo(end) < 0),
"Start pointer %s should be before end pointer %s", start, end);
return new DefaultTextRange(start, end);
}
private int findLine(int globalOffset) {
return Math.abs(Arrays.binarySearch(originalLineStartOffsets(), globalOffset) + 1);
}
public DefaultInputFile setMetadata(Metadata metadata) {
this.metadata = metadata;
return this;
}
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (this.getClass() != obj.getClass()) {
return false;
}
DefaultInputFile that = (DefaultInputFile) obj;
return this.getProjectRelativePath().equals(that.getProjectRelativePath());
}
@Override
public boolean isFile() {
return true;
}
@Override
public String filename() {
return indexedFile.filename();
}
@Override
public URI uri() {
return indexedFile.uri();
}
public void noSonarAt(Set noSonarLines) {
if (this.noSonarLines == null) {
this.noSonarLines = new BitSet(lines());
}
noSonarLines.forEach(l -> this.noSonarLines.set(l - 1));
}
public boolean hasNoSonarAt(int line) {
if (this.noSonarLines == null) {
return false;
}
return this.noSonarLines.get(line - 1);
}
public boolean isIgnoreAllIssues() {
checkMetadata();
return ignoreAllIssues;
}
public void setIgnoreAllIssues(boolean ignoreAllIssues) {
this.ignoreAllIssues = ignoreAllIssues;
}
public void addIgnoreIssuesOnLineRanges(Collection lineRanges) {
if (this.ignoreIssuesOnlineRanges == null) {
this.ignoreIssuesOnlineRanges = new ArrayList<>();
}
this.ignoreIssuesOnlineRanges.addAll(lineRanges);
}
public boolean isIgnoreAllIssuesOnLine(@Nullable Integer line) {
checkMetadata();
if (line == null || ignoreIssuesOnlineRanges == null) {
return false;
}
return ignoreIssuesOnlineRanges.stream().anyMatch(r -> r[0] <= line && line <= r[1]);
}
public void setExecutableLines(Set executableLines) {
checkState(this.executableLines == null, "Executable lines have already been saved for file: {}", this.toString());
this.executableLines = new BitSet(lines());
executableLines.forEach(l -> this.executableLines.set(l - 1));
}
public Optional> getExecutableLines() {
if (this.executableLines == null) {
return Optional.empty();
}
return Optional.of(this.executableLines.stream().map(i -> i + 1).boxed().collect(Collectors.toSet()));
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy