blob: 377b06026acaa1a0b7696fe25060a5a0e449c335 [file] [log] [blame]
// Copyright 2016 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.windows;
import java.io.IOException;
/** File operations on Windows. */
public class WindowsFileOperations {
private WindowsFileOperations() {
// Prevent construction
}
// Keep IS_JUNCTION_* values in sync with src/main/native/windows_file_operations.cc.
private static final int IS_JUNCTION_YES = 0;
private static final int IS_JUNCTION_NO = 1;
private static final int IS_JUNCTION_ERROR = 2;
static native int nativeIsJunction(String path, String[] error);
static native boolean nativeGetLongPath(String path, String[] result, String[] error);
/** Determines whether `path` is a junction point or directory symlink. */
public static boolean isJunction(String path) throws IOException {
WindowsJniLoader.loadJni();
String[] error = new String[] {null};
switch (nativeIsJunction(asLongPath(path), error)) {
case IS_JUNCTION_YES:
return true;
case IS_JUNCTION_NO:
return false;
default:
throw new IOException(error[0]);
}
}
/**
* Returns the long path associated with the input `path`.
*
* <p>This method resolves all 8dot3 style components of the path and returns the long format. For
* example, if the input is "C:/progra~1/micros~1" the result may be "C:\Program Files\Microsoft
* Visual Studio 14.0". The returned path is Windows-style in that it uses backslashes, even if
* the input uses forward slashes.
*
* <p>May return an UNC path if `path` or its resolution is sufficiently long.
*
* @throws IOException if the `path` is not found or some other I/O error occurs
*/
public static String getLongPath(String path) throws IOException {
WindowsJniLoader.loadJni();
String[] result = new String[] {null};
String[] error = new String[] {null};
if (nativeGetLongPath(asLongPath(path), result, error)) {
return result[0];
} else {
throw new IOException(error[0]);
}
}
/**
* Returns a Windows-style path suitable to pass to Widechar Win32 API functions.
*
* <p>Returns an UNC path if `path` is longer than `MAX_PATH` (in <windows.h>). If it's shorter or
* is already an UNC path, then this method returns `path` itself.
*/
static String asLongPath(String path) {
return path.length() > 260 && !path.startsWith("\\\\?\\")
? ("\\\\?\\" + path.replace('/', '\\'))
: path;
}
}