org.sonar.plugins.javascript.analysis.AbstractAnalysis Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of sonar-javascript-plugin Show documentation
Show all versions of sonar-javascript-plugin Show documentation
Code Analyzer for JavaScript/TypeScript/CSS
/*
* SonarQube JavaScript Plugin
* Copyright (C) 2011-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.plugins.javascript.analysis;
import java.io.IOException;
import java.util.List;
import java.util.concurrent.TimeUnit;
import javax.annotation.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sonar.api.batch.fs.InputFile;
import org.sonar.api.batch.sensor.SensorContext;
import org.sonar.plugins.javascript.CancellationException;
import org.sonar.plugins.javascript.JavaScriptFilePredicate;
import org.sonar.plugins.javascript.JavaScriptLanguage;
import org.sonar.plugins.javascript.TypeScriptLanguage;
import org.sonar.plugins.javascript.analysis.cache.CacheAnalysis;
import org.sonar.plugins.javascript.analysis.cache.CacheStrategies;
import org.sonar.plugins.javascript.api.JsFile;
import org.sonar.plugins.javascript.api.estree.ESTree;
import org.sonar.plugins.javascript.bridge.AnalysisMode;
import org.sonar.plugins.javascript.bridge.AnalysisWarningsWrapper;
import org.sonar.plugins.javascript.bridge.BridgeServer;
import org.sonar.plugins.javascript.bridge.BridgeServer.TsProgram;
import org.sonar.plugins.javascript.bridge.ESTreeFactory;
import org.sonar.plugins.javascript.bridge.protobuf.Node;
import org.sonar.plugins.javascript.utils.ProgressReport;
abstract class AbstractAnalysis {
private static final Logger LOG = LoggerFactory.getLogger(AbstractAnalysis.class);
static final String PROGRESS_REPORT_TITLE = "Progress of JavaScript/TypeScript analysis";
static final long PROGRESS_REPORT_PERIOD = TimeUnit.SECONDS.toMillis(10);
final BridgeServer bridgeServer;
final AnalysisProcessor analysisProcessor;
SensorContext context;
ContextUtils contextUtils;
JsTsChecks checks;
ProgressReport progressReport;
AnalysisMode analysisMode;
protected final AnalysisWarningsWrapper analysisWarnings;
private AnalysisConsumers consumers;
AbstractAnalysis(
BridgeServer bridgeServer,
AnalysisProcessor analysisProcessor,
AnalysisWarningsWrapper analysisWarnings
) {
this.bridgeServer = bridgeServer;
this.analysisProcessor = analysisProcessor;
this.analysisWarnings = analysisWarnings;
}
protected static String inputFileLanguage(InputFile file) {
return JavaScriptFilePredicate.isTypeScriptFile(file)
? TypeScriptLanguage.KEY
: JavaScriptLanguage.KEY;
}
void initialize(SensorContext context, JsTsChecks checks, AnalysisMode analysisMode, AnalysisConsumers consumers) {
LOG.debug("Initializing {}", getClass().getName());
this.context = context;
contextUtils = new ContextUtils(context);
this.checks = checks;
this.analysisMode = analysisMode;
this.consumers = consumers;
}
protected boolean isJavaScript(InputFile file) {
return inputFileLanguage(file).equals(JavaScriptLanguage.KEY);
}
abstract void analyzeFiles(List inputFiles, List tsConfigs) throws IOException;
protected void analyzeFile(InputFile file, @Nullable List tsConfigs, @Nullable TsProgram tsProgram) throws IOException {
if (context.isCancelled()) {
throw new CancellationException(
"Analysis interrupted because the SensorContext is in cancelled state"
);
}
var cacheStrategy = CacheStrategies.getStrategyFor(context, file);
if (cacheStrategy.isAnalysisRequired()) {
try {
LOG.debug("Analyzing file: {}", file.uri());
progressReport.nextFile(file.toString());
var fileContent = contextUtils.shouldSendFileContent(file) ? file.contents() : null;
var skipAst = !consumers.hasConsumers() || !contextUtils.isSonarArmorEnabled();
var request = getJsAnalysisRequest(file, fileContent, tsProgram, tsConfigs, skipAst);
var response = isJavaScript(file)
? bridgeServer.analyzeJavaScript(request)
: bridgeServer.analyzeTypeScript(request);
analysisProcessor.processResponse(context, checks, file, response);
cacheStrategy.writeAnalysisToCache(
CacheAnalysis.fromResponse(response.ucfgPaths(), response.cpdTokens()),
file
);
acceptAstResponse(response, file);
} catch (IOException e) {
LOG.error("Failed to get response while analyzing " + file.uri(), e);
throw e;
}
} else {
LOG.debug("Processing cache analysis of file: {}", file.uri());
var cacheAnalysis = cacheStrategy.readAnalysisFromCache();
analysisProcessor.processCacheAnalysis(context, file, cacheAnalysis);
}
}
private void acceptAstResponse(BridgeServer.AnalysisResponse response, InputFile file) {
Node responseAst = response.ast();
if (responseAst != null) {
// When we haven't serialized the AST:
// either because no consumer is listening
// or the file extension or AST nodes are unsupported
try {
ESTree.Program program = ESTreeFactory.from(responseAst, ESTree.Program.class);
consumers.accept(new JsFile(file, program));
} catch (Exception e) {
LOG.debug("Failed to deserialize AST for file: {}", file.uri(), e);
}
}
}
private BridgeServer.JsAnalysisRequest getJsAnalysisRequest(
InputFile file,
@Nullable String fileContent,
@Nullable TsProgram tsProgram,
@Nullable List tsConfigs,
boolean skipAst
) {
return new BridgeServer.JsAnalysisRequest(
file.absolutePath(),
file.type().toString(),
inputFileLanguage(file),
fileContent,
contextUtils.ignoreHeaderComments(),
tsConfigs,
tsProgram != null ? tsProgram.programId() : null,
analysisMode.getLinterIdFor(file),
skipAst
);
}
}