blob: c7592197e1e4a95eb02bf7b928c405b1d278a4a2 [file] [log] [blame]
// Copyright 2017 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.vfs;
import static com.google.common.truth.Truth.assertThat;
import com.google.common.testing.EqualsTester;
import com.google.devtools.build.lib.testutil.MoreAsserts;
import com.google.devtools.build.lib.vfs.WindowsOsPathPolicy.ShortPathResolver;
import java.util.HashMap;
import java.util.Map;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/** Tests windows-specific parts of {@link Path} */
@RunWith(JUnit4.class)
public class WindowsPathTest extends PathAbstractTest {
private static final class MockShortPathResolver implements ShortPathResolver {
// Full path to resolved child mapping.
private Map<String, String> resolutions = new HashMap<>();
@Override
public String resolveShortPath(String path) {
String[] segments = path.split("[\\\\/]+");
String result = "";
for (int i = 0; i < segments.length; ) {
String segment = segments[i];
String queryString = (result + segment).toLowerCase();
segment = resolutions.getOrDefault(queryString, segment);
result = result + segment;
++i;
if (i != segments.length) {
result += "/";
}
}
return result;
}
}
@Test
public void testEqualsAndHashcodeWindows() {
new EqualsTester()
.addEqualityGroup(create("/a/b"), create("/A/B"))
.addEqualityGroup(create("c:/a/b"), create("C:\\A\\B"))
.addEqualityGroup(create("C:/something/else"))
.testEquals();
}
@Test
public void testCaseIsPreserved() {
assertThat(create("C:/a/B").getPathString()).isEqualTo("C:/a/B");
}
@Test
public void testNormalizeWindows() {
assertThat(create("C:/")).isEqualTo(create("C:/"));
assertThat(create("c:/")).isEqualTo(create("C:/"));
assertThat(create("c:\\")).isEqualTo(create("C:/"));
assertThat(create("c:\\foo\\..\\bar\\")).isEqualTo(create("C:/bar"));
}
@Test
public void testStartsWithWindows() {
assertThat(create("C:/").startsWith(create("C:/"))).isTrue();
assertThat(create("C:/foo").startsWith(create("C:/"))).isTrue();
assertThat(create("C:/foo").startsWith(create("D:/"))).isFalse();
// Case insensitivity test
assertThat(create("C:/foo/bar").startsWith(create("C:/FOO"))).isTrue();
}
@Test
public void testGetParentDirectoryWindows() {
assertThat(create("C:/foo").getParentDirectory()).isEqualTo(create("C:/"));
assertThat(create("C:/").getParentDirectory()).isNull();
assertThat(create("/").getParentDirectory()).isNull();
}
@Test
public void testParentOfRootIsRootWindows() {
assertThat(create("C:/..")).isEqualTo(create("C:/"));
assertThat(create("C:/../../../../../..")).isEqualTo(create("C:/"));
assertThat(create("C:/../../../foo")).isEqualTo(create("C:/foo"));
}
@Test
public void testRelativeToWindows() {
assertThat(create("C:/foo").relativeTo(create("C:/")).getPathString()).isEqualTo("foo");
// Case insensitivity test
assertThat(create("C:/foo/bar").relativeTo(create("C:/FOO")).getPathString()).isEqualTo("bar");
MoreAsserts.assertThrows(
IllegalArgumentException.class, () -> create("D:/foo").relativeTo(create("C:/")));
}
@Test
public void testResolvesShortenedPaths() {
MockShortPathResolver shortPathResolver = new MockShortPathResolver();
WindowsOsPathPolicy osPathPolicy = new WindowsOsPathPolicy(shortPathResolver);
shortPathResolver.resolutions.put("d:/progra~1", "program files");
shortPathResolver.resolutions.put("d:/program files/micros~1", "microsoft something");
shortPathResolver.resolutions.put(
"d:/program files/microsoft something/foo/~bar~1", "~bar_hello");
// Assert normal shortpath resolution.
assertThat(normalize(osPathPolicy, "d:/progra~1/micros~1/foo/~bar~1/baz"))
.isEqualTo("D:/program files/microsoft something/foo/~bar_hello/baz");
assertThat(normalize(osPathPolicy, "d:/progra~1/micros~1/foo/will~1.exi/bar"))
.isEqualTo("D:/program files/microsoft something/foo/will~1.exi/bar");
assertThat(normalize(osPathPolicy, "d:/progra~1/micros~1"))
.isEqualTo("D:/program files/microsoft something");
// Pretend that a path we already failed to resolve once came into existence.
shortPathResolver.resolutions.put(
"d:/program files/microsoft something/foo/will~1.exi", "will.exist");
// Assert that this time we can resolve the previously non-existent path.
// The path string has an upper-case drive letter because that's how path printing works.
assertThat(normalize(osPathPolicy, "d:/progra~1/micros~1/foo/will~1.exi/bar"))
.isEqualTo("D:/program files/microsoft something/foo/will.exist/bar");
// Check needsToNormalized
assertThat(osPathPolicy.needsToNormalize("d:/progra~1/micros~1/foo/will~1.exi/bar"))
.isEqualTo(WindowsOsPathPolicy.NEEDS_SHORT_PATH_NORMALIZATION);
assertThat(osPathPolicy.needsToNormalize("will~1.exi"))
.isEqualTo(WindowsOsPathPolicy.NEEDS_SHORT_PATH_NORMALIZATION);
assertThat(osPathPolicy.needsToNormalize("d:/no-normalization"))
.isEqualTo(WindowsOsPathPolicy.NORMALIZED); // Sanity check
}
private static String normalize(OsPathPolicy osPathPolicy, String str) {
return osPathPolicy.normalize(str, osPathPolicy.needsToNormalize(str));
}
}