-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWealthsimplePosition.swift
165 lines (153 loc) · 6.24 KB
/
WealthsimplePosition.swift
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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
//
// Position.swift
//
//
// Created by Steffen Kötte on 2020-07-12.
//
import Foundation
#if canImport(FoundationNetworking)
import FoundationNetworking
#endif
/// Errors which can happen when retrieving a Position
public enum PositionError: Error {
/// When no data is received from the HTTP request
case noDataReceived
/// When an HTTP error occurs
case httpError(error: String)
/// When the received data is not valid JSON
case invalidJson(error: String)
/// When the received JSON does not have the right type
case invalidJsonType(json: Any)
/// When the received JSON does not have all expected values
case missingResultParamenter(json: [String: Any])
/// When the received JSON does have an unexpected value
case invalidResultParamenter(json: [String: Any])
/// An error with the assets occured
case assetError(_ error: AssetError)
/// An error with the token occured
case tokenError(_ error: TokenError)
}
/// A Position, like certain amount of a stock or a currency held in an account
public protocol Position {
/// Wealthsimple identifier of the account in which this position is held
var accountId: String { get }
/// Asset which is held
var asset: Asset { get }
/// Number of units of the asset held
var quantity: String { get }
/// Price per pice of the asset on `priceDate`
var priceAmount: String { get }
/// Currency of the price
var priceCurrency: String { get }
/// Date of the positon
var positionDate: Date { get }
}
struct WealthsimplePosition: Position {
private static let baseUrl = URLComponents(string: "https://api.production.wealthsimple.com/v1/positions")!
private static var dateFormatter: DateFormatter = {
var dateFormatter = DateFormatter()
dateFormatter.dateFormat = "yyyy-MM-dd"
return dateFormatter
}()
let accountId: String
let asset: Asset
let quantity: String
let priceAmount: String
let priceCurrency: String
let positionDate: Date
private init(json: [String: Any]) throws {
guard let quantity = json["quantity"] as? String,
let accountId = json["account_id"] as? String,
let assetDict = json["asset"] as? [String: Any],
let price = json["market_price"] as? [String: Any],
let dateString = json["position_date"] as? String,
let priceAmount = price["amount"] as? String,
let priceCurrency = price["currency"] as? String,
let object = json["object"] as? String
else {
throw PositionError.missingResultParamenter(json: json)
}
guard let date = Self.dateFormatter.date(from: dateString),
object == "position" else {
throw PositionError.invalidResultParamenter(json: json)
}
do {
self.asset = try WealthsimpleAsset(json: assetDict)
} catch {
throw PositionError.assetError(error as! AssetError) // swiftlint:disable:this force_cast
}
self.accountId = accountId
self.quantity = quantity
self.priceAmount = priceAmount
self.priceCurrency = priceCurrency
self.positionDate = date
}
static func getPositions(token: Token, account: Account, date: Date?, completion: @escaping (Result<[Position], PositionError>) -> Void) {
var url = baseUrl
url.queryItems = [
URLQueryItem(name: "account_id", value: account.id),
URLQueryItem(name: "limit", value: "250")
]
if let date {
url.queryItems?.append(URLQueryItem(name: "date", value: dateFormatter.string(from: date)))
}
var request = URLRequest(url: url.url!)
let session = URLSession.shared
request.setValue("application/json", forHTTPHeaderField: "Content-Type")
token.authenticateRequest(request) {
switch $0 {
case let .failure(error):
completion(.failure(.tokenError(error)))
case let .success(request):
let task = session.dataTask(with: request) { data, response, error in
handleResponse(data: data, response: response, error: error, completion: completion)
}
task.resume()
}
}
}
private static func handleResponse(data: Data?, response: URLResponse?, error: Error?, completion: @escaping (Result<[Position], PositionError>) -> Void) {
guard let data else {
if let error {
completion(.failure(PositionError.httpError(error: error.localizedDescription)))
} else {
completion(.failure(PositionError.noDataReceived))
}
return
}
guard let httpResponse = response as? HTTPURLResponse else {
completion(.failure(PositionError.httpError(error: "No HTTPURLResponse")))
return
}
guard httpResponse.statusCode == 200 else {
completion(.failure(PositionError.httpError(error: "Status code \(httpResponse.statusCode)")))
return
}
do {
completion(try parse(data: data))
} catch {
completion(.failure(PositionError.invalidJson(error: error.localizedDescription)))
return
}
}
private static func parse(data: Data) throws -> Result<[Position], PositionError> {
guard let json = try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] else {
return .failure(PositionError.invalidJsonType(json: try JSONSerialization.jsonObject(with: data, options: [])))
}
do {
guard let results = json["results"] as? [[String: Any]], let object = json["object"] as? String else {
throw PositionError.missingResultParamenter(json: json)
}
guard object == "position" else {
throw PositionError.invalidResultParamenter(json: json)
}
var positions = [Position]()
for result in results {
positions.append(try Self(json: result))
}
return .success(positions)
} catch {
return .failure(error as! PositionError) // swiftlint:disable:this force_cast
}
}
}