1*b804eef0SSam McCall //===-- FileDistanceTests.cpp ------------------------*- C++ -*-----------===//
2*b804eef0SSam McCall //
3*b804eef0SSam McCall // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*b804eef0SSam McCall // See https://llvm.org/LICENSE.txt for license information.
5*b804eef0SSam McCall // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*b804eef0SSam McCall //
7*b804eef0SSam McCall //===----------------------------------------------------------------------===//
8*b804eef0SSam McCall
9*b804eef0SSam McCall #include "FileDistance.h"
10*b804eef0SSam McCall #include "TestFS.h"
11*b804eef0SSam McCall #include "gmock/gmock.h"
12*b804eef0SSam McCall #include "gtest/gtest.h"
13*b804eef0SSam McCall
14*b804eef0SSam McCall namespace clang {
15*b804eef0SSam McCall namespace clangd {
16*b804eef0SSam McCall namespace {
17*b804eef0SSam McCall
TEST(FileDistanceTests,Distance)18*b804eef0SSam McCall TEST(FileDistanceTests, Distance) {
19*b804eef0SSam McCall FileDistanceOptions Opts;
20*b804eef0SSam McCall Opts.UpCost = 5;
21*b804eef0SSam McCall Opts.DownCost = 3;
22*b804eef0SSam McCall SourceParams CostTwo;
23*b804eef0SSam McCall CostTwo.Cost = 2;
24*b804eef0SSam McCall FileDistance D(
25*b804eef0SSam McCall {{"tools/clang/lib/Format/FormatToken.cpp", SourceParams()},
26*b804eef0SSam McCall {"tools/clang/include/clang/Format/FormatToken.h", SourceParams()},
27*b804eef0SSam McCall {"include/llvm/ADT/StringRef.h", CostTwo}},
28*b804eef0SSam McCall Opts);
29*b804eef0SSam McCall
30*b804eef0SSam McCall // Source
31*b804eef0SSam McCall EXPECT_EQ(D.distance("tools/clang/lib/Format/FormatToken.cpp"), 0u);
32*b804eef0SSam McCall EXPECT_EQ(D.distance("include/llvm/ADT/StringRef.h"), 2u);
33*b804eef0SSam McCall // Parent
34*b804eef0SSam McCall EXPECT_EQ(D.distance("tools/clang/lib/Format/"), 5u);
35*b804eef0SSam McCall // Child
36*b804eef0SSam McCall EXPECT_EQ(D.distance("tools/clang/lib/Format/FormatToken.cpp/Oops"), 3u);
37*b804eef0SSam McCall // Ancestor (up+up+up+up)
38*b804eef0SSam McCall EXPECT_EQ(D.distance("/"), 22u);
39*b804eef0SSam McCall // Sibling (up+down)
40*b804eef0SSam McCall EXPECT_EQ(D.distance("tools/clang/lib/Format/AnotherFile.cpp"), 8u);
41*b804eef0SSam McCall // Cousin (up+up+down+down)
42*b804eef0SSam McCall EXPECT_EQ(D.distance("include/llvm/Support/Allocator.h"), 18u);
43*b804eef0SSam McCall // First cousin, once removed (up+up+up+down+down)
44*b804eef0SSam McCall EXPECT_EQ(D.distance("include/llvm-c/Core.h"), 23u);
45*b804eef0SSam McCall }
46*b804eef0SSam McCall
TEST(FileDistanceTests,BadSource)47*b804eef0SSam McCall TEST(FileDistanceTests, BadSource) {
48*b804eef0SSam McCall // We mustn't assume that paths above sources are best reached via them.
49*b804eef0SSam McCall FileDistanceOptions Opts;
50*b804eef0SSam McCall Opts.UpCost = 5;
51*b804eef0SSam McCall Opts.DownCost = 3;
52*b804eef0SSam McCall SourceParams CostLots;
53*b804eef0SSam McCall CostLots.Cost = 100;
54*b804eef0SSam McCall FileDistance D({{"a", SourceParams()}, {"b/b/b", CostLots}}, Opts);
55*b804eef0SSam McCall EXPECT_EQ(D.distance("b"), 8u); // a+up+down, not b+up+up
56*b804eef0SSam McCall EXPECT_EQ(D.distance("b/b/b"), 14u); // a+up+down+down+down, not b
57*b804eef0SSam McCall EXPECT_EQ(D.distance("b/b/b/c"), 17u); // a+up+down+down+down+down, not b+down
58*b804eef0SSam McCall }
59*b804eef0SSam McCall
60*b804eef0SSam McCall // Force the unittest URI scheme to be linked,
61*b804eef0SSam McCall static int LLVM_ATTRIBUTE_UNUSED UseUnittestScheme = UnittestSchemeAnchorSource;
62*b804eef0SSam McCall
TEST(FileDistanceTests,URI)63*b804eef0SSam McCall TEST(FileDistanceTests, URI) {
64*b804eef0SSam McCall FileDistanceOptions Opts;
65*b804eef0SSam McCall Opts.UpCost = 5;
66*b804eef0SSam McCall Opts.DownCost = 3;
67*b804eef0SSam McCall SourceParams CostLots;
68*b804eef0SSam McCall CostLots.Cost = 1000;
69*b804eef0SSam McCall
70*b804eef0SSam McCall URIDistance D({{testPath("foo"), CostLots},
71*b804eef0SSam McCall {"/not/a/testpath", SourceParams()},
72*b804eef0SSam McCall {"C:\\not\\a\\testpath", SourceParams()}},
73*b804eef0SSam McCall Opts);
74*b804eef0SSam McCall #ifdef _WIN32
75*b804eef0SSam McCall EXPECT_EQ(D.distance("file:///C%3a/not/a/testpath/either"), 3u);
76*b804eef0SSam McCall #else
77*b804eef0SSam McCall EXPECT_EQ(D.distance("file:///not/a/testpath/either"), 3u);
78*b804eef0SSam McCall #endif
79*b804eef0SSam McCall EXPECT_EQ(D.distance("unittest:///foo"), 1000u);
80*b804eef0SSam McCall EXPECT_EQ(D.distance("unittest:///bar"), 1008u);
81*b804eef0SSam McCall }
82*b804eef0SSam McCall
TEST(FileDistance,LimitUpTraversals)83*b804eef0SSam McCall TEST(FileDistance, LimitUpTraversals) {
84*b804eef0SSam McCall FileDistanceOptions Opts;
85*b804eef0SSam McCall Opts.UpCost = Opts.DownCost = 1;
86*b804eef0SSam McCall SourceParams CheapButLimited, CostLots;
87*b804eef0SSam McCall CheapButLimited.MaxUpTraversals = 1;
88*b804eef0SSam McCall CostLots.Cost = 100;
89*b804eef0SSam McCall
90*b804eef0SSam McCall FileDistance D({{"/", CostLots}, {"/a/b/c", CheapButLimited}}, Opts);
91*b804eef0SSam McCall EXPECT_EQ(D.distance("/a"), 101u);
92*b804eef0SSam McCall EXPECT_EQ(D.distance("/a/z"), 102u);
93*b804eef0SSam McCall EXPECT_EQ(D.distance("/a/b"), 1u);
94*b804eef0SSam McCall EXPECT_EQ(D.distance("/a/b/z"), 2u);
95*b804eef0SSam McCall }
96*b804eef0SSam McCall
TEST(FileDistance,DisallowDownTraversalsFromRoot)97*b804eef0SSam McCall TEST(FileDistance, DisallowDownTraversalsFromRoot) {
98*b804eef0SSam McCall FileDistanceOptions Opts;
99*b804eef0SSam McCall Opts.UpCost = Opts.DownCost = 1;
100*b804eef0SSam McCall Opts.AllowDownTraversalFromRoot = false;
101*b804eef0SSam McCall SourceParams CostLots;
102*b804eef0SSam McCall CostLots.Cost = 100;
103*b804eef0SSam McCall
104*b804eef0SSam McCall FileDistance D({{"/", SourceParams()}, {"/a/b/c", CostLots}}, Opts);
105*b804eef0SSam McCall EXPECT_EQ(D.distance("/"), 0u);
106*b804eef0SSam McCall EXPECT_EQ(D.distance("/a"), 102u);
107*b804eef0SSam McCall EXPECT_EQ(D.distance("/a/b"), 101u);
108*b804eef0SSam McCall EXPECT_EQ(D.distance("/x"), FileDistance::Unreachable);
109*b804eef0SSam McCall }
110*b804eef0SSam McCall
TEST(ScopeDistance,Smoke)111*b804eef0SSam McCall TEST(ScopeDistance, Smoke) {
112*b804eef0SSam McCall ScopeDistance D({"x::y::z", "x::", "", "a::"});
113*b804eef0SSam McCall EXPECT_EQ(D.distance("x::y::z::"), 0u);
114*b804eef0SSam McCall EXPECT_GT(D.distance("x::y::"), D.distance("x::y::z::"));
115*b804eef0SSam McCall EXPECT_GT(D.distance("x::"), D.distance("x::y::"));
116*b804eef0SSam McCall EXPECT_GT(D.distance("x::y::z::down::"), D.distance("x::y::"));
117*b804eef0SSam McCall EXPECT_GT(D.distance(""), D.distance("a::"));
118*b804eef0SSam McCall EXPECT_GT(D.distance("x::"), D.distance("a::"));
119*b804eef0SSam McCall }
120*b804eef0SSam McCall
121*b804eef0SSam McCall } // namespace
122*b804eef0SSam McCall } // namespace clangd
123*b804eef0SSam McCall } // namespace clang
124