|  | // 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 com.google.common.annotations.VisibleForTesting; | 
|  | import com.google.common.base.Splitter; | 
|  | import com.google.common.collect.Iterables; | 
|  | import com.google.devtools.build.lib.windows.WindowsFileOperations; | 
|  | import com.google.devtools.build.lib.windows.WindowsShortPath; | 
|  | import java.io.IOException; | 
|  |  | 
|  | @VisibleForTesting | 
|  | class WindowsOsPathPolicy implements OsPathPolicy { | 
|  |  | 
|  | static final WindowsOsPathPolicy INSTANCE = new WindowsOsPathPolicy(); | 
|  |  | 
|  | static final int NEEDS_SHORT_PATH_NORMALIZATION = NEEDS_NORMALIZE + 1; | 
|  |  | 
|  | private static final Splitter WINDOWS_PATH_SPLITTER = | 
|  | Splitter.onPattern("[\\\\/]+").omitEmptyStrings(); | 
|  |  | 
|  | private final ShortPathResolver shortPathResolver; | 
|  |  | 
|  | interface ShortPathResolver { | 
|  | String resolveShortPath(String path); | 
|  | } | 
|  |  | 
|  | static class DefaultShortPathResolver implements ShortPathResolver { | 
|  | @Override | 
|  | public String resolveShortPath(String path) { | 
|  | try { | 
|  | return WindowsFileOperations.getLongPath(path); | 
|  | } catch (IOException e) { | 
|  | return path; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | WindowsOsPathPolicy() { | 
|  | this(new DefaultShortPathResolver()); | 
|  | } | 
|  |  | 
|  | WindowsOsPathPolicy(ShortPathResolver shortPathResolver) { | 
|  | this.shortPathResolver = shortPathResolver; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int needsToNormalize(String path) { | 
|  | int n = path.length(); | 
|  | int normalizationLevel = NORMALIZED; | 
|  | int dotCount = 0; | 
|  | char prevChar = 0; | 
|  | int segmentBeginIndex = 0; // The start index of the current path index | 
|  | boolean segmentHasShortPathChar = false; // Triggers more expensive short path regex test | 
|  | for (int i = 0; i < n; i++) { | 
|  | char c = path.charAt(i); | 
|  | if (isSeparator(c)) { | 
|  | if (c == '\\') { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_NORMALIZE); | 
|  | } | 
|  | // No need to check for '\\' here because that already causes normalization | 
|  | if (prevChar == '/') { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_NORMALIZE); | 
|  | } | 
|  | if (dotCount == 1 || dotCount == 2) { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_NORMALIZE); | 
|  | } | 
|  | if (segmentHasShortPathChar) { | 
|  | if (WindowsShortPath.isShortPath(path.substring(segmentBeginIndex, i))) { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_SHORT_PATH_NORMALIZATION); | 
|  | } | 
|  | } | 
|  | segmentBeginIndex = i + 1; | 
|  | segmentHasShortPathChar = false; | 
|  | } else if (c == '~') { | 
|  | // This path segment might be a Windows short path segment | 
|  | segmentHasShortPathChar = true; | 
|  | } | 
|  | dotCount = c == '.' ? dotCount + 1 : 0; | 
|  | prevChar = c; | 
|  | } | 
|  | if (segmentHasShortPathChar) { | 
|  | if (WindowsShortPath.isShortPath(path.substring(segmentBeginIndex))) { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_SHORT_PATH_NORMALIZATION); | 
|  | } | 
|  | } | 
|  | if ((n > 1 && isSeparator(prevChar)) || dotCount == 1 || dotCount == 2) { | 
|  | normalizationLevel = Math.max(normalizationLevel, NEEDS_NORMALIZE); | 
|  | } | 
|  | return normalizationLevel; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int needsToNormalizeSuffix(String normalizedSuffix) { | 
|  | // On Windows, all bets are off because of short paths, so we have to check the entire string | 
|  | return needsToNormalize(normalizedSuffix); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String normalize(String path, int normalizationLevel) { | 
|  | if (normalizationLevel == NORMALIZED) { | 
|  | return path; | 
|  | } | 
|  | if (normalizationLevel == NEEDS_SHORT_PATH_NORMALIZATION) { | 
|  | String resolvedPath = shortPathResolver.resolveShortPath(path); | 
|  | if (resolvedPath != null) { | 
|  | path = resolvedPath; | 
|  | } | 
|  | } | 
|  | String[] segments = Iterables.toArray(WINDOWS_PATH_SPLITTER.splitToList(path), String.class); | 
|  | int driveStrLength = getDriveStrLength(path); | 
|  | boolean isAbsolute = driveStrLength > 0; | 
|  | int segmentSkipCount = isAbsolute && driveStrLength > 1 ? 1 : 0; | 
|  |  | 
|  | StringBuilder sb = new StringBuilder(path.length()); | 
|  | if (isAbsolute) { | 
|  | char c = path.charAt(0); | 
|  | if (isSeparator(c)) { | 
|  | sb.append('/'); | 
|  | } else { | 
|  | sb.append(Character.toUpperCase(c)); | 
|  | sb.append(":/"); | 
|  | } | 
|  | } | 
|  | int segmentCount = Utils.removeRelativePaths(segments, segmentSkipCount, isAbsolute); | 
|  | for (int i = 0; i < segmentCount; ++i) { | 
|  | sb.append(segments[i]); | 
|  | sb.append('/'); | 
|  | } | 
|  | if (segmentCount > 0) { | 
|  | sb.deleteCharAt(sb.length() - 1); | 
|  | } | 
|  | return sb.toString(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int getDriveStrLength(String path) { | 
|  | int n = path.length(); | 
|  | if (n == 0) { | 
|  | return 0; | 
|  | } | 
|  | char c0 = path.charAt(0); | 
|  | if (isSeparator(c0)) { | 
|  | return 1; | 
|  | } | 
|  | if (n < 3) { | 
|  | return 0; | 
|  | } | 
|  | char c1 = path.charAt(1); | 
|  | char c2 = path.charAt(2); | 
|  | if (isDriveLetter(c0) && c1 == ':' && isSeparator(c2)) { | 
|  | return 3; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | private static boolean isDriveLetter(char c) { | 
|  | return ((c >= 'a') && (c <= 'z')) || ((c >= 'A') && (c <= 'Z')); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int compare(String s1, String s2) { | 
|  | // Windows is case-insensitive | 
|  | return s1.compareToIgnoreCase(s2); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int compare(char c1, char c2) { | 
|  | return Character.compare(Character.toLowerCase(c1), Character.toLowerCase(c2)); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean equals(String s1, String s2) { | 
|  | return (s1 == null && s2 == null) || (s1 != null && s1.equalsIgnoreCase(s2)); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int hash(String s) { | 
|  | // Windows is case-insensitive | 
|  | if (s == null) { | 
|  | return 0; | 
|  | } | 
|  | return s.toLowerCase().hashCode(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean startsWith(String path, String prefix) { | 
|  | int pathn = path.length(); | 
|  | int prefixn = prefix.length(); | 
|  | if (pathn < prefixn) { | 
|  | return false; | 
|  | } | 
|  | for (int i = 0; i < prefixn; ++i) { | 
|  | if (Character.toLowerCase(path.charAt(i)) != Character.toLowerCase(prefix.charAt(i))) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean endsWith(String path, String suffix) { | 
|  | int pathn = path.length(); | 
|  | int suffixLength = suffix.length(); | 
|  | if (pathn < suffixLength) { | 
|  | return false; | 
|  | } | 
|  | int offset = pathn - suffixLength; | 
|  | for (int i = 0; i < suffixLength; ++i) { | 
|  | if (Character.toLowerCase(path.charAt(i + offset)) | 
|  | != Character.toLowerCase(suffix.charAt(i))) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isSeparator(char c) { | 
|  | return c == '/' || c == '\\'; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public char additionalSeparator() { | 
|  | return '\\'; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean isCaseSensitive() { | 
|  | return false; | 
|  | } | 
|  | } |