File: XCTestCase%2BTemporaryDirectory.swift

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (89 lines) | stat: -rw-r--r-- 4,028 bytes parent folder | download
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
/*
 This source file is part of the Swift.org open source project

 Copyright (c) 2021 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 Swift project authors
*/

import Foundation
import XCTest

// These helpers methods exist to put temp files for different test executions in different locations when running in Swift CI.

extension FileManager {
    
    @available(*, deprecated, message: "Use `createTemporaryDirectory` instead in unit tests to avoid referencing a shared location in Swift CI.")
    var temporaryDirectory: URL {
        XCTFail("Use `createTemporaryDirectory` instead in unit tests to avoid referencing a shared location in Swift CI.")
        return URL(fileURLWithPath: Foundation.NSTemporaryDirectory())
    }
}

public extension XCTestCase {
    
    @available(*, deprecated, message: "Use `createTemporaryDirectory` instead in unit tests to avoid referencing a shared location in Swift CI.")
    func NSTemporaryDirectory() -> String {
        XCTFail("Use `createTemporaryDirectory` instead in unit tests to avoid referencing a shared location in Swift CI.")
        return Foundation.NSTemporaryDirectory()
    }
    
    /// Creates a new temporary directory and returns the URL of that directory.
    ///
    /// After the current test method has returned the temporary directory is automatically removed.
    ///
    /// - Parameters:
    ///   - pathComponents: The name of the temporary directory.
    ///   - fileManager: The file manager that will create the directory.
    /// - Returns: The URL of the newly created directory.
    func createTemporaryDirectory(named: String, fileManager: FileManager = .default) throws -> URL {
        try createTemporaryDirectory(pathComponents: named)
    }
        
    /// Creates a new temporary directory and returns the URL of that directory.
    ///
    /// After the current test method has returned the temporary directory is automatically removed.
    ///
    /// - Parameters:
    ///   - pathComponents: Additional path components to add to the temporary URL.
    ///   - fileManager: The file manager that will create the directory.
    /// - Returns: The URL of the newly created directory.
    func createTemporaryDirectory(pathComponents: String..., fileManager: FileManager = .default) throws -> URL {
        let bundleParentDir = Bundle(for: Self.self).bundleURL.deletingLastPathComponent()
        let baseURL = bundleParentDir.appendingPathComponent(name.replacingWhitespaceAndPunctuation(with: "-"))
        
        var tempURL = baseURL.appendingPathComponent(ProcessInfo.processInfo.globallyUniqueString)
        for component in pathComponents {
            tempURL.appendPathComponent(component)
        }
        tempURL.standardize()
        
        addTeardownBlock {
            do {
                if fileManager.fileExists(atPath: baseURL.path) {
                    try fileManager.removeItem(at: baseURL)
                }
            } catch {
                XCTFail("Failed to remove temporary directory: '\(error)'")
            }
        }
        
        if !fileManager.fileExists(atPath: bundleParentDir.path) {
            // Create the base URL directory without intermediate directories so that an error is raised if the parent directory doesn't exist.
            try fileManager.createDirectory(at: baseURL, withIntermediateDirectories: false, attributes: nil)
        }
         
        try fileManager.createDirectory(at: tempURL, withIntermediateDirectories: true, attributes: nil)
        
        return tempURL
    }
}

private extension String {
    func replacingWhitespaceAndPunctuation(with separator: String) -> String {
        let charactersToStrip = CharacterSet.whitespaces.union(.punctuationCharacters)
        return components(separatedBy: charactersToStrip).filter({ !$0.isEmpty }).joined(separator: separator)
    }
}