org.sonar.python.semantic.Scope Maven / Gradle / Ivy
/*
* SonarQube Python Plugin
* Copyright (C) 2011-2023 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.python.semantic;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.sonar.plugins.python.api.PythonFile;
import org.sonar.plugins.python.api.symbols.AmbiguousSymbol;
import org.sonar.plugins.python.api.symbols.FunctionSymbol;
import org.sonar.plugins.python.api.symbols.Symbol;
import org.sonar.plugins.python.api.symbols.Usage;
import org.sonar.plugins.python.api.tree.ClassDef;
import org.sonar.plugins.python.api.tree.FunctionDef;
import org.sonar.plugins.python.api.tree.ImportFrom;
import org.sonar.plugins.python.api.tree.Name;
import org.sonar.plugins.python.api.tree.Parameter;
import org.sonar.plugins.python.api.tree.Tree;
import org.sonar.python.tree.FunctionDefImpl;
import org.sonar.python.types.TypeShed;
class Scope {
final Tree rootTree;
private PythonFile pythonFile;
private String fullyQualifiedModuleName;
private final ProjectLevelSymbolTable projectLevelSymbolTable;
private final Scope parent;
final Map symbolsByName = new HashMap<>();
private final Set symbols = new HashSet<>();
final Set builtinSymbols = new HashSet<>();
private final Set globalNames = new HashSet<>();
private final Set nonlocalNames = new HashSet<>();
final Map instanceAttributesByName = new HashMap<>();
Scope(@Nullable Scope parent, Tree rootTree, PythonFile pythonFile, String fullyQualifiedModuleName, ProjectLevelSymbolTable projectLevelSymbolTable) {
this.parent = parent;
this.rootTree = rootTree;
this.pythonFile = pythonFile;
this.fullyQualifiedModuleName = fullyQualifiedModuleName;
this.projectLevelSymbolTable = projectLevelSymbolTable;
}
Set symbols() {
return Collections.unmodifiableSet(symbols);
}
void createBuiltinSymbol(String name, Map typeShedSymbols) {
SymbolImpl symbol;
Symbol typeShedSymbol = typeShedSymbols.get(name);
if (typeShedSymbol != null) {
symbol = ((SymbolImpl) typeShedSymbol).copyWithoutUsages();
} else {
symbol = new SymbolImpl(name, name);
}
symbols.add(symbol);
builtinSymbols.add(symbol);
symbolsByName.put(name, symbol);
}
void createSymbolsFromWildcardImport(Set importedSymbols, ImportFrom importFrom) {
importedSymbols.forEach(symbol -> {
if (!isExistingSymbol(symbol.name())) {
symbols.add(symbol);
symbolsByName.put(symbol.name(), symbol);
((SymbolImpl) symbol).addUsage(importFrom, Usage.Kind.IMPORT);
} else {
SymbolImpl originalSymbol = resolve(symbol.name());
if (originalSymbol != null) {
resetSymbolInfo(symbol.fullyQualifiedName(), originalSymbol);
originalSymbol.addUsage(importFrom, Usage.Kind.IMPORT);
}
}
});
}
void createSelfParameter(Parameter parameter) {
Name nameTree = parameter.name();
if (nameTree == null) {
return;
}
String symbolName = nameTree.name();
SymbolImpl symbol = new SelfSymbolImpl(symbolName, parent);
symbols.add(symbol);
symbolsByName.put(symbolName, symbol);
symbol.addUsage(nameTree, Usage.Kind.PARAMETER);
}
void addFunctionSymbol(FunctionDef functionDef, @Nullable String fullyQualifiedName) {
String symbolName = functionDef.name().name();
if (isExistingSymbol(symbolName)) {
addBindingUsage(functionDef.name(), Usage.Kind.FUNC_DECLARATION, fullyQualifiedName);
} else {
FunctionSymbolImpl functionSymbol = new FunctionSymbolImpl(functionDef, fullyQualifiedName, pythonFile);
functionSymbol.setIsDjangoView(projectLevelSymbolTable.isDjangoView(fullyQualifiedName));
((FunctionDefImpl) functionDef).setFunctionSymbol(functionSymbol);
symbols.add(functionSymbol);
symbolsByName.put(symbolName, functionSymbol);
functionSymbol.addUsage(functionDef.name(), Usage.Kind.FUNC_DECLARATION);
}
}
public Symbol copySymbol(String symbolName, Symbol symbol) {
return copySymbol(symbolName, symbol, new HashSet<>());
}
private Symbol copySymbol(String symbolName, Symbol symbol, Set alreadyVisitedSymbols) {
alreadyVisitedSymbols.add(symbol);
switch (symbol.kind()) {
case FUNCTION:
return new FunctionSymbolImpl(symbolName, (FunctionSymbol) symbol);
case CLASS:
return copyClassSymbol(symbolName, (ClassSymbolImpl) symbol, alreadyVisitedSymbols);
case AMBIGUOUS:
Set alternativeSymbols = ((AmbiguousSymbol) symbol).alternatives().stream()
.map(s -> copySymbol(symbolName, s, alreadyVisitedSymbols))
.collect(Collectors.toSet());
return new AmbiguousSymbolImpl(symbolName, symbol.fullyQualifiedName(), alternativeSymbols);
default:
SymbolImpl copiedSymbol = ((SymbolImpl) symbol).copyWithoutUsages(symbolName);
for (Map.Entry kv: ((SymbolImpl) symbol).getChildrenSymbolByName().entrySet()) {
copiedSymbol.addChildSymbol(((SymbolImpl) kv.getValue()).copyWithoutUsages());
}
return copiedSymbol;
}
}
private ClassSymbolImpl copyClassSymbol(String symbolName, ClassSymbolImpl originalClassSymbol, Set alreadyVisitedSymbols) {
// Must use symbolName to preserve import aliases
ClassSymbolImpl classSymbol = (ClassSymbolImpl) ClassSymbolImpl.copyFrom(symbolName, originalClassSymbol);
if (originalClassSymbol.hasEvaluatedSuperClasses()) {
for (Symbol originalSymbol : originalClassSymbol.superClasses()) {
Symbol globalSymbol = projectLevelSymbolTable.getSymbol(originalSymbol.fullyQualifiedName());
if (globalSymbol != null && globalSymbol.kind() == Symbol.Kind.CLASS) {
Symbol parentClass = alreadyVisitedSymbols.contains(globalSymbol)
? new SymbolImpl(globalSymbol.name(), globalSymbol.fullyQualifiedName())
: copySymbol(globalSymbol.name(), globalSymbol, alreadyVisitedSymbols);
classSymbol.addSuperClass(parentClass);
} else {
classSymbol.addSuperClass(originalSymbol);
}
}
}
classSymbol.addMembers(originalClassSymbol
.declaredMembers().stream()
.map(m -> ((SymbolImpl) m).copyWithoutUsages())
.collect(Collectors.toList()));
if (originalClassSymbol.hasSuperClassWithoutSymbol()) {
classSymbol.setHasSuperClassWithoutSymbol();
}
return classSymbol;
}
void addModuleSymbol(Name nameTree, @CheckForNull String fullyQualifiedName) {
String symbolName = nameTree.name();
Set moduleExportedSymbols = projectLevelSymbolTable.getSymbolsFromModule(fullyQualifiedName);
if (moduleExportedSymbols != null && !isExistingSymbol(symbolName)) {
SymbolImpl moduleSymbol = new SymbolImpl(symbolName, fullyQualifiedName);
moduleExportedSymbols.forEach(moduleSymbol::addChildSymbol);
this.symbols.add(moduleSymbol);
symbolsByName.put(symbolName, moduleSymbol);
} else if (fullyQualifiedName != null && !fullyQualifiedName.equals(fullyQualifiedModuleName)) {
Collection standardLibrarySymbols = TypeShed.symbolsForModule(fullyQualifiedName).values();
if (!standardLibrarySymbols.isEmpty()) {
if (!isExistingSymbol(symbolName)) {
Symbol moduleSymbol = new SymbolImpl(symbolName, fullyQualifiedName);
this.symbols.add(moduleSymbol);
symbolsByName.put(symbolName, moduleSymbol);
}
symbolsByName.computeIfPresent(symbolName, (k, v) -> {
standardLibrarySymbols.forEach(symbol -> ((SymbolImpl) v).addChildSymbol(copySymbol(symbol.name(), symbol)));
this.symbols.add(v);
return v;
});
}
}
addBindingUsage(nameTree, Usage.Kind.IMPORT, fullyQualifiedName);
}
void addSubmoduleSymbol(Name nameTree, String fullyQualifiedName) {
String symbolName = nameTree.name();
List names = Arrays.stream(fullyQualifiedName.split("\\.")).collect(Collectors.toList());
Set moduleExportedSymbols = projectLevelSymbolTable.getSymbolsFromModule(fullyQualifiedName);
if (moduleExportedSymbols != null) {
addSymbolAndChildren(fullyQualifiedName, symbolName, names, moduleExportedSymbols);
} else {
Collection standardLibrarySymbols = TypeShed.symbolsForModule(fullyQualifiedName).values();
if (!standardLibrarySymbols.isEmpty()) {
addSymbolAndChildren(fullyQualifiedName, symbolName, names, standardLibrarySymbols);
} else {
// fall back on default mechanism
addModuleSymbol(nameTree, nameTree.name());
return;
}
}
addBindingUsage(nameTree, Usage.Kind.IMPORT, names.get(0));
}
private void addSymbolAndChildren(String fullyQualifiedName, String symbolName, List names, Collection standardLibrarySymbols) {
String parentName = names.get(0);
SymbolImpl parentSymbol = (SymbolImpl) symbolsByName.getOrDefault(symbolName, new SymbolImpl(parentName, parentName));
SymbolImpl currentParent = parentSymbol;
for (int i = 1; i< names.size(); i++) {
String s = names.get(i);
SymbolImpl moduleSymbol = (SymbolImpl) currentParent.getChildrenSymbolByName().get(s);
if (moduleSymbol == null) {
moduleSymbol = new SymbolImpl(s, fullyQualifiedName);
currentParent.addChildSymbol(moduleSymbol);
}
currentParent = moduleSymbol;
}
for (Symbol stdLibSymbol : standardLibrarySymbols) {
if (!currentParent.getChildrenSymbolByName().containsKey(stdLibSymbol.name())) {
currentParent.addChildSymbol(copySymbol(stdLibSymbol.name(), stdLibSymbol));
}
}
this.symbols.add(parentSymbol);
symbolsByName.put(symbolName, parentSymbol);
}
void addImportedSymbol(Name nameTree, @CheckForNull String fullyQualifiedName, String fromModuleName) {
String symbolName = nameTree.name();
Symbol globalSymbol = projectLevelSymbolTable.getSymbol(fullyQualifiedName, symbolName);
if (globalSymbol == null && fullyQualifiedName != null && !fromModuleName.equals(fullyQualifiedModuleName)) {
globalSymbol = TypeShed.symbolWithFQN(fromModuleName, fullyQualifiedName);
globalSymbol = globalSymbol != null ? copySymbol(symbolName, globalSymbol) : null;
}
if (globalSymbol == null || isExistingSymbol(symbolName)) {
addBindingUsage(nameTree, Usage.Kind.IMPORT, globalSymbol != null ? globalSymbol.fullyQualifiedName() : fullyQualifiedName);
} else {
this.symbols.add(globalSymbol);
symbolsByName.put(symbolName, globalSymbol);
((SymbolImpl) globalSymbol).addUsage(nameTree, Usage.Kind.IMPORT);
}
}
private boolean isExistingSymbol(String symbolName) {
return symbolsByName.containsKey(symbolName) || globalNames.contains(symbolName) || nonlocalNames.contains(symbolName);
}
void addBindingUsage(Name nameTree, Usage.Kind kind, @Nullable String fullyQualifiedName) {
String symbolName = nameTree.name();
if (!isExistingSymbol(symbolName)) {
SymbolImpl symbol = new SymbolImpl(symbolName, fullyQualifiedName);
symbols.add(symbol);
symbolsByName.put(symbolName, symbol);
}
SymbolImpl symbol = resolve(symbolName);
if (symbol != null) {
resetSymbolInfo(fullyQualifiedName, symbol);
symbol.addUsage(nameTree, kind);
}
}
private static void resetSymbolInfo(@Nullable String fullyQualifiedName, SymbolImpl symbol) {
if (!Symbol.Kind.OTHER.equals(symbol.kind())) {
symbol.setKind(Symbol.Kind.OTHER);
}
if (fullyQualifiedName != null && !fullyQualifiedName.equals(symbol.fullyQualifiedName)) {
symbol.fullyQualifiedName = null;
}
if (fullyQualifiedName == null && symbol.fullyQualifiedName != null) {
symbol.fullyQualifiedName = null;
}
}
@CheckForNull
SymbolImpl resolve(String symbolName) {
Symbol symbol = symbolsByName.get(symbolName);
if (parent == null || symbol != null) {
return (SymbolImpl) symbol;
}
if (parent.rootTree.is(Tree.Kind.CLASSDEF)) {
return parent.parent.resolve(symbolName);
}
return parent.resolve(symbolName);
}
void addGlobalName(String name) {
globalNames.add(name);
}
void addNonLocalName(String name) {
nonlocalNames.add(name);
}
void addClassSymbol(ClassDef classDef, @Nullable String fullyQualifiedName) {
String symbolName = classDef.name().name();
if (isExistingSymbol(symbolName)) {
addBindingUsage(classDef.name(), Usage.Kind.CLASS_DECLARATION, fullyQualifiedName);
} else {
ClassSymbolImpl classSymbol = new ClassSymbolImpl(classDef, fullyQualifiedName, pythonFile);
symbols.add(classSymbol);
symbolsByName.put(symbolName, classSymbol);
classSymbol.addUsage(classDef.name(), Usage.Kind.CLASS_DECLARATION);
}
}
void replaceSymbolWithAmbiguousSymbol(Symbol symbol, AmbiguousSymbol ambiguousSymbol) {
symbols.remove(symbol);
symbols.add(ambiguousSymbol);
symbolsByName.remove(symbol.name());
symbolsByName.put(symbol.name(), ambiguousSymbol);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy