xref: /llvm-project/clang-tools-extra/clang-tidy/objc/AssertEquals.cpp (revision 7d2ea6c422d3f5712b7253407005e1a465a76946)
1aba8f320SVy Nguyen //===--- AssertEquals.cpp - clang-tidy --------------------------*- C++ -*-===//
2aba8f320SVy Nguyen //
3aba8f320SVy Nguyen // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4aba8f320SVy Nguyen // See https://llvm.org/LICENSE.txt for license information.
5aba8f320SVy Nguyen // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6aba8f320SVy Nguyen //
7aba8f320SVy Nguyen //===----------------------------------------------------------------------===//
8aba8f320SVy Nguyen 
9aba8f320SVy Nguyen #include "AssertEquals.h"
10aba8f320SVy Nguyen 
11aba8f320SVy Nguyen #include <map>
12aba8f320SVy Nguyen #include <string>
13aba8f320SVy Nguyen 
14aba8f320SVy Nguyen using namespace clang::ast_matchers;
15aba8f320SVy Nguyen 
16*7d2ea6c4SCarlos Galvez namespace clang::tidy::objc {
17aba8f320SVy Nguyen 
18aba8f320SVy Nguyen // Mapping from `XCTAssert*Equal` to `XCTAssert*EqualObjects` name.
NameMap()19aba8f320SVy Nguyen static const std::map<std::string, std::string> &NameMap() {
20aba8f320SVy Nguyen   static std::map<std::string, std::string> map{
21aba8f320SVy Nguyen       {"XCTAssertEqual", "XCTAssertEqualObjects"},
22aba8f320SVy Nguyen       {"XCTAssertNotEqual", "XCTAssertNotEqualObjects"},
23aba8f320SVy Nguyen 
24aba8f320SVy Nguyen   };
25aba8f320SVy Nguyen   return map;
26aba8f320SVy Nguyen }
27aba8f320SVy Nguyen 
registerMatchers(MatchFinder * finder)28aba8f320SVy Nguyen void AssertEquals::registerMatchers(MatchFinder *finder) {
29aba8f320SVy Nguyen   for (const auto &pair : NameMap()) {
30aba8f320SVy Nguyen     finder->addMatcher(
31aba8f320SVy Nguyen         binaryOperator(anyOf(hasOperatorName("!="), hasOperatorName("==")),
32aba8f320SVy Nguyen                        isExpandedFromMacro(pair.first),
33aba8f320SVy Nguyen                        anyOf(hasLHS(hasType(qualType(
34aba8f320SVy Nguyen                                  hasCanonicalType(asString("NSString *"))))),
35aba8f320SVy Nguyen                              hasRHS(hasType(qualType(
36aba8f320SVy Nguyen                                  hasCanonicalType(asString("NSString *"))))))
37aba8f320SVy Nguyen 
38aba8f320SVy Nguyen                            )
39aba8f320SVy Nguyen             .bind(pair.first),
40aba8f320SVy Nguyen         this);
41aba8f320SVy Nguyen   }
42aba8f320SVy Nguyen }
43aba8f320SVy Nguyen 
check(const ast_matchers::MatchFinder::MatchResult & result)44aba8f320SVy Nguyen void AssertEquals::check(const ast_matchers::MatchFinder::MatchResult &result) {
45aba8f320SVy Nguyen   for (const auto &pair : NameMap()) {
46aba8f320SVy Nguyen     if (const auto *root = result.Nodes.getNodeAs<BinaryOperator>(pair.first)) {
47aba8f320SVy Nguyen       SourceManager *sm = result.SourceManager;
48aba8f320SVy Nguyen       // The macros are nested two levels, so going up twice.
49aba8f320SVy Nguyen       auto macro_callsite = sm->getImmediateMacroCallerLoc(
50aba8f320SVy Nguyen           sm->getImmediateMacroCallerLoc(root->getBeginLoc()));
51aba8f320SVy Nguyen       diag(macro_callsite, "use " + pair.second + " for comparing objects")
52aba8f320SVy Nguyen           << FixItHint::CreateReplacement(
53aba8f320SVy Nguyen                  clang::CharSourceRange::getCharRange(
54aba8f320SVy Nguyen                      macro_callsite,
55aba8f320SVy Nguyen                      macro_callsite.getLocWithOffset(pair.first.length())),
56aba8f320SVy Nguyen                  pair.second);
57aba8f320SVy Nguyen     }
58aba8f320SVy Nguyen   }
59aba8f320SVy Nguyen }
60aba8f320SVy Nguyen 
61*7d2ea6c4SCarlos Galvez } // namespace clang::tidy::objc
62