2017-08-19 14:36:56 +00:00
// Copyright (C) 2014 The Syncthing Authors.
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
// You can obtain one at https://mozilla.org/MPL/2.0/.
// +build windows
package fs
import (
"bytes"
"errors"
"fmt"
"os"
"path/filepath"
2018-06-04 11:41:03 +00:00
"strings"
2017-08-19 14:36:56 +00:00
"syscall"
"unsafe"
)
var errNotSupported = errors . New ( "symlinks not supported" )
func ( BasicFilesystem ) SymlinksSupported ( ) bool {
return false
}
func ( BasicFilesystem ) ReadSymlink ( path string ) ( string , error ) {
return "" , errNotSupported
}
2018-01-14 14:25:04 +00:00
func ( BasicFilesystem ) CreateSymlink ( target , name string ) error {
2017-08-19 14:36:56 +00:00
return errNotSupported
}
// Required due to https://github.com/golang/go/issues/10900
func ( f * BasicFilesystem ) mkdirAll ( path string , perm os . FileMode ) error {
// Fast path: if we can tell whether path is a directory or file, stop with success or error.
dir , err := os . Stat ( path )
if err == nil {
if dir . IsDir ( ) {
return nil
}
return & os . PathError {
Op : "mkdir" ,
Path : path ,
Err : syscall . ENOTDIR ,
}
}
// Slow path: make sure parent exists and then call Mkdir for path.
i := len ( path )
for i > 0 && IsPathSeparator ( path [ i - 1 ] ) { // Skip trailing path separator.
i --
}
j := i
for j > 0 && ! IsPathSeparator ( path [ j - 1 ] ) { // Scan backward over element.
j --
}
if j > 1 {
// Create parent
parent := path [ 0 : j - 1 ]
if parent != filepath . VolumeName ( parent ) {
2018-02-16 14:19:20 +00:00
err = f . mkdirAll ( parent , perm )
2017-08-19 14:36:56 +00:00
if err != nil {
return err
}
}
}
// Parent now exists; invoke Mkdir and use its result.
err = os . Mkdir ( path , perm )
if err != nil {
// Handle arguments like "foo/." by
// double-checking that directory doesn't exist.
dir , err1 := os . Lstat ( path )
if err1 == nil && dir . IsDir ( ) {
return nil
}
return err
}
return nil
}
func ( f * BasicFilesystem ) Unhide ( name string ) error {
name , err := f . rooted ( name )
if err != nil {
return err
}
p , err := syscall . UTF16PtrFromString ( name )
if err != nil {
return err
}
attrs , err := syscall . GetFileAttributes ( p )
if err != nil {
return err
}
attrs &^= syscall . FILE_ATTRIBUTE_HIDDEN
return syscall . SetFileAttributes ( p , attrs )
}
func ( f * BasicFilesystem ) Hide ( name string ) error {
name , err := f . rooted ( name )
if err != nil {
return err
}
p , err := syscall . UTF16PtrFromString ( name )
if err != nil {
return err
}
attrs , err := syscall . GetFileAttributes ( p )
if err != nil {
return err
}
attrs |= syscall . FILE_ATTRIBUTE_HIDDEN
return syscall . SetFileAttributes ( p , attrs )
}
func ( f * BasicFilesystem ) Roots ( ) ( [ ] string , error ) {
kernel32 , err := syscall . LoadDLL ( "kernel32.dll" )
if err != nil {
return nil , err
}
getLogicalDriveStringsHandle , err := kernel32 . FindProc ( "GetLogicalDriveStringsA" )
if err != nil {
return nil , err
}
buffer := [ 1024 ] byte { }
bufferSize := uint32 ( len ( buffer ) )
hr , _ , _ := getLogicalDriveStringsHandle . Call ( uintptr ( unsafe . Pointer ( & bufferSize ) ) , uintptr ( unsafe . Pointer ( & buffer ) ) )
if hr == 0 {
return nil , fmt . Errorf ( "Syscall failed" )
}
var drives [ ] string
parts := bytes . Split ( buffer [ : ] , [ ] byte { 0 } )
for _ , part := range parts {
if len ( part ) == 0 {
break
}
drives = append ( drives , string ( part ) )
}
return drives , nil
}
2018-06-04 11:41:03 +00:00
2018-09-11 20:30:32 +00:00
// unrootedChecked returns the path relative to the folder root (same as
// unrooted). It panics if the given path is not a subpath and handles the
// special case when the given path is the folder root without a trailing
// pathseparator.
func ( f * BasicFilesystem ) unrootedChecked ( absPath , root string ) string {
absPath = f . resolveWin83 ( absPath )
lowerAbsPath := UnicodeLowercase ( absPath )
lowerRoot := UnicodeLowercase ( root )
if lowerAbsPath + string ( PathSeparator ) == lowerRoot {
return "."
}
if ! strings . HasPrefix ( lowerAbsPath , lowerRoot ) {
panic ( fmt . Sprintf ( "bug: Notify backend is processing a change outside of the filesystem root: f.root==%v, root==%v (lower), path==%v (lower)" , f . root , lowerRoot , lowerAbsPath ) )
}
return rel ( absPath , root )
}
func rel ( path , prefix string ) string {
lowerRel := strings . TrimPrefix ( strings . TrimPrefix ( UnicodeLowercase ( path ) , UnicodeLowercase ( prefix ) ) , string ( PathSeparator ) )
return path [ len ( path ) - len ( lowerRel ) : ]
}
2018-06-04 11:41:03 +00:00
func ( f * BasicFilesystem ) resolveWin83 ( absPath string ) string {
if ! isMaybeWin83 ( absPath ) {
return absPath
}
if in , err := syscall . UTF16FromString ( absPath ) ; err == nil {
out := make ( [ ] uint16 , 4 * len ( absPath ) ) // *2 for UTF16 and *2 to double path length
if n , err := syscall . GetLongPathName ( & in [ 0 ] , & out [ 0 ] , uint32 ( len ( out ) ) ) ; err == nil {
if n <= uint32 ( len ( out ) ) {
return syscall . UTF16ToString ( out [ : n ] )
}
out = make ( [ ] uint16 , n )
if _ , err = syscall . GetLongPathName ( & in [ 0 ] , & out [ 0 ] , n ) ; err == nil {
return syscall . UTF16ToString ( out )
}
}
}
// Failed getting the long path. Return the part of the path which is
// already a long path.
2018-09-11 20:30:32 +00:00
lowerRoot := UnicodeLowercase ( f . root )
for absPath = filepath . Dir ( absPath ) ; strings . HasPrefix ( UnicodeLowercase ( absPath ) , lowerRoot ) ; absPath = filepath . Dir ( absPath ) {
2018-06-04 11:41:03 +00:00
if ! isMaybeWin83 ( absPath ) {
return absPath
}
}
2018-08-11 20:24:36 +00:00
return f . root
2018-06-04 11:41:03 +00:00
}
func isMaybeWin83 ( absPath string ) bool {
if ! strings . Contains ( absPath , "~" ) {
return false
}
if strings . Contains ( filepath . Dir ( absPath ) , "~" ) {
return true
}
return strings . Contains ( strings . TrimPrefix ( filepath . Base ( absPath ) , WindowsTempPrefix ) , "~" )
}
2018-09-26 18:28:20 +00:00
func evalSymlinks ( in string ) ( string , error ) {
out , err := filepath . EvalSymlinks ( in )
if err != nil && strings . HasPrefix ( in , ` \\?\ ` ) {
// Try again without the `\\?\` prefix
out , err = filepath . EvalSymlinks ( in [ 4 : ] )
}
if err != nil {
return "" , err
}
return longFilenameSupport ( out ) , nil
}
2019-05-11 08:06:04 +00:00
// watchPaths adjust the folder root for use with the notify backend and the
// corresponding absolute path to be passed to notify to watch name.
func ( f * BasicFilesystem ) watchPaths ( name string ) ( string , string , error ) {
root , err := evalSymlinks ( f . root )
if err != nil {
return "" , "" , err
}
// Remove `\\?\` prefix if the path is just a drive letter as a dirty
// fix for https://github.com/syncthing/syncthing/issues/5578
if filepath . Clean ( name ) == "." && len ( root ) <= 7 && len ( root ) > 4 && root [ : 4 ] == ` \\?\ ` {
root = root [ 4 : ]
}
absName , err := rooted ( name , root )
if err != nil {
return "" , "" , err
}
root = f . resolveWin83 ( root )
absName = f . resolveWin83 ( absName )
return filepath . Join ( absName , "..." ) , root , nil
}