From 41469ee91f54938c2ed5e3c991aaaeb6ae4cca13 Mon Sep 17 00:00:00 2001 From: Serge Kazakov Date: Fri, 2 Aug 2024 21:40:01 +0100 Subject: [PATCH] Collection nested queries (#44) * added collection nested query extension * fixed save inverse relation bug * added tests for nested queries * refactoring --- Sources/SwiftletModel/Model/Query.swift | 51 ++ Sources/SwiftletModel/Relation/Relation.swift | 2 +- .../SwiftletModel/Relation/RelationSave.swift | 2 +- .../QueriesTests/NestedModelsQueryTest.swift | 681 ++++++++++++++++++ 4 files changed, 734 insertions(+), 2 deletions(-) create mode 100644 Tests/SwiftletModelTests/QueriesTests/NestedModelsQueryTest.swift diff --git a/Sources/SwiftletModel/Model/Query.swift b/Sources/SwiftletModel/Model/Query.swift index 87f6406..d2a0eaf 100644 --- a/Sources/SwiftletModel/Model/Query.swift +++ b/Sources/SwiftletModel/Model/Query.swift @@ -132,6 +132,57 @@ public extension Query { } } +public extension Collection { + func with( + _ keyPath: WritableKeyPath>, + nested: @escaping QueryModifier = { $0 }) -> [Query] where Element == Query { + + map { $0.with(keyPath, nested: nested) } + } + + func with( + _ keyPath: WritableKeyPath>, + nested: @escaping QueryModifier = { $0 } + + ) -> [Query] where Element == Query { + + map { $0.with(keyPath, nested: nested) } + } + + func with( + fragment keyPath: WritableKeyPath>, + nested: @escaping QueryModifier = { $0 } + + ) -> [Query] where Element == Query { + + map { $0.with(fragment: keyPath, nested: nested) } + } + + func id( + _ keyPath: WritableKeyPath> + + ) -> [Query] where Element == Query { + + map { $0.id(keyPath) } + } + + func ids( + _ keyPath: WritableKeyPath> + + ) -> [Query] where Element == Query { + + map { $0.ids(keyPath) } + } + + func ids( + fragment keyPath: WritableKeyPath> + + ) -> [Query] where Element == Query { + + map { $0.ids(fragment: keyPath) } + } +} + extension Context { func query(_ id: Entity.ID) -> Query { Query(context: self, id: id) diff --git a/Sources/SwiftletModel/Relation/Relation.swift b/Sources/SwiftletModel/Relation/Relation.swift index da9efed..069f6fe 100644 --- a/Sources/SwiftletModel/Relation/Relation.swift +++ b/Sources/SwiftletModel/Relation/Relation.swift @@ -112,7 +112,7 @@ extension Relation { } } - var inverseLinkUpdateOption: Option { + static var inverseLinkUpdateOption: Option { Cardinality.isToMany ? .append : .replace } } diff --git a/Sources/SwiftletModel/Relation/RelationSave.swift b/Sources/SwiftletModel/Relation/RelationSave.swift index 68a5773..87073a6 100644 --- a/Sources/SwiftletModel/Relation/RelationSave.swift +++ b/Sources/SwiftletModel/Relation/RelationSave.swift @@ -142,7 +142,7 @@ private extension EntityModel { children: [id], attribute: LinkAttribute( name: inverse.name, - updateOption: relation(keyPath).inverseLinkUpdateOption + updateOption: MutualRelation.inverseLinkUpdateOption ) ) } diff --git a/Tests/SwiftletModelTests/QueriesTests/NestedModelsQueryTest.swift b/Tests/SwiftletModelTests/QueriesTests/NestedModelsQueryTest.swift new file mode 100644 index 0000000..c58b31b --- /dev/null +++ b/Tests/SwiftletModelTests/QueriesTests/NestedModelsQueryTest.swift @@ -0,0 +1,681 @@ +// +// File.swift +// +// +// Created by Sergey Kazakov on 02/08/2024. +// + +import Foundation +import XCTest +@testable import SwiftletModel + +final class NestedModelsQueryTest: XCTestCase { + var context = Context() + + override func setUpWithError() throws { + let chat = Chat( + id: "1", + users: .relation([.bob, .alice, .tom, .john, .michael]), + messages: .relation([ + Message( + id: "0", + text: "hello, ya'll", + author: .relation(.michael) + ), + + Message( + id: "1", + text: "hello", + author: .relation(.alice), + replyTo: .relation(id: "0") + ), + + Message( + id: "2", + text: "howdy", + author: .relation(.bob), + replyTo: .relation(id: "0") + ), + + Message( + id: "3", + text: "yo!", + author: .relation(.tom), + replyTo: .relation(id: "0") + ), + + Message( + id: "4", + text: "wassap!", + author: .relation(.john), + replyTo: .relation(id: "0") + ) + ]), + admins: .relation([.bob]) + ) + + try chat.save(to: &context) + } + + func test_WhenQueryWithNestedModel_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .plain + + let messages = Message + .query(in: context) + .with(\.$author) + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : { + "adminOf" : null, + "chats" : null, + "id" : "4", + "name" : "Michael" + }, + "chat" : null, + "id" : "0", + "replies" : null, + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "adminOf" : null, + "chats" : null, + "id" : "2", + "name" : "Alice" + }, + "chat" : null, + "id" : "1", + "replies" : null, + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "adminOf" : null, + "chats" : null, + "id" : "1", + "name" : "Bob" + }, + "chat" : null, + "id" : "2", + "replies" : null, + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "adminOf" : null, + "chats" : null, + "id" : "5", + "name" : "Tom" + }, + "chat" : null, + "id" : "3", + "replies" : null, + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "adminOf" : null, + "chats" : null, + "id" : "3", + "name" : "John" + }, + "chat" : null, + "id" : "4", + "replies" : null, + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } + + func test_WhenQueryWithNestedModelId_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .plain + + let messages = Message + .query(in: context) + .id(\.$author) + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : { + "id" : "4" + }, + "chat" : null, + "id" : "0", + "replies" : null, + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "id" : "2" + }, + "chat" : null, + "id" : "1", + "replies" : null, + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "id" : "1" + }, + "chat" : null, + "id" : "2", + "replies" : null, + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "id" : "5" + }, + "chat" : null, + "id" : "3", + "replies" : null, + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : { + "id" : "3" + }, + "chat" : null, + "id" : "4", + "replies" : null, + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } + + func test_WhenQueryWithNestedModelIds_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .plain + + let messages = Message + .query(in: context) + .ids(\.$replies) + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "0", + "replies" : [ + { + "id" : "1" + }, + { + "id" : "2" + }, + { + "id" : "3" + }, + { + "id" : "4" + } + ], + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : [ + + ], + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : [ + + ], + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : [ + + ], + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : [ + + ], + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } + + func test_WhenQueryWithNestedModels_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .plain + + let messages = Message + .query(in: context) + .with(\.$replies) { + $0.id(\.$replyTo) + } + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "0", + "replies" : [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "wassap!", + "viewedBy" : null + } + ], + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : [ + + ], + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : [ + + ], + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : [ + + ], + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : [ + + ], + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } + + func test_WhenQueryWithNestedModelsFragment_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .explicitKeyedContainer + + let messages = Message + .query(in: context) + .with(fragment: \.$replies) { + $0.id(\.$replyTo) + } + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "0", + "replies" : { + "fragment" : [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : null, + "replyTo" : { + "id" : "0" + }, + "text" : "wassap!", + "viewedBy" : null + } + ] + }, + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : { + "fragment" : [ + + ] + }, + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : { + "fragment" : [ + + ] + }, + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : { + "fragment" : [ + + ] + }, + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : { + "fragment" : [ + + ] + }, + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } + + func test_WhenQueryWithNestedIdsFragment_EqualExpectedJSON() { + let encoder = JSONEncoder.prettyPrinting + encoder.relationEncodingStrategy = .explicitKeyedContainer + + let messages = Message + .query(in: context) + .ids(fragment: \.$replies) + .resolve() + .sorted(by: { $0.id < $1.id}) + + let expectedJSON = """ + [ + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "0", + "replies" : { + "fragment_ids" : [ + "1", + "2", + "3", + "4" + ] + }, + "replyTo" : null, + "text" : "hello, ya'll", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "1", + "replies" : { + "fragment_ids" : [ + + ] + }, + "replyTo" : null, + "text" : "hello", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "2", + "replies" : { + "fragment_ids" : [ + + ] + }, + "replyTo" : null, + "text" : "howdy", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "3", + "replies" : { + "fragment_ids" : [ + + ] + }, + "replyTo" : null, + "text" : "yo!", + "viewedBy" : null + }, + { + "attachment" : null, + "author" : null, + "chat" : null, + "id" : "4", + "replies" : { + "fragment_ids" : [ + + ] + }, + "replyTo" : null, + "text" : "wassap!", + "viewedBy" : null + } + ] + """ + + let json = messages.prettyDescription(with: encoder)! + XCTAssertEqual(json, expectedJSON) + } +} +