@@ -8,7 +8,7 @@ import Foundation
8
8
/// A success `Result` returning `value`
9
9
/// This form is preferred to `Result.Success(Box(value))` because it
10
10
// does not require dealing with `Box()`
11
- public func success< T> ( value: T ) -> Result < T > {
11
+ public func success< T, E > ( value: T ) -> Result < T , E > {
12
12
return . Success( Box ( value) )
13
13
}
14
14
@@ -21,6 +21,10 @@ public func success<T>(value: T) -> Result<T> {
21
21
/// let fail: Result<Int> = failure()
22
22
///
23
23
24
+ /// Dictionary keys for default errors
25
+ public let ErrorFileKey = " LMErrorFile "
26
+ public let ErrorLineKey = " LMErrorLine "
27
+
24
28
private func defaultError( userInfo: [ NSObject : AnyObject ] ) -> NSError {
25
29
return NSError ( domain: " " , code: 0 , userInfo: userInfo)
26
30
}
@@ -33,35 +37,37 @@ private func defaultError(file: String = __FILE__, line: Int = __LINE__) -> NSEr
33
37
return defaultError ( [ ErrorFileKey: file, ErrorLineKey: line] )
34
38
}
35
39
36
- public func failure< T> ( message: String , file: String = __FILE__, line: Int = __LINE__) -> Result < T > {
37
- return failure ( defaultError ( message, file: file, line: line) )
40
+ public func failure< T> ( message: String , file: String = __FILE__, line: Int = __LINE__) -> Result < T , NSError > {
41
+ let userInfo : [ NSObject : AnyObject ] = [ NSLocalizedDescriptionKey: message, ErrorFileKey: file, ErrorLineKey: line]
42
+ return failure ( defaultError ( userInfo) )
38
43
}
39
44
40
- public func failure< T> ( file: String = __FILE__, line: Int = __LINE__) -> Result < T > {
41
- return failure ( defaultError ( file: file, line: line) )
45
+ public func failure< T> ( file: String = __FILE__, line: Int = __LINE__) -> Result < T , NSError > {
46
+ let userInfo : [ NSObject : AnyObject ] = [ ErrorFileKey: file, ErrorLineKey: line]
47
+ return failure ( defaultError ( userInfo) )
42
48
}
43
49
44
- public func failure< T> ( error: ErrorType ) -> Result < T > {
45
- return . Failure( error)
50
+ public func failure< T, E > ( error: E ) -> Result < T , E > {
51
+ return . Failure( Box ( error) )
46
52
}
47
53
48
54
/// Construct a `Result` using a block which receives an error parameter.
49
55
/// Expected to return non-nil for success.
50
56
51
- public func try< T> ( f: NSErrorPointer -> T ? , file: String = __FILE__, line: Int = __LINE__) -> Result < T > {
57
+ public func try< T> ( f: NSErrorPointer -> T ? , file: String = __FILE__, line: Int = __LINE__) -> Result < T , NSError > {
52
58
var error : NSError ?
53
59
return f ( & error) . map ( success) ?? failure ( error ?? defaultError ( file: file, line: line) )
54
60
}
55
61
56
- public func try( f: NSErrorPointer -> Bool , file: String = __FILE__, line: Int = __LINE__) -> Result < ( ) > {
62
+ public func try( f: NSErrorPointer -> Bool , file: String = __FILE__, line: Int = __LINE__) -> Result < ( ) , NSError > {
57
63
var error : NSError ?
58
64
return f ( & error) ? success ( ( ) ) : failure ( error ?? defaultError ( file: file, line: line) )
59
65
}
60
66
61
- /// Container for a successful value (T) or a failure with an NSError
62
- public enum Result < T> {
67
+ /// Container for a successful value (T) or a failure with an E
68
+ public enum Result < T, E > {
63
69
case Success( Box < T > )
64
- case Failure( ErrorType )
70
+ case Failure( Box < E > )
65
71
66
72
/// The successful value as an Optional
67
73
public var value : T ? {
@@ -72,10 +78,10 @@ public enum Result<T> {
72
78
}
73
79
74
80
/// The failing error as an Optional
75
- public var error : ErrorType ? {
81
+ public var error : E ? {
76
82
switch self {
77
83
case . Success: return nil
78
- case . Failure( let err) : return err
84
+ case . Failure( let err) : return err. unbox
79
85
}
80
86
}
81
87
@@ -88,7 +94,7 @@ public enum Result<T> {
88
94
89
95
/// Return a new result after applying a transformation to a successful value.
90
96
/// Mapping a failure returns a new failure without evaluating the transform
91
- public func map< U> ( transform: T -> U ) -> Result < U > {
97
+ public func map< U> ( transform: T -> U ) -> Result < U , E > {
92
98
switch self {
93
99
case Success ( let box) :
94
100
return . Success( Box ( transform ( box. unbox) ) )
@@ -99,8 +105,8 @@ public enum Result<T> {
99
105
100
106
/// Return a new result after applying a transformation (that itself
101
107
/// returns a result) to a successful value.
102
- /// Flat mapping a failure returns a new failure without evaluating the transform
103
- public func flatMap< U> ( transform: T -> Result < U > ) -> Result < U > {
108
+ /// Calling with a failure returns a new failure without evaluating the transform
109
+ public func flatMap< U> ( transform: T -> Result < U , E > ) -> Result < U , E > {
104
110
switch self {
105
111
case Success ( let value) : return transform ( value. unbox)
106
112
case Failure ( let error) : return . Failure( error)
@@ -114,19 +120,26 @@ extension Result: Printable {
114
120
case . Success( let box) :
115
121
return " Success: \( box. unbox) "
116
122
case . Failure( let error) :
117
- return " Failure: \( error) "
123
+ return " Failure: \( error. unbox ) "
118
124
}
119
125
}
120
126
}
121
127
122
128
/// Failure coalescing
123
129
/// .Success(Box(42)) ?? 0 ==> 42
124
130
/// .Failure(NSError()) ?? 0 ==> 0
125
- public func ?? < T> ( result: Result < T > , defaultValue: @autoclosure ( ) -> T ) -> T {
131
+ public func ?? < T, E > ( result: Result < T , E > , defaultValue: @autoclosure ( ) -> T ) -> T {
126
132
switch result {
127
133
case . Success( let value) :
128
134
return value. unbox
129
135
case . Failure( let error) :
130
136
return defaultValue ( )
131
137
}
132
138
}
139
+
140
+ /// Due to current swift limitations, we have to include this Box in Result.
141
+ /// Swift cannot handle an enum with multiple associated data (A, NSError) where one is of unknown size (A)
142
+ final public class Box < T> {
143
+ public let unbox : T
144
+ public init ( _ value: T ) { self . unbox = value }
145
+ }
0 commit comments