forked from OSchip/llvm-project
158 lines
4.3 KiB
C++
158 lines
4.3 KiB
C++
//===-- RenameTests.cpp -----------------------------------------*- C++ -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "Annotations.h"
|
|
#include "TestFS.h"
|
|
#include "TestTU.h"
|
|
#include "refactor/Rename.h"
|
|
#include "clang/Tooling/Core/Replacement.h"
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
namespace {
|
|
|
|
MATCHER_P2(RenameRange, Code, Range, "") {
|
|
return replacementToEdit(Code, arg).range == Range;
|
|
}
|
|
|
|
TEST(RenameTest, SingleFile) {
|
|
struct Test {
|
|
const char* Before;
|
|
const char* After;
|
|
} Tests[] = {
|
|
// Rename function.
|
|
{
|
|
R"cpp(
|
|
void foo() {
|
|
fo^o();
|
|
}
|
|
)cpp",
|
|
R"cpp(
|
|
void abcde() {
|
|
abcde();
|
|
}
|
|
)cpp",
|
|
},
|
|
// Rename type.
|
|
{
|
|
R"cpp(
|
|
struct foo{};
|
|
foo test() {
|
|
f^oo x;
|
|
return x;
|
|
}
|
|
)cpp",
|
|
R"cpp(
|
|
struct abcde{};
|
|
abcde test() {
|
|
abcde x;
|
|
return x;
|
|
}
|
|
)cpp",
|
|
},
|
|
// Rename variable.
|
|
{
|
|
R"cpp(
|
|
void bar() {
|
|
if (auto ^foo = 5) {
|
|
foo = 3;
|
|
}
|
|
}
|
|
)cpp",
|
|
R"cpp(
|
|
void bar() {
|
|
if (auto abcde = 5) {
|
|
abcde = 3;
|
|
}
|
|
}
|
|
)cpp",
|
|
},
|
|
};
|
|
for (const Test &T : Tests) {
|
|
Annotations Code(T.Before);
|
|
auto TU = TestTU::withCode(Code.code());
|
|
TU.HeaderCode = "void foo();"; // outside main file, will not be touched.
|
|
auto AST = TU.build();
|
|
auto RenameResult =
|
|
renameWithinFile(AST, testPath(TU.Filename), Code.point(), "abcde");
|
|
ASSERT_TRUE(bool(RenameResult)) << RenameResult.takeError();
|
|
auto ApplyResult =
|
|
tooling::applyAllReplacements(Code.code(), *RenameResult);
|
|
ASSERT_TRUE(bool(ApplyResult)) << ApplyResult.takeError();
|
|
|
|
EXPECT_EQ(T.After, *ApplyResult) << T.Before;
|
|
}
|
|
}
|
|
|
|
TEST(RenameTest, Renameable) {
|
|
// Test cases where the symbol is declared in header.
|
|
const char *Headers[] = {
|
|
R"cpp(// allow -- function-local
|
|
void f(int [[Lo^cal]]) {
|
|
[[Local]] = 2;
|
|
}
|
|
)cpp",
|
|
|
|
R"cpp(// allow -- symbol is indexable and has no refs in index.
|
|
void [[On^lyInThisFile]]();
|
|
)cpp",
|
|
|
|
R"cpp(// disallow -- symbol is indexable and has other refs in index.
|
|
void f() {
|
|
Out^side s;
|
|
}
|
|
)cpp",
|
|
|
|
R"cpp(// disallow -- symbol is not indexable.
|
|
namespace {
|
|
class Unin^dexable {};
|
|
}
|
|
)cpp",
|
|
};
|
|
const char *CommonHeader = "class Outside {};";
|
|
TestTU OtherFile = TestTU::withCode("Outside s;");
|
|
OtherFile.HeaderCode = CommonHeader;
|
|
OtherFile.Filename = "other.cc";
|
|
// The index has a "Outside" reference.
|
|
auto OtherFileIndex = OtherFile.index();
|
|
|
|
for (const char *Header : Headers) {
|
|
Annotations T(Header);
|
|
// We open the .h file as the main file.
|
|
TestTU TU = TestTU::withCode(T.code());
|
|
TU.Filename = "test.h";
|
|
TU.HeaderCode = CommonHeader;
|
|
TU.ExtraArgs.push_back("-xc++");
|
|
auto AST = TU.build();
|
|
|
|
auto Results = renameWithinFile(AST, testPath(TU.Filename), T.point(),
|
|
"dummyNewName", OtherFileIndex.get());
|
|
bool WantRename = true;
|
|
if (T.ranges().empty())
|
|
WantRename = false;
|
|
if (!WantRename) {
|
|
EXPECT_FALSE(Results) << "expected renameWithinFile returned an error: "
|
|
<< T.code();
|
|
llvm::consumeError(Results.takeError());
|
|
} else {
|
|
EXPECT_TRUE(bool(Results)) << "renameWithinFile returned an error: "
|
|
<< llvm::toString(Results.takeError());
|
|
std::vector<testing::Matcher<tooling::Replacement>> Expected;
|
|
for (const auto &R : T.ranges())
|
|
Expected.push_back(RenameRange(TU.Code, R));
|
|
EXPECT_THAT(*Results, UnorderedElementsAreArray(Expected));
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace clangd
|
|
} // namespace clang
|