From a9b7bcfaf84d7defb92ad966b1695e6dedb913d6 Mon Sep 17 00:00:00 2001 From: Andrew Edwards Date: Thu, 28 Feb 2019 14:59:36 -0500 Subject: [PATCH 1/4] Added support for Persons API. --- README.md | 2 +- Sources/Stripe/API/Helpers/Endpoints.swift | 7 + Sources/Stripe/API/Routes/PersonRoutes.swift | 350 +++++++++++++++++++ Sources/Stripe/Models/Connect/Person.swift | 17 + Sources/Stripe/Provider/Provider.swift | 2 + 5 files changed, 377 insertions(+), 1 deletion(-) create mode 100644 Sources/Stripe/API/Routes/PersonRoutes.swift diff --git a/README.md b/README.md index bbc365c..aa3c828 100644 --- a/README.md +++ b/README.md @@ -85,7 +85,7 @@ And you can always check the documentation to see the required paramaters for sp * [ ] Application Fees * [ ] Country Specs * [ ] External Accounts -* [ ] Persons +* [x] Persons * [ ] Top-ups * [x] Transfers * [x] Transfer Reversals diff --git a/Sources/Stripe/API/Helpers/Endpoints.swift b/Sources/Stripe/API/Helpers/Endpoints.swift index a42d9c9..97d039b 100644 --- a/Sources/Stripe/API/Helpers/Endpoints.swift +++ b/Sources/Stripe/API/Helpers/Endpoints.swift @@ -123,6 +123,10 @@ internal enum StripeAPIEndpoint { case file case files(String) + // MARK: - PERSONS + case person(String) + case persons(String, String) + var endpoint: String { switch self { case .balance: return APIBase + APIVersion + "balance" @@ -210,6 +214,9 @@ internal enum StripeAPIEndpoint { case .file: return FilesAPIBase + APIVersion + "files" case .files(let id): return FilesAPIBase + APIVersion + "files/\(id)" + + case .person(let account): return APIBase + APIVersion + "accounts/\(account)/persons" + case .persons(let account, let person): return APIBase + APIVersion + "accounts/\(account)/persons\(person)" } } } diff --git a/Sources/Stripe/API/Routes/PersonRoutes.swift b/Sources/Stripe/API/Routes/PersonRoutes.swift new file mode 100644 index 0000000..b10f3ba --- /dev/null +++ b/Sources/Stripe/API/Routes/PersonRoutes.swift @@ -0,0 +1,350 @@ +// +// PersonRoutes.swift +// Stripe +// +// Created by Andrew Edwards on 2/28/19. +// + +import Vapor + +public protocol PersonRoutes { + /// Creates a new person. + /// + /// - Parameters: + /// - account: The unique identifier of the account the person is associated with. + /// - address: The person’s address. + /// - dob: The person’s date of birth. + /// - email: The person’s email address. + /// - firstName: The person’s first name. + /// - gender: The person’s gender (International regulations require either “male” or “female”). + /// - idNumber: The person’s ID number, as appropriate for their country. For example, a social security number in the U.S., social insurance number in Canada, etc. Instead of the number itself, you can also provide a PII token provided by Stripe.js. + /// - lastName: The person’s last name. + /// - maidenName: The person’s maiden name. + /// - metadata: Set of key-value pairs that you can attach to an object. This can be useful for storing additional information about the object in a structured format. Individual keys can be unset by posting an empty value to them. All keys can be unset by posting an empty value to metadata. + /// - phone: The person’s phone number. + /// - relationship: The relationship that this person has with the account’s legal entity. + /// - ssnLast4: The last 4 digits of the person’s social security number. + /// - verification: The person’s verification status. + /// - Returns: Returns a person object. + /// - Throws: A `StripeError` + func create(account: String, + address: [String: Any]?, + dob: [String: Any]?, + email: String?, + firstName: String?, + gender: StripePersonGender?, + idNumber: String?, + lastName: String?, + maidenName: String?, + metadata: [String: String]?, + phone: String?, + relationship: [String: Any]?, + ssnLast4: String?, + verification: [String: Any]?) throws -> EventLoopFuture + + /// Retrieves an existing person. + /// + /// - Parameters: + /// - account: The unique identifier of the account the person is associated with. + /// - person: The ID of a person to retrieve. + /// - Returns: Returns a person object. + /// - Throws: A `StripeError` + func retrieve(account: String, person: String) throws -> EventLoopFuture + + /// Updates an existing person. + /// + /// - Parameters: + /// - account: The unique identifier of the account the person is associated with. + /// - person: The ID of a person to update. + /// - address: The person’s address. + /// - dob: The person’s date of birth. + /// - email: The person’s email address. + /// - firstName: The person’s first name. + /// - gender: The person’s gender (International regulations require either “male” or “female”). + /// - idNumber: The person’s ID number, as appropriate for their country. For example, a social security number in the U.S., social insurance number in Canada, etc. Instead of the number itself, you can also provide a PII token provided by Stripe.js. + /// - lastName: The person’s last name. + /// - maidenName: The person’s maiden name. + /// - metadata: Set of key-value pairs that you can attach to an object. This can be useful for storing additional information about the object in a structured format. Individual keys can be unset by posting an empty value to them. All keys can be unset by posting an empty value to metadata. + /// - phone: The person’s phone number. + /// - relationship: The relationship that this person has with the account’s legal entity. + /// - ssnLast4: The last 4 digits of the person’s social security number. + /// - verification: The person’s verification status. + /// - Returns: Returns a person object. + /// - Throws: A `StripeError` + func update(account: String, + person: String, + address: [String: Any]?, + dob: [String: Any]?, + email: String?, + firstName: String?, + gender: StripePersonGender?, + idNumber: String?, + lastName: String?, + maidenName: String?, + metadata: [String: String]?, + phone: String?, + relationship: [String: Any]?, + ssnLast4: String?, + verification: [String: Any]?) throws -> EventLoopFuture + + /// Deletes an existing person’s relationship to the account’s legal entity. + /// + /// - Parameters: + /// - account: The unique identifier of the account the person is associated with. + /// - person: The ID of a person to update. + /// - Returns: Returns the deleted person object. + /// - Throws: A `StripeError` + func delete(account: String, person: String) throws -> EventLoopFuture + + /// Returns a list of people associated with the account’s legal entity. The people are returned sorted by creation date, with the most recent people appearing first. + /// + /// - Parameters: + /// - account: The unique identifier of the account the person is associated with. + /// - filter: A dictionary that will be used for the query parameters. [See More →](https://stripe.com/docs/api/persons/list?&lang=curl) + /// - Returns: A `PersonsList` + /// - Throws: A `StripeError` + func listAll(account: String, filter: [String: Any]?) throws -> EventLoopFuture +} + +extension PersonRoutes { + public func create(account: String, + address: [String: Any]? = nil, + dob: [String: Any]? = nil, + email: String? = nil, + firstName: String? = nil, + gender: StripePersonGender? = nil, + idNumber: String? = nil, + lastName: String? = nil, + maidenName: String? = nil, + metadata: [String: String]? = nil, + phone: String? = nil, + relationship: [String: Any]? = nil, + ssnLast4: String?, + verification: [String: Any]? = nil) throws -> EventLoopFuture { + return try create(account: account, + address: address, + dob: dob, + email: email, + firstName: firstName, + gender: gender, + idNumber: idNumber, + lastName: lastName, + maidenName: maidenName, + metadata: metadata, + phone: phone, + relationship: relationship, + ssnLast4: ssnLast4, + verification: verification) + } + + public func retrieve(account: String, person: String) throws -> EventLoopFuture { + return try retrieve(account: account, person: person) + } + + public func update(account: String, + person: String, + address: [String: Any]? = nil, + dob: [String: Any]? = nil, + email: String? = nil, + firstName: String? = nil, + gender: StripePersonGender? = nil, + idNumber: String? = nil, + lastName: String? = nil, + maidenName: String? = nil, + metadata: [String: String]? = nil, + phone: String? = nil, + relationship: [String: Any]? = nil, + ssnLast4: String? = nil, + verification: [String: Any]? = nil) throws -> EventLoopFuture { + return try update(account: account, + person: person, + address: address, + dob: dob, + email: email, + firstName: firstName, + gender: gender, + idNumber: idNumber, + lastName: lastName, + maidenName: maidenName, + metadata: metadata, + phone: phone, + relationship: relationship, + ssnLast4: ssnLast4, + verification: verification) + } + + func delete(account: String, person: String) throws -> EventLoopFuture { + return try delete(account: account, person: person) + } + + func listAll(account: String, filter: [String: Any]? = nil) throws -> EventLoopFuture { + return try listAll(account: account, filter: filter) + } +} + +public struct StripePersonRoutes: PersonRoutes { + private let request: StripeRequest + + init(request: StripeRequest) { + self.request = request + } + + public func create(account: String, + address: [String: Any]?, + dob: [String: Any]?, + email: String?, + firstName: String?, + gender: StripePersonGender?, + idNumber: String?, + lastName: String?, + maidenName: String?, + metadata: [String: String]?, + phone: String?, + relationship: [String: Any]?, + ssnLast4: String?, + verification: [String : Any]?) throws -> EventLoopFuture { + var body: [String: Any] = [:] + + if let address = address { + address.forEach { body["address[\($0)]"] = $1 } + } + + if let dob = dob { + dob.forEach { body["dob[\($0)]"] = $1 } + } + + if let email = email { + body["email"] = email + } + + if let firstName = firstName { + body["first_name"] = firstName + } + + if let gender = gender { + body["gender"] = gender.rawValue + } + + if let idNumber = idNumber { + body["id_number"] = idNumber + } + + if let lastName = lastName { + body["last_name"] = lastName + } + + if let maidenName = maidenName { + body["maiden_name"] = maidenName + } + + if let metadata = metadata { + metadata.forEach { body["metadata[\($0)]"] = $1 } + } + + if let phone = phone { + body["phone"] = phone + } + + if let relationship = relationship { + relationship.forEach { body["relationship[\($0)]"] = $1 } + } + + if let ssnLast4 = ssnLast4 { + body["ssn_last_4"] = ssnLast4 + } + + if let verification = verification { + verification.forEach { body["verification[\($0)]"] = $1 } + } + + return try request.send(method: .POST, path: StripeAPIEndpoint.person(account).endpoint, body: body.queryParameters) + } + + public func retrieve(account: String, person: String) throws -> EventLoopFuture { + return try request.send(method: .GET, path: StripeAPIEndpoint.persons(account, person).endpoint) + } + + public func update(account: String, + person: String, + address: [String: Any]?, + dob: [String: Any]?, + email: String?, + firstName: String?, + gender: StripePersonGender?, + idNumber: String?, + lastName: String?, + maidenName: String?, + metadata: [String: String]?, + phone: String?, + relationship: [String: Any]?, + ssnLast4: String?, + verification: [String: Any]?) throws -> EventLoopFuture { + var body: [String: Any] = [:] + + if let address = address { + address.forEach { body["address[\($0)]"] = $1 } + } + + if let dob = dob { + dob.forEach { body["dob[\($0)]"] = $1 } + } + + if let email = email { + body["email"] = email + } + + if let firstName = firstName { + body["first_name"] = firstName + } + + if let gender = gender { + body["gender"] = gender.rawValue + } + + if let idNumber = idNumber { + body["id_number"] = idNumber + } + + if let lastName = lastName { + body["last_name"] = lastName + } + + if let maidenName = maidenName { + body["maiden_name"] = maidenName + } + + if let metadata = metadata { + metadata.forEach { body["metadata[\($0)]"] = $1 } + } + + if let phone = phone { + body["phone"] = phone + } + + if let relationship = relationship { + relationship.forEach { body["relationship[\($0)]"] = $1 } + } + + if let ssnLast4 = ssnLast4 { + body["ssn_last_4"] = ssnLast4 + } + + if let verification = verification { + verification.forEach { body["verification[\($0)]"] = $1 } + } + + return try request.send(method: .POST, path: StripeAPIEndpoint.persons(account, person).endpoint, body: body.queryParameters) + } + + public func delete(account: String, person: String) throws -> EventLoopFuture { + return try request.send(method: .DELETE, path: StripeAPIEndpoint.persons(account, person).endpoint) + } + + public func listAll(account: String, filter: [String : Any]?) throws -> EventLoopFuture { + var queryParams = "" + if let filter = filter { + queryParams = filter.queryParameters + } + return try request.send(method: .GET, path: StripeAPIEndpoint.person(account).endpoint, query: queryParams) + } +} diff --git a/Sources/Stripe/Models/Connect/Person.swift b/Sources/Stripe/Models/Connect/Person.swift index 729500c..682af2d 100644 --- a/Sources/Stripe/Models/Connect/Person.swift +++ b/Sources/Stripe/Models/Connect/Person.swift @@ -7,6 +7,7 @@ import Foundation +// https://stripe.com/docs/api/persons/object?&lang=curl public struct StripePerson: StripeModel { public var id: String public var object: String @@ -147,3 +148,19 @@ public enum StripePersonVerificationStatus: String, StripeModel { case pending case verified } + +public struct PersonsList: StripeModel { + public var object: String + public var hasMore: Bool + public var totalCount: Int? + public var url: String? + public var data: [StripePerson]? + + private enum CodingKeys: String, CodingKey { + case object + case hasMore = "has_more" + case totalCount = "total_count" + case url + case data + } +} diff --git a/Sources/Stripe/Provider/Provider.swift b/Sources/Stripe/Provider/Provider.swift index dfe23c8..cc2afa9 100644 --- a/Sources/Stripe/Provider/Provider.swift +++ b/Sources/Stripe/Provider/Provider.swift @@ -68,6 +68,7 @@ public final class StripeClient: Service { public var payouts: PayoutRoutes public var fileLinks: FileLinkRoutes public var files: FileRoutes + public var person: PersonRoutes internal init(apiKey: String, testKey: String?, client: Client) { let apiRequest = StripeAPIRequest(httpClient: client, apiKey: apiKey, testApiKey: testKey) @@ -96,5 +97,6 @@ public final class StripeClient: Service { payouts = StripePayoutRoutes(request: apiRequest) fileLinks = StripeFileLinkRoutes(request: apiRequest) files = StripeFileRoutes(request: apiRequest) + person = StripePersonRoutes(request: apiRequest) } } From 39909df3e996bf06e287999519637475cd53c2ec Mon Sep 17 00:00:00 2001 From: Andrew Edwards Date: Thu, 28 Feb 2019 14:59:52 -0500 Subject: [PATCH 2/4] Re-order. --- Sources/Stripe/Models/Other/DeletedObject.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Sources/Stripe/Models/Other/DeletedObject.swift b/Sources/Stripe/Models/Other/DeletedObject.swift index e5a412e..2c2fe28 100644 --- a/Sources/Stripe/Models/Other/DeletedObject.swift +++ b/Sources/Stripe/Models/Other/DeletedObject.swift @@ -14,6 +14,7 @@ import Vapor */ public struct StripeDeletedObject: StripeModel { - public var deleted: Bool public var id: String + public var object: String + public var deleted: Bool } From 55b5987e0ca2c30647bdaa4409eb92d7cd868222 Mon Sep 17 00:00:00 2001 From: Andrew Edwards Date: Thu, 28 Feb 2019 15:00:12 -0500 Subject: [PATCH 3/4] Increased max size for decoding. --- Sources/Stripe/API/StripeRequest.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Sources/Stripe/API/StripeRequest.swift b/Sources/Stripe/API/StripeRequest.swift index 6997a35..baa8971 100644 --- a/Sources/Stripe/API/StripeRequest.swift +++ b/Sources/Stripe/API/StripeRequest.swift @@ -25,12 +25,12 @@ public extension StripeRequest { decoder.dateDecodingStrategy = .secondsSince1970 guard response.status == .ok else { - return try decoder.decode(StripeError.self, from: response, maxSize: 65_536, on: worker).map(to: SM.self){ error in + return try decoder.decode(StripeError.self, from: response, maxSize: 1_000_000, on: worker).map(to: SM.self){ error in throw error } } - return try decoder.decode(SM.self, from: response, maxSize: 65_536, on: worker) + return try decoder.decode(SM.self, from: response, maxSize: 1_000_000, on: worker) } } From 89708dd5312f91bf46fb499103f4a0c513ab1450 Mon Sep 17 00:00:00 2001 From: Andrew Edwards Date: Thu, 28 Feb 2019 15:09:37 -0500 Subject: [PATCH 4/4] Formatting. --- Sources/Stripe/API/Routes/PersonRoutes.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/Stripe/API/Routes/PersonRoutes.swift b/Sources/Stripe/API/Routes/PersonRoutes.swift index b10f3ba..a131620 100644 --- a/Sources/Stripe/API/Routes/PersonRoutes.swift +++ b/Sources/Stripe/API/Routes/PersonRoutes.swift @@ -202,7 +202,7 @@ public struct StripePersonRoutes: PersonRoutes { phone: String?, relationship: [String: Any]?, ssnLast4: String?, - verification: [String : Any]?) throws -> EventLoopFuture { + verification: [String: Any]?) throws -> EventLoopFuture { var body: [String: Any] = [:] if let address = address {