swift-libs/Tests/Persistence/Classes/FetcherTests.swift
Javier Cicchelli c0a49b2a85 [Feature] Persistence (#5)
This PR contains the work done to implement some useful protocols, classes and extensions to use to setup a persistence layer in an application.

To provide further details about the work done:
 - [x] declared the `Persistence` target in the Package file;
 - [x] forgot to declare the `Communications` and `Persistence` target to the `SwiftLibs` library in the `Package` file;
 - [x] defined the `Service` public protocol;
 - [x] implemented the `Fetcher` generic class;
 - [x] implemented the `bitBucket` static property in the `URL+Devices` public extension;
 - [x] updated the `README` file.

Co-authored-by: Javier Cicchelli <javier@rock-n-code.com>
Reviewed-on: #5
2023-04-17 23:12:11 +00:00

320 lines
8.5 KiB
Swift

//
// FetcherTests.swift
// PersistenceTests
//
// Created by Javier Cicchelli on 17/04/2023.
// Copyright © 2023 Röck+Cöde. All rights reserved.
//
import Combine
import Persistence
import XCTest
final class FetcherTests: XCTestCase {
// MARK: Properties
private lazy var persistence: TestPersistenceService = .shared
private lazy var fetcher: Fetcher<TestEntity> = .init(
fetchRequest: .allTestEntities(),
managedObjectContext: persistence.viewContext
)
// MARK: Setup
override func tearDownWithError() throws {
try persistence.clean()
}
// MARK: Number of sections tests
func test_numberOfSections_whenModelIsEmpty() throws {
// GIVEN
try fetcher.fetch()
// WHEN
let numberOfSections = fetcher.numberOfSections
// THEN
XCTAssertEqual(numberOfSections, 1)
}
func test_numberOfSections_whenModelIsFilled() throws {
// GIVEN
let context = persistence.makeChildContext()
let _ = [
TestEntity(context: context),
TestEntity(context: context),
TestEntity(context: context)
]
try persistence.save(childContext: context)
try fetcher.fetch()
// WHEN
let numberOfSections = fetcher.numberOfSections
// THEN
XCTAssertEqual(numberOfSections, 1)
}
func test_numberOfSections_whenNoFetch() async throws {
// GIVEN
// WHEN
let numberOfSections = fetcher.numberOfSections
// THEN
XCTAssertEqual(numberOfSections, 0)
}
// MARK: Number of objects tests
func test_numberOfObjects_inFirstSection_whenModelIsEmpty() throws {
// GIVEN
try fetcher.fetch()
// WHEN
let section = fetcher.numberOfSections - 1
let numberOfObjects = try fetcher.numberOfObjects(in: section)
// THEN
XCTAssertEqual(numberOfObjects, 0)
}
func test_numberOfObjects_inFirstSection_whenModelIsFilled() throws {
// GIVEN
let context = persistence.makeChildContext()
let entities = [
TestEntity(context: context),
TestEntity(context: context),
TestEntity(context: context)
]
try persistence.save(childContext: context)
try fetcher.fetch()
// WHEN
let section = fetcher.numberOfSections - 1
let numberOfObjects = try fetcher.numberOfObjects(in: section)
// THEN
XCTAssertEqual(numberOfObjects, entities.count)
}
func test_numberOfObjects_inNonExistingSection() throws {
// GIVEN
try fetcher.fetch()
// WHEN & THEN
let section = fetcher.numberOfSections
XCTAssertThrowsError(try fetcher.numberOfObjects(in: section)) { error in
XCTAssertEqual(error as? FetcherError, .sectionNotFound)
}
}
func test_numberOfObjects_whenNoFetch() throws {
// GIVEN
// WHEN & THEN
XCTAssertThrowsError(try fetcher.numberOfObjects(in: 1)) { error in
XCTAssertEqual(error as? FetcherError, .fetchNotExecuted)
}
}
// MARK: Object at tests
func test_objectAt_whenModelIsEmpty() throws {
// GIVEN
try fetcher.fetch()
// WHEN & THEN
let _ = IndexPath(
item: 0,
section: fetcher.numberOfSections - 1
)
// TODO: Need to find out how to handle NSInvalidArgumentException in this test.
// let object = try fetcher.object(at: indexPath)
}
func test_objectAt_whenModelIsFilled() throws {
// GIVEN
let context = persistence.makeChildContext()
let entities = [TestEntity(context: context)]
try persistence.save(childContext: context)
try fetcher.fetch()
// WHEN & THEN
let indexPath = IndexPath(
item: entities.count - 1,
section: fetcher.numberOfSections - 1
)
let object = try fetcher.object(at: indexPath)
XCTAssertNotNil(object)
}
func test_objectAt_withOutOfBoundsIndexPath() throws {
// GIVEN
let context = persistence.makeChildContext()
let entities = [TestEntity(context: context)]
try persistence.save(childContext: context)
try fetcher.fetch()
// WHEN & THEN
let _ = IndexPath(
item: entities.count,
section: fetcher.numberOfSections
)
// TODO: Need to find out how to handle NSInvalidArgumentException in this test.
// let object = try fetcher.object(at: indexPath)
}
func test_objectAt_whenNoFetch() throws {
// GIVEN
// WHEN & THEN
let indexPath = IndexPath(
item: 0,
section: 0
)
XCTAssertThrowsError(try fetcher.object(at: indexPath)) { error in
XCTAssertEqual(error as? FetcherError, .fetchNotExecuted)
}
}
// MARK: Did change publisher tests
func test_didChangePublisher_whenModelIsEmpty() throws {
let expectation = self.expectation(description: "didChangePublisher when model is filled.")
var result: [Change]?
// GIVEN
let cancellable = fetcher
.didChangePublisher
.sink(receiveValue: { value in
result = value
expectation.fulfill()
})
// WHEN
try fetcher.fetch()
// THEN
let waiter = XCTWaiter.wait(for: [expectation], timeout: 1.0)
guard waiter == .timedOut else {
XCTFail("Waiter expected to time out.")
return
}
cancellable.cancel()
XCTAssertNil(result)
}
func test_didChangePublisher_whenModelIsFilled() throws {
let expectation = self.expectation(description: "didChangePublisher when model is filled.")
var result: [Change]?
// GIVEN
let context = persistence.makeChildContext()
let _ = [
TestEntity(context: context),
TestEntity(context: context),
TestEntity(context: context)
]
let cancellable = fetcher
.didChangePublisher
.sink(receiveValue: { value in
result = value
expectation.fulfill()
})
// WHEN
try persistence.save(childContext: context)
try fetcher.fetch()
// THEN
let waiter = XCTWaiter.wait(for: [expectation], timeout: 1.0)
guard waiter == .timedOut else {
XCTFail("Waiter expected to time out.")
return
}
cancellable.cancel()
XCTAssertNil(result)
}
func test_didChangePublisher_whenModelIsUpdated() throws {
let expectation = self.expectation(description: "didChangePublisher when model is updated.")
var result: [Change]?
// GIVEN
let context = persistence.makeChildContext()
let entities = [
TestEntity(context: context),
TestEntity(context: context),
TestEntity(context: context)
]
let cancellable = fetcher
.didChangePublisher
.sink(receiveValue: { value in
result = value
expectation.fulfill()
})
// WHEN
try fetcher.fetch()
try persistence.save(childContext: context)
// THEN
waitForExpectations(timeout: 1.0)
cancellable.cancel()
XCTAssertNotNil(result)
XCTAssertEqual(result?.count, entities.count)
XCTAssertEqual(result, [
.object(.inserted(at: .init(item: 2, section: 0))),
.object(.inserted(at: .init(item: 1, section: 0))),
.object(.inserted(at: .init(item: 0, section: 0))),
])
}
}
// MARK: - TestPersistenceService+Functions
private extension TestPersistenceService {
// MARK: Functions
func clean() throws {
let context = makeChildContext()
try context.performAndWait {
try context
.fetch(.allTestEntities())
.forEach(context.delete)
}
try save(childContext: context)
}
}