mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
317 lines
11 KiB
C++
317 lines
11 KiB
C++
|
/*
|
||
|
This file is part of solidity.
|
||
|
|
||
|
solidity is free software: you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation, either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
solidity 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 General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
// SPDX-License-Identifier: GPL-3.0
|
||
|
#include <libsolidity/lsp/RenameSymbol.h>
|
||
|
#include <libsolidity/lsp/Utils.h>
|
||
|
|
||
|
#include <libyul/AST.h>
|
||
|
|
||
|
#include <fmt/format.h>
|
||
|
|
||
|
#include <memory>
|
||
|
#include <string>
|
||
|
#include <vector>
|
||
|
|
||
|
using namespace solidity::frontend;
|
||
|
using namespace solidity::langutil;
|
||
|
using namespace solidity::lsp;
|
||
|
using namespace std;
|
||
|
|
||
|
namespace
|
||
|
{
|
||
|
|
||
|
CallableDeclaration const* extractCallableDeclaration(FunctionCall const& _functionCall)
|
||
|
{
|
||
|
if (
|
||
|
auto const* functionType = dynamic_cast<FunctionType const*>(_functionCall.expression().annotation().type);
|
||
|
functionType && functionType->hasDeclaration()
|
||
|
)
|
||
|
if (auto const* functionDefinition = dynamic_cast<FunctionDefinition const*>(&functionType->declaration()))
|
||
|
return functionDefinition;
|
||
|
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::operator()(MessageID _id, Json::Value const& _args)
|
||
|
{
|
||
|
auto const&& [sourceUnitName, lineColumn] = extractSourceUnitNameAndLineColumn(_args);
|
||
|
string const newName = _args["newName"].asString();
|
||
|
string const uri = _args["textDocument"]["uri"].asString();
|
||
|
|
||
|
ASTNode const* sourceNode = m_server.astNodeAtSourceLocation(sourceUnitName, lineColumn);
|
||
|
|
||
|
m_symbolName = {};
|
||
|
m_declarationToRename = nullptr;
|
||
|
m_sourceUnits = { &m_server.compilerStack().ast(sourceUnitName) };
|
||
|
m_locations.clear();
|
||
|
|
||
|
optional<int> cursorBytePosition = charStreamProvider()
|
||
|
.charStream(sourceUnitName)
|
||
|
.translateLineColumnToPosition(lineColumn);
|
||
|
solAssert(cursorBytePosition.has_value(), "Expected source pos");
|
||
|
|
||
|
extractNameAndDeclaration(*sourceNode, *cursorBytePosition);
|
||
|
|
||
|
// Find all source units using this symbol
|
||
|
for (auto const& [name, content]: fileRepository().sourceUnits())
|
||
|
{
|
||
|
auto const& sourceUnit = m_server.compilerStack().ast(name);
|
||
|
for (auto const* referencedSourceUnit: sourceUnit.referencedSourceUnits(true, util::convertContainer<set<SourceUnit const*>>(m_sourceUnits)))
|
||
|
if (*referencedSourceUnit->location().sourceName == sourceUnitName)
|
||
|
{
|
||
|
m_sourceUnits.insert(&sourceUnit);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Origin source unit should always be checked
|
||
|
m_sourceUnits.insert(&m_declarationToRename->sourceUnit());
|
||
|
|
||
|
Visitor visitor(*this);
|
||
|
|
||
|
for (auto const* sourceUnit: m_sourceUnits)
|
||
|
sourceUnit->accept(visitor);
|
||
|
|
||
|
// Apply changes in reverse order (will iterate in reverse)
|
||
|
sort(m_locations.begin(), m_locations.end());
|
||
|
|
||
|
Json::Value reply = Json::objectValue;
|
||
|
reply["changes"] = Json::objectValue;
|
||
|
|
||
|
Json::Value edits = Json::arrayValue;
|
||
|
|
||
|
for (auto i = m_locations.rbegin(); i != m_locations.rend(); i++)
|
||
|
{
|
||
|
solAssert(i->isValid());
|
||
|
|
||
|
// Replace in our file repository
|
||
|
string const uri = fileRepository().sourceUnitNameToUri(*i->sourceName);
|
||
|
string buffer = fileRepository().sourceUnits().at(*i->sourceName);
|
||
|
buffer.replace((size_t)i->start, (size_t)(i->end - i->start), newName);
|
||
|
fileRepository().setSourceByUri(uri, std::move(buffer));
|
||
|
|
||
|
Json::Value edit = Json::objectValue;
|
||
|
edit["range"] = toRange(*i);
|
||
|
edit["newText"] = newName;
|
||
|
|
||
|
// Record changes for the client
|
||
|
edits.append(edit);
|
||
|
if (i + 1 == m_locations.rend() || (i + 1)->sourceName != i->sourceName)
|
||
|
{
|
||
|
reply["changes"][uri] = edits;
|
||
|
edits = Json::arrayValue;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
client().reply(_id, reply);
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::extractNameAndDeclaration(ASTNode const& _node, int _cursorBytePosition)
|
||
|
{
|
||
|
// Identify symbol name and node
|
||
|
if (auto const* declaration = dynamic_cast<Declaration const*>(&_node))
|
||
|
{
|
||
|
if (declaration->nameLocation().containsOffset(_cursorBytePosition))
|
||
|
{
|
||
|
m_symbolName = declaration->name();
|
||
|
m_declarationToRename = declaration;
|
||
|
}
|
||
|
else if (auto const* importDirective = dynamic_cast<ImportDirective const*>(declaration))
|
||
|
extractNameAndDeclaration(*importDirective, _cursorBytePosition);
|
||
|
}
|
||
|
else if (auto const* identifier = dynamic_cast<Identifier const*>(&_node))
|
||
|
{
|
||
|
if (auto const* declReference = dynamic_cast<Declaration const*>(identifier->annotation().referencedDeclaration))
|
||
|
{
|
||
|
m_symbolName = identifier->name();
|
||
|
m_declarationToRename = declReference;
|
||
|
}
|
||
|
}
|
||
|
else if (auto const* identifierPath = dynamic_cast<IdentifierPath const*>(&_node))
|
||
|
extractNameAndDeclaration(*identifierPath, _cursorBytePosition);
|
||
|
else if (auto const* memberAccess = dynamic_cast<MemberAccess const*>(&_node))
|
||
|
{
|
||
|
m_symbolName = memberAccess->memberName();
|
||
|
m_declarationToRename = memberAccess->annotation().referencedDeclaration;
|
||
|
}
|
||
|
else if (auto const* functionCall = dynamic_cast<FunctionCall const*>(&_node))
|
||
|
extractNameAndDeclaration(*functionCall, _cursorBytePosition);
|
||
|
else if (auto const* inlineAssembly = dynamic_cast<InlineAssembly const*>(&_node))
|
||
|
extractNameAndDeclaration(*inlineAssembly, _cursorBytePosition);
|
||
|
else
|
||
|
solAssert(false, "Unexpected ASTNODE id: " + to_string(_node.id()));
|
||
|
|
||
|
lspDebug(fmt::format("Goal: rename '{}', loc: {}-{}", m_symbolName, m_declarationToRename->nameLocation().start, m_declarationToRename->nameLocation().end));
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::extractNameAndDeclaration(ImportDirective const& _importDirective, int _cursorBytePosition)
|
||
|
{
|
||
|
for (ImportDirective::SymbolAlias const& symbolAlias: _importDirective.symbolAliases())
|
||
|
if (symbolAlias.location.containsOffset(_cursorBytePosition))
|
||
|
{
|
||
|
solAssert(symbolAlias.alias);
|
||
|
m_symbolName = *symbolAlias.alias;
|
||
|
m_declarationToRename = symbolAlias.symbol->annotation().referencedDeclaration;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(ImportDirective const& _node)
|
||
|
{
|
||
|
// Handles SourceUnit aliases
|
||
|
if (handleGenericDeclaration(_node))
|
||
|
return;
|
||
|
|
||
|
for (ImportDirective::SymbolAlias const& symbolAlias: _node.symbolAliases())
|
||
|
if (
|
||
|
symbolAlias.alias != nullptr &&
|
||
|
*symbolAlias.alias == m_outer.m_symbolName &&
|
||
|
symbolAlias.symbol->annotation().referencedDeclaration == m_outer.m_declarationToRename
|
||
|
)
|
||
|
m_outer.m_locations.emplace_back(symbolAlias.location);
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::extractNameAndDeclaration(FunctionCall const& _functionCall, int _cursorBytePosition)
|
||
|
{
|
||
|
if (auto const* functionDefinition = extractCallableDeclaration(_functionCall))
|
||
|
for (size_t i = 0; i < _functionCall.names().size(); i++)
|
||
|
if (_functionCall.nameLocations()[i].containsOffset(_cursorBytePosition))
|
||
|
{
|
||
|
m_symbolName = *_functionCall.names()[i];
|
||
|
for (size_t j = 0; j < functionDefinition->parameters().size(); j++)
|
||
|
if (
|
||
|
functionDefinition->parameters()[j] &&
|
||
|
functionDefinition->parameters()[j]->name() == m_symbolName
|
||
|
)
|
||
|
m_declarationToRename = functionDefinition->parameters()[j].get();
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(FunctionCall const& _node)
|
||
|
{
|
||
|
SourceLocation nameLocationInFunctionCall;
|
||
|
|
||
|
for (size_t i = 0; i < _node.names().size(); i++)
|
||
|
if (_node.names()[i] && *_node.names()[i] == m_outer.m_symbolName)
|
||
|
nameLocationInFunctionCall = _node.nameLocations()[i];
|
||
|
|
||
|
if (!nameLocationInFunctionCall.isValid())
|
||
|
return;
|
||
|
|
||
|
if (auto const* functionDefinition = extractCallableDeclaration(_node))
|
||
|
for (size_t j = 0; j < functionDefinition->parameters().size(); j++)
|
||
|
if (
|
||
|
functionDefinition->parameters()[j] &&
|
||
|
*functionDefinition->parameters()[j] == *m_outer.m_declarationToRename
|
||
|
)
|
||
|
m_outer.m_locations.emplace_back(nameLocationInFunctionCall);
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(MemberAccess const& _node)
|
||
|
{
|
||
|
if (
|
||
|
m_outer.m_symbolName == _node.memberName() &&
|
||
|
*m_outer.m_declarationToRename == *_node.annotation().referencedDeclaration
|
||
|
)
|
||
|
m_outer.m_locations.emplace_back(_node.memberLocation());
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(Identifier const& _node)
|
||
|
{
|
||
|
if (
|
||
|
m_outer.m_symbolName == _node.name() &&
|
||
|
*m_outer.m_declarationToRename == *_node.annotation().referencedDeclaration
|
||
|
)
|
||
|
m_outer.m_locations.emplace_back(_node.location());
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::extractNameAndDeclaration(IdentifierPath const& _identifierPath, int _cursorBytePosition)
|
||
|
{
|
||
|
// iterate through the elements of the path to find the one the cursor is on
|
||
|
size_t numIdentifiers = _identifierPath.pathLocations().size();
|
||
|
for (size_t i = 0; i < numIdentifiers; i++)
|
||
|
{
|
||
|
auto& location = _identifierPath.pathLocations()[i];
|
||
|
|
||
|
if (location.containsOffset(_cursorBytePosition))
|
||
|
{
|
||
|
solAssert(_identifierPath.annotation().pathDeclarations.size() == numIdentifiers);
|
||
|
solAssert(_identifierPath.path().size() == numIdentifiers);
|
||
|
|
||
|
m_declarationToRename = _identifierPath.annotation().pathDeclarations[i];
|
||
|
m_symbolName = _identifierPath.path()[i];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(IdentifierPath const& _node)
|
||
|
{
|
||
|
std::vector<Declaration const*>& declarations = _node.annotation().pathDeclarations;
|
||
|
solAssert(declarations.size() == _node.path().size());
|
||
|
|
||
|
for (size_t i = 0; i < _node.path().size(); i++)
|
||
|
if (
|
||
|
_node.path()[i] == m_outer.m_symbolName &&
|
||
|
declarations[i] == m_outer.m_declarationToRename
|
||
|
)
|
||
|
m_outer.m_locations.emplace_back(_node.pathLocations()[i]);
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::extractNameAndDeclaration(InlineAssembly const& _inlineAssembly, int _cursorBytePosition)
|
||
|
{
|
||
|
for (auto&& [identifier, externalReference]: _inlineAssembly.annotation().externalReferences)
|
||
|
{
|
||
|
SourceLocation location = yul::nativeLocationOf(*identifier);
|
||
|
location.end -= static_cast<int>(externalReference.suffix.size() + 1);
|
||
|
|
||
|
if (location.containsOffset(_cursorBytePosition))
|
||
|
{
|
||
|
m_declarationToRename = externalReference.declaration;
|
||
|
m_symbolName = identifier->name.str();
|
||
|
|
||
|
if (!externalReference.suffix.empty())
|
||
|
m_symbolName = m_symbolName.substr(0, m_symbolName.length() - externalReference.suffix.size() - 1);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RenameSymbol::Visitor::endVisit(InlineAssembly const& _node)
|
||
|
{
|
||
|
for (auto&& [identifier, externalReference]: _node.annotation().externalReferences)
|
||
|
{
|
||
|
string identifierName = identifier->name.str();
|
||
|
if (!externalReference.suffix.empty())
|
||
|
identifierName = identifierName.substr(0, identifierName.length() - externalReference.suffix.size() - 1);
|
||
|
|
||
|
if (
|
||
|
externalReference.declaration == m_outer.m_declarationToRename &&
|
||
|
identifierName == m_outer.m_symbolName
|
||
|
)
|
||
|
{
|
||
|
SourceLocation location = yul::nativeLocationOf(*identifier);
|
||
|
location.end -= static_cast<int>(externalReference.suffix.size() + 1);
|
||
|
|
||
|
m_outer.m_locations.emplace_back(location);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|