Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Renaming, Reorganization, PrimitiveObjectStore, new ParseEncoder #13

Merged
merged 32 commits into from
Aug 22, 2020
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
bea067c
move stuff around
pranjalsatija Jul 19, 2020
77c84dc
rename things
pranjalsatija Jul 19, 2020
3e6092d
organize ParseObject and ParseUser
pranjalsatija Jul 19, 2020
ef37863
organize Coding
pranjalsatija Jul 20, 2020
00f4622
add IDEWorkspaceChecks
pranjalsatija Jul 20, 2020
3547627
add PrimitiveObjectStore and friends
pranjalsatija Jul 20, 2020
bdfd709
simplify ParseEncoder
pranjalsatija Jul 20, 2020
1819601
remove force unwrap
pranjalsatija Jul 20, 2020
d876bd1
use ParseError
pranjalsatija Jul 20, 2020
b5c5259
retry build
pranjalsatija Jul 20, 2020
69b2136
Merge branch 'master' of https://github.com/parse-community/Parse-Swi…
pranjalsatija Aug 2, 2020
2cd23a9
finish merging
pranjalsatija Aug 3, 2020
1070fe9
add hasSameObjectId
pranjalsatija Aug 3, 2020
f209e2e
add rough NewParseEncoder
pranjalsatija Aug 3, 2020
9a74666
better
pranjalsatija Aug 3, 2020
72684f2
more improvements
pranjalsatija Aug 3, 2020
4cb2019
finished for now
pranjalsatija Aug 4, 2020
38021ae
add key skipping
pranjalsatija Aug 4, 2020
e41acdf
remove old ParseEncoder
pranjalsatija Aug 4, 2020
e48f242
fix tests
pranjalsatija Aug 4, 2020
2e2a164
add MARK comments to ParseEncoder
pranjalsatija Aug 4, 2020
bd09c9e
fix target membership
pranjalsatija Aug 4, 2020
e24503f
Merge branch 'master' of https://github.com/parse-community/Parse-Swi…
pranjalsatija Aug 4, 2020
c5fd3a8
finish merging master
pranjalsatija Aug 4, 2020
55f0d7d
add array support to ParseEncoder; replace _ with <root>
pranjalsatija Aug 4, 2020
81c2cbd
update file credits
pranjalsatija Aug 4, 2020
18c08a5
fix more tests
pranjalsatija Aug 4, 2020
8189db8
address PR feedback
pranjalsatija Aug 5, 2020
c28829c
add ParseEncoderTests
pranjalsatija Aug 22, 2020
f621266
bump test host deployment target
pranjalsatija Aug 22, 2020
a8c43ac
add availability check
pranjalsatija Aug 22, 2020
ee7e1fe
increase coverage target
pranjalsatija Aug 22, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ PlaygroundPage.current.needsIndefiniteExecution = true

initializeParse()

struct GameScore: ParseSwift.ObjectType {
struct GameScore: ParseObject {
//: Those are required for Object
var objectId: String?
var createdAt: Date?
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ PlaygroundPage.current.needsIndefiniteExecution = true

initializeParse()

struct GameScore: ParseSwift.ObjectType {
struct GameScore: ParseObject {
var objectId: String?
var createdAt: Date?
var updatedAt: Date?
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ PlaygroundPage.current.needsIndefiniteExecution = true
import ParseSwift
initializeParse()

struct User: ParseSwift.UserType {
struct User: ParseUser {
//: Those are required for Object
var objectId: String?
var createdAt: Date?
Expand Down
749 changes: 408 additions & 341 deletions ParseSwift.xcodeproj/project.pbxproj

Large diffs are not rendered by default.

27 changes: 15 additions & 12 deletions Sources/ParseSwift/API/API+Commands.swift
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ internal extension API {
let params: [String: String?]?

internal var data: Data? {
return try? getJSONEncoder().encode(body)
return try? ParseCoding.jsonEncoder().encode(body)
}

init(method: API.Method,
Expand Down Expand Up @@ -94,27 +94,27 @@ internal extension API {

internal extension API.Command {
// MARK: Saving
static func saveCommand<T>(_ object: T) -> API.Command<T, T> where T: ObjectType {
static func saveCommand<T>(_ object: T) -> API.Command<T, T> where T: ParseObject {
if object.isSaved {
return updateCommand(object)
}
return createCommand(object)
}

// MARK: Saving - private
private static func createCommand<T>(_ object: T) -> API.Command<T, T> where T: ObjectType {
private static func createCommand<T>(_ object: T) -> API.Command<T, T> where T: ParseObject {
let mapper = { (data) -> T in
try getDecoder().decode(SaveResponse.self, from: data).apply(to: object)
try ParseCoding.jsonDecoder().decode(SaveResponse.self, from: data).apply(to: object)
}
return API.Command<T, T>(method: .POST,
path: object.endpoint,
body: object,
mapper: mapper)
}

private static func updateCommand<T>(_ object: T) -> API.Command<T, T> where T: ObjectType {
private static func updateCommand<T>(_ object: T) -> API.Command<T, T> where T: ParseObject {
let mapper = { (data: Data) -> T in
try getDecoder().decode(UpdateResponse.self, from: data).apply(to: object)
try ParseCoding.jsonDecoder().decode(UpdateResponse.self, from: data).apply(to: object)
}
return API.Command<T, T>(method: .PUT,
path: object.endpoint,
Expand All @@ -123,18 +123,21 @@ internal extension API.Command {
}

// MARK: Fetching
static func fetchCommand<T>(_ object: T) throws -> API.Command<T, T> where T: ObjectType {
static func fetchCommand<T>(_ object: T) throws -> API.Command<T, T> where T: ParseObject {
guard object.isSaved else {
throw ParseError(code: .unknownError, message: "Cannot Fetch an object without id")
}
return API.Command<T, T>(method: .GET,
path: object.endpoint) { (data) -> T in
try getDecoder().decode(FetchResponse.self, from: data).apply(to: object)

return API.Command<T, T>(
method: .GET,
path: object.endpoint
) { (data) -> T in
try ParseCoding.jsonDecoder().decode(FetchResponse.self, from: data).apply(to: object)
}
}
}

extension API.Command where T: ObjectType {
extension API.Command where T: ParseObject {

internal var data: Data? {
guard let body = body else { return nil }
Expand All @@ -159,7 +162,7 @@ extension API.Command where T: ObjectType {
let mapper = { (data: Data) -> [Result<T, ParseError>] in
let decodingType = [BatchResponseItem<WriteResponse>].self
do {
let responses = try getDecoder().decode(decodingType, from: data)
let responses = try ParseCoding.jsonDecoder().decode(decodingType, from: data)
return bodies.enumerated().map({ (object) -> (Result<T, ParseError>) in
let response = responses[object.offset]
if let success = response.success {
Expand Down
2 changes: 1 addition & 1 deletion Sources/ParseSwift/API/API.swift
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ public struct API {
headers["X-Parse-Client-Key"] = clientKey
}

if let token = CurrentUserInfo.currentSessionToken {
if let token = BaseParseUser.currentUserContainer?.sessionToken {
headers["X-Parse-Session-Token"] = token
}

Expand Down
6 changes: 3 additions & 3 deletions Sources/ParseSwift/API/BatchUtils.swift
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,10 @@

import Foundation

typealias ParseObjectBatchCommand<T> = BatchCommand<T, T> where T: ObjectType
typealias ParseObjectBatchCommand<T> = BatchCommand<T, T> where T: ParseObject
typealias ParseObjectBatchResponse<T> = [(Result<T, ParseError>)]
// swiftlint:disable line_length
typealias RESTBatchCommandType<T> = API.Command<ParseObjectBatchCommand<T>, ParseObjectBatchResponse<T>> where T: ObjectType
typealias RESTBatchCommandType<T> = API.Command<ParseObjectBatchCommand<T>, ParseObjectBatchResponse<T>> where T: ParseObject
// swiftlint:enable line_length

internal struct BatchCommand<T, U>: Encodable where T: Encodable {
Expand Down Expand Up @@ -49,7 +49,7 @@ internal struct WriteResponse: Codable {
return FetchResponse(createdAt: createdAt, updatedAt: updatedAt)
}

func apply<T>(to object: T, method: API.Method) -> T where T: ObjectType {
func apply<T>(to object: T, method: API.Method) -> T where T: ParseObject {
switch method {
case .POST:
return asSaveResponse().apply(to: object)
Expand Down
6 changes: 3 additions & 3 deletions Sources/ParseSwift/API/Responses.swift
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ internal struct SaveResponse: Decodable {
return createdAt
}

func apply<T>(to object: T) -> T where T: ObjectType {
func apply<T>(to object: T) -> T where T: ParseObject {
var object = object
object.objectId = objectId
object.createdAt = createdAt
Expand All @@ -27,7 +27,7 @@ internal struct SaveResponse: Decodable {
internal struct UpdateResponse: Decodable {
var updatedAt: Date

func apply<T>(to object: T) -> T where T: ObjectType {
func apply<T>(to object: T) -> T where T: ParseObject {
var object = object
object.updatedAt = updatedAt
return object
Expand All @@ -38,7 +38,7 @@ internal struct FetchResponse: Decodable {
var createdAt: Date
var updatedAt: Date

func apply<T>(to object: T) -> T where T: ObjectType {
func apply<T>(to object: T) -> T where T: ParseObject {
var object = object
object.createdAt = createdAt
object.updatedAt = updatedAt
Expand Down
2 changes: 1 addition & 1 deletion Sources/ParseSwift/API/URLSession+extensions.swift
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ extension URLSession {
do {
return try .success(mapper(responseData))
} catch {
let parseError = try? getDecoder().decode(ParseError.self, from: responseData)
let parseError = try? ParseCoding.jsonDecoder().decode(ParseError.self, from: responseData)
return .failure(parseError ?? ParseError(code: .unknownError, message: "cannot decode error"))
}
} else if let responseError = responseError {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,18 @@ import Foundation
Source: https://github.com/Flight-School/AnyCodable
*/
public struct AnyCodable: Codable {
public typealias DateEncodingStrategy = (Date, Encoder) throws -> Void

public let dateEncodingStrategy: DateEncodingStrategy?
public let value: Any

public init<T>(_ value: T?) {
self.dateEncodingStrategy = nil
self.value = value ?? ()
}

public init<T>(_ value: T?, dateEncodingStrategy: DateEncodingStrategy?) {
self.dateEncodingStrategy = dateEncodingStrategy
self.value = value ?? ()
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,22 +45,23 @@ extension AnyDecodable: _AnyDecodable {}
extension _AnyDecodable {
public init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()

if container.decodeNil() {
self.init(())
} else if let bool = try? container.decode(Bool.self) {
self.init(bool)
} else if let dictionary = try? container.decode([String: AnyCodable].self) {
self.init(dictionary.mapValues { $0.value })
} else if let array = try? container.decode([AnyCodable].self) {
self.init(array.map { $0.value })
} else if let string = try? container.decode(String.self) {
self.init(string)
} else if let int = try? container.decode(Int.self) {
self.init(int)
} else if let uint = try? container.decode(UInt.self) {
self.init(uint)
} else if let double = try? container.decode(Double.self) {
self.init(double)
} else if let string = try? container.decode(String.self) {
self.init(string)
} else if let array = try? container.decode([AnyCodable].self) {
self.init(array.map { $0.value })
} else if let dictionary = try? container.decode([String: AnyCodable].self) {
self.init(dictionary.mapValues { $0.value })
} else if let bool = try? container.decode(Bool.self) {
self.init(bool)
} else {
throw DecodingError.dataCorruptedError(in: container,
debugDescription: "AnyCodable value cannot be decoded")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,14 +29,24 @@ import Foundation
Source: https://github.com/Flight-School/AnyCodable
*/
public struct AnyEncodable: Encodable {
public let dateEncodingStrategy: AnyCodable.DateEncodingStrategy?
public let value: Any

public init<T>(_ value: T?, dateEncodingStrategy: AnyCodable.DateEncodingStrategy?) {
self.dateEncodingStrategy = dateEncodingStrategy
self.value = value ?? ()
}

public init<T>(_ value: T?) {
self.dateEncodingStrategy = nil
self.value = value ?? ()
}
}

@usableFromInline
protocol _AnyEncodable {
var dateEncodingStrategy: AnyCodable.DateEncodingStrategy? { get }

var value: Any { get }
init<T>(_ value: T?)
}
Expand All @@ -46,13 +56,25 @@ extension AnyEncodable: _AnyEncodable {}
// MARK: - Encodable

extension _AnyEncodable {
public func encode(to encoder: Encoder) throws { // swiftlint:disable:this cyclomatic_complexity function_body_length line_length
// swiftlint:disable:next cyclomatic_complexity function_body_length
public func encode(to encoder: Encoder) throws {
if let date = self.value as? Date, let strategy = dateEncodingStrategy {
try strategy(date, encoder)
return
}

var container = encoder.singleValueContainer()
switch self.value {
case is Void:
try container.encodeNil()
case let bool as Bool:
try container.encode(bool)
case let dictionary as [String: Any?]:
try container.encode(dictionary.mapValues { AnyCodable($0, dateEncodingStrategy: dateEncodingStrategy) })
case let array as [Any?]:
try container.encode(array.map { AnyCodable($0, dateEncodingStrategy: dateEncodingStrategy) })
case let url as URL:
try container.encode(url)
case let string as String:
try container.encode(string)
case let date as Date:
try container.encode(date)
case let int as Int:
try container.encode(int)
case let int8 as Int8:
Expand All @@ -77,16 +99,10 @@ extension _AnyEncodable {
try container.encode(float)
case let double as Double:
try container.encode(double)
case let string as String:
try container.encode(string)
case let date as Date:
try container.encode(date)
case let url as URL:
try container.encode(url)
case let array as [Any?]:
try container.encode(array.map { AnyCodable($0) })
case let dictionary as [String: Any?]:
try container.encode(dictionary.mapValues { AnyCodable($0) })
case let bool as Bool:
try container.encode(bool)
case is Void:
try container.encodeNil()
default:
let context = EncodingError.Context(codingPath: container.codingPath,
debugDescription: "AnyCodable value cannot be encoded")
Expand Down
41 changes: 41 additions & 0 deletions Sources/ParseSwift/Coding/Extensions.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
//
// Extensions.swift
//
pranjalsatija marked this conversation as resolved.
Show resolved Hide resolved
//
// Created by Pranjal Satija on 7/19/20.
//

import Foundation

// MARK: Date
internal extension Date {
func parseFormatted() -> String {
return ParseCoding.dateFormatter.string(from: self)
}

var parseRepresentation: [String: String] {
return ["__type": "Date", "iso": parseFormatted()]
}
}

// MARK: JSONEncoder
extension JSONEncoder {
func encodeAsString<T>(_ value: T) throws -> String where T: Encodable {
guard let string = String(data: try encode(value), encoding: .utf8) else {
throw ParseError(code: .unknownError, message: "Unable to encode object...")
}

return string
}
}

// MARK: ParseObject
internal extension ParseObject {
func getEncoder(skipKeys: Bool = true) -> ParseEncoder {
return ParseCoding.parseEncoder(skipKeys: skipKeys)
}

func getTestDecoder() -> JSONDecoder {
ParseCoding.jsonDecoder()
}
}
Loading