@@ -3,29 +3,15 @@ import UIKit
3
3
4
4
/// The testing tool which generates snapshot images from scenarios managed by `Playbook`.
5
5
public struct Snapshot : TestTool {
6
- /// A base directory for exporting snapshot image files.
7
- public var directory : URL
8
-
9
- /// Specifies whether that to clean directory before generating snapshots.
10
- public var clean : Bool
11
-
12
- /// An image file format of exported data.
13
- public var format : SnapshotSupport . ImageFormat
14
-
15
- /// A timeout interval until the finish snapshot of all scenarios.
16
- public var timeout : TimeInterval
17
-
18
- /// A rendering scale of the snapshot image.
19
- public var scale : CGFloat
20
-
21
- /// The key window of the application.
22
- public weak var keyWindow : UIWindow ?
23
-
24
- /// A set of snapshot environment simulating devices.
25
- public var devices : [ SnapshotDevice ]
26
-
27
- /// A closure to preprocess scenario UIView before generating snapshot.
28
- public var viewPreprocessor : ( ( UIView ) -> UIView ) ?
6
+ private let directory : URL
7
+ private let clean : Bool
8
+ private let format : SnapshotSupport . ImageFormat
9
+ private let timeout : TimeInterval
10
+ private let scale : CGFloat
11
+ private let keyWindow : UIWindow ?
12
+ private let devices : [ SnapshotDevice ]
13
+ private let logOutput : TextOutputStream ?
14
+ private let viewPreprocessor : ( ( UIView ) -> UIView ) ?
29
15
30
16
/// Creates a new snapshot tool for export all image files into specified directory.
31
17
///
@@ -46,6 +32,7 @@ public struct Snapshot: TestTool {
46
32
scale: CGFloat = UIScreen . main. scale,
47
33
keyWindow: UIWindow ? = nil ,
48
34
devices: [ SnapshotDevice ] ,
35
+ logOutput: TextOutputStream ? = nil ,
49
36
viewPreprocessor: ( ( UIView ) -> UIView ) ? = nil
50
37
) {
51
38
self . directory = directory
@@ -55,6 +42,7 @@ public struct Snapshot: TestTool {
55
42
self . scale = scale
56
43
self . keyWindow = keyWindow
57
44
self . devices = devices
45
+ self . logOutput = logOutput
58
46
self . viewPreprocessor = viewPreprocessor
59
47
}
60
48
@@ -64,11 +52,8 @@ public struct Snapshot: TestTool {
64
52
/// - playbook: A `Playbook` instance to be tested.
65
53
public func run( with playbook: Playbook ) throws {
66
54
let timeoutTimestamp = Date . timeIntervalSinceReferenceDate + timeout
67
-
68
55
let group = DispatchGroup ( )
69
-
70
56
let fileManager = FileManager . default
71
-
72
57
let directoryURL = URL ( fileURLWithPath: directory. path, isDirectory: true )
73
58
74
59
if clean && fileManager. fileExists ( atPath: directoryURL. path) {
@@ -86,21 +71,21 @@ public struct Snapshot: TestTool {
86
71
87
72
try fileManager. createDirectory ( at: directoryURL, withIntermediateDirectories: true )
88
73
89
- func attemptToWrite ( data : Data , scenario: Scenario ) {
74
+ for scenario in store . scenarios {
90
75
let fileURL =
91
76
directoryURL
92
77
. appendingPathComponent ( normalize ( scenario. title. rawValue) )
93
78
. appendingPathExtension ( format. fileExtension)
94
79
95
- do {
96
- try data. write ( to: fileURL)
97
- }
98
- catch {
99
- writingFailureURLs. append ( fileURL)
100
- }
101
- }
80
+ log (
81
+ """
82
+ Generating snapshot for:
83
+ - Device: \( device. name. rawString)
84
+ - Category: \( store. category. rawValue. rawString)
85
+ - Scenario: \( scenario. title. rawValue. rawString)
86
+ """
87
+ )
102
88
103
- for scenario in store. scenarios {
104
89
group. enter ( )
105
90
106
91
SnapshotSupport . data (
@@ -111,7 +96,20 @@ public struct Snapshot: TestTool {
111
96
keyWindow: keyWindow,
112
97
viewPreprocessor: viewPreprocessor,
113
98
handler: { data in
114
- attemptToWrite ( data: data, scenario: scenario)
99
+ do {
100
+ try data. write ( to: fileURL)
101
+ log ( " Generated in \( fileURL. absoluteString) " )
102
+ }
103
+ catch {
104
+ log (
105
+ """
106
+ Failed to write the snapshot in \( fileURL. absoluteString)
107
+ Error: \( error)
108
+ """
109
+ )
110
+ writingFailureURLs. append ( fileURL)
111
+ }
112
+
115
113
group. leave ( )
116
114
}
117
115
)
@@ -133,14 +131,14 @@ public struct Snapshot: TestTool {
133
131
let timeIntervalToRun = min ( 0.1 , remaining)
134
132
135
133
if timeIntervalToRun <= 0 {
136
- throw SnapshotError . timeout ( timeout)
134
+ try log ( error : . timeout( timeout) )
137
135
}
138
136
139
137
runLoop. run ( mode: . default, before: Date ( timeIntervalSinceNow: timeIntervalToRun) )
140
138
}
141
139
142
- guard writingFailureURLs. isEmpty else {
143
- throw SnapshotError . fileWritingFailure ( urls: writingFailureURLs)
140
+ if ! writingFailureURLs. isEmpty {
141
+ try log ( error : . fileWritingFailure( urls: writingFailureURLs) )
144
142
}
145
143
}
146
144
}
@@ -154,4 +152,26 @@ private extension Snapshot {
154
152
func normalize( _ string: String ) -> String {
155
153
string. components ( separatedBy: Self . nameNormalizationCharacters) . joined ( separator: " _ " )
156
154
}
155
+
156
+ func log( error: SnapshotError ) throws {
157
+ log ( error. debugDescription)
158
+ throw error
159
+ }
160
+
161
+ func log( _ message: String ) {
162
+ let log = " [Playbook] \( message) "
163
+
164
+ if var logOutput {
165
+ logOutput. write ( log)
166
+ }
167
+ else {
168
+ print ( log)
169
+ }
170
+ }
171
+ }
172
+
173
+ private extension String {
174
+ var rawString : String {
175
+ unicodeScalars. reduce ( into: " " ) { $0 += $1. escaped ( asASCII: true ) }
176
+ }
157
177
}
0 commit comments