1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
|
//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2023 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#if FOUNDATION_FRAMEWORK
internal import Foundation_Private.NSFileManager
#endif
extension FileManager.SearchPathDirectory {
#if FOUNDATION_FRAMEWORK
static var _homeDirectory: Self {
Self(rawValue: NSSearchPathDirectory_Private.homeDirectory.rawValue)!
}
#endif
}
extension FileManager.SearchPathDomainMask {
#if FOUNDATION_FRAMEWORK
#if os(macOS)
static var _sharedUserDomainMask: Self {
Self(rawValue: NSSearchPathDomainMask_Private.sharedUserDomainMask.rawValue)
}
#endif
static var _partitionedSystemDomainMask: Self {
Self(rawValue: NSSearchPathDomainMask_Private.partitionedSystemDomainMask.rawValue)
}
static var _appCryptexDomainMask: Self {
Self(rawValue: NSSearchPathDomainMask_Private.appCryptexDomainMask.rawValue)
}
static var _osCryptexDomainMask: Self {
Self(rawValue: NSSearchPathDomainMask_Private.osCryptexDomainMask.rawValue)
}
#endif
internal var firstMask: Self? {
guard !self.isEmpty else { return nil }
return Self(rawValue: 1 << self.rawValue.trailingZeroBitCount)
}
fileprivate static var valid: Self {
#if FOUNDATION_FRAMEWORK
[.userDomainMask, .localDomainMask, .networkDomainMask, .systemDomainMask, ._appCryptexDomainMask, ._osCryptexDomainMask]
#else
[.userDomainMask, .localDomainMask, .networkDomainMask, .systemDomainMask]
#endif
}
}
func _SearchPathURLs(for directory: FileManager.SearchPathDirectory, in domain: FileManager.SearchPathDomainMask, expandTilde: Bool) -> some Sequence<URL> {
#if canImport(Darwin)
let basic = _DarwinSearchPathsSequence(directory: directory, domainMask: domain.intersection(.valid)).lazy.map {
if expandTilde {
return URL(filePath: $0.expandingTildeInPath, directoryHint: .isDirectory)
} else {
return URL(filePath: $0, directoryHint: .isDirectory)
}
}
#if os(macOS) && FOUNDATION_FRAMEWORK
// NSSharedUserDomainMask is basically just a wrapper around NSUserDomainMask.
let compatibleSharedUserDomainMask = domain != .allDomainsMask && (domain.rawValue & 16) != 0
if domain.contains(._sharedUserDomainMask) || compatibleSharedUserDomainMask {
var result = Array(basic)
for path in _DarwinSearchPathsSequence(directory: directory, domainMask: .userDomainMask) {
let expandedURL = URL(filePath: expandTilde ? path.replacingTildeWithRealHomeDirectory : path, directoryHint: .isDirectory)
// Avoid duplicates, which would occur with (NSUserDomainMask | NSSharedUserDomainMask) in non-sandboxed apps.
if !result.contains(expandedURL) {
// Insert this path after NSUserDomainMask and before any of the more general paths.
let insertionIndex = domain.contains(.userDomainMask) ? 1 : 0
result.insert(expandedURL, at: insertionIndex)
}
}
return result
}
#endif
return Array(basic)
#else
var result = Set<URL>()
var domain = domain.intersection(.valid)
while let currentDomain = domain.firstMask {
domain.remove(currentDomain)
#if os(Windows)
let url = _WindowsSearchPathURL(for: directory, in: currentDomain)
#else
let url = _XDGSearchPathURL(for: directory, in: currentDomain)
#endif
if let url {
result.insert(url)
}
}
return result
#endif
}
|