blob: 12054b622e028172e0ca8c98fa092258b058ab3f [file] [log] [blame]
Haojian Wub7da0eb2017-03-10 10:30:14 +00001//===-- ClangRenameTests.cpp - clang-rename unit tests --------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "RenamingAction.h"
11#include "USRFindingAction.h"
12#include "unittests/Tooling/RewriterTestContext.h"
13#include "clang/ASTMatchers/ASTMatchFinder.h"
14#include "clang/Basic/FileManager.h"
15#include "clang/Basic/FileSystemOptions.h"
16#include "clang/Basic/VirtualFileSystem.h"
17#include "clang/Format/Format.h"
18#include "clang/Frontend/CompilerInstance.h"
19#include "clang/Frontend/PCHContainerOperations.h"
20#include "clang/Tooling/Refactoring.h"
21#include "clang/Tooling/Tooling.h"
22#include "llvm/ADT/IntrusiveRefCntPtr.h"
23#include "llvm/ADT/StringRef.h"
24#include "llvm/Support/Format.h"
25#include "llvm/Support/MemoryBuffer.h"
26#include "gtest/gtest.h"
27#include <memory>
28#include <string>
29#include <vector>
30
31namespace clang {
32namespace clang_rename {
33namespace {
34
35struct Case {
36 std::string Before;
37 std::string After;
38};
39
40class ClangRenameTest : public testing::Test,
41 public testing::WithParamInterface<Case> {
42protected:
43 void AppendToHeader(StringRef Code) {
44 HeaderContent += Code.str();
45 }
46
47 std::string runClangRenameOnCode(llvm::StringRef Code,
48 llvm::StringRef OldName,
49 llvm::StringRef NewName) {
50 std::string NewCode;
51 llvm::raw_string_ostream(NewCode) << llvm::format(
52 "#include \"%s\"\n%s", HeaderName.c_str(), Code.str().c_str());
53 tooling::FileContentMappings FileContents = {{HeaderName, HeaderContent},
54 {CCName, NewCode}};
55 clang::RewriterTestContext Context;
56 Context.createInMemoryFile(HeaderName, HeaderContent);
57 clang::FileID InputFileID = Context.createInMemoryFile(CCName, NewCode);
58
59 rename::USRFindingAction FindingAction({}, {OldName});
60 std::unique_ptr<tooling::FrontendActionFactory> USRFindingActionFactory =
61 tooling::newFrontendActionFactory(&FindingAction);
62
63 if (!tooling::runToolOnCodeWithArgs(
64 USRFindingActionFactory->create(), NewCode, {"-std=c++11"}, CCName,
65 "clang-rename", std::make_shared<PCHContainerOperations>(),
66 FileContents))
67 return "";
68
69 const std::vector<std::vector<std::string>> &USRList =
70 FindingAction.getUSRList();
71 const std::vector<std::string> &PrevNames = FindingAction.getUSRSpellings();
72 std::vector<std::string> NewNames = {NewName};
73 std::map<std::string, tooling::Replacements> FileToReplacements;
74 rename::RenamingAction RenameAction(NewNames, PrevNames, USRList,
75 FileToReplacements);
76 auto RenameActionFactory = tooling::newFrontendActionFactory(&RenameAction);
77 if (!tooling::runToolOnCodeWithArgs(
78 RenameActionFactory->create(), NewCode, {"-std=c++11"}, CCName,
79 "clang-rename", std::make_shared<PCHContainerOperations>(),
80 FileContents))
81 return "";
82
83 formatAndApplyAllReplacements(FileToReplacements, Context.Rewrite, "llvm");
84 return Context.getRewrittenText(InputFileID);
85 }
86
87 void CompareSnippets(StringRef Expected, StringRef Actual) {
88 std::string ExpectedCode;
89 llvm::raw_string_ostream(ExpectedCode) << llvm::format(
90 "#include \"%s\"\n%s", HeaderName.c_str(), Expected.str().c_str());
91 EXPECT_EQ(format(ExpectedCode), format(Actual));
92 }
93
94 std::string format(llvm::StringRef Code) {
95 tooling::Replacements Replaces = format::reformat(
96 format::getLLVMStyle(), Code, {tooling::Range(0, Code.size())});
97 auto ChangedCode = tooling::applyAllReplacements(Code, Replaces);
98 EXPECT_TRUE(static_cast<bool>(ChangedCode));
99 if (!ChangedCode) {
100 llvm::errs() << llvm::toString(ChangedCode.takeError());
101 return "";
102 }
103 return *ChangedCode;
104 }
105
106 std::string HeaderContent;
107 std::string HeaderName = "header.h";
108 std::string CCName = "input.cc";
109};
110
111class RenameClassTest : public ClangRenameTest {
112 public:
113 RenameClassTest() {
114 AppendToHeader("\nclass Foo {};\n");
115 }
116};
117
118INSTANTIATE_TEST_CASE_P(
119 RenameTests, RenameClassTest,
120 testing::ValuesIn(std::vector<Case>({
121 {"Foo f;", "Bar f;"},
122 {"void f(Foo f) {}", "void f(Bar f) {}"},
123 {"void f(Foo *f) {}", "void f(Bar *f) {}"},
124 {"Foo f() { return Foo(); }", "Bar f() { return Bar(); }"},
125 })));
126
127TEST_P(RenameClassTest, RenameClasses) {
128 auto Param = GetParam();
129 std::string OldName = "Foo";
130 std::string NewName = "Bar";
131 std::string Actual = runClangRenameOnCode(Param.Before, OldName, NewName);
132 CompareSnippets(Param.After, Actual);
133}
134
135class RenameFunctionTest : public ClangRenameTest {};
136
137INSTANTIATE_TEST_CASE_P(
138 RenameTests, RenameFunctionTest,
139 testing::ValuesIn(std::vector<Case>({
140 {"void func1() {}", "void func2() {}"},
141 })));
142
143TEST_P(RenameFunctionTest, RenameFunctions) {
144 auto Param = GetParam();
145 std::string OldName = "func1";
146 std::string NewName = "func2";
147 std::string Actual = runClangRenameOnCode(Param.Before, OldName, NewName);
148 CompareSnippets(Param.After, Actual);
149}
150
151} // anonymous namespace
152} // namespace clang_rename
153} // namesdpace clang