Defaults/Tests/DefaultsTests/DefaultsTests.swift

596 lines
17 KiB
Swift

// swiftlint:disable discouraged_optional_boolean
import Foundation
import Combine
import Testing
@testable import Defaults
func createSuite() -> UserDefaults {
UserDefaults(suiteName: UUID().uuidString)!
}
private let suite_ = createSuite()
let fixtureURL = URL(string: "https://sindresorhus.com")!
let fixtureFileURL = URL(string: "file://~/icon.png")!
let fixtureURL2 = URL(string: "https://example.com")!
let fixtureDate = Date()
extension Defaults.Keys {
static let key = Key<Bool>("key", default: false, suite: suite_)
static let url = Key<URL>("url", default: fixtureURL, suite: suite_)
static let file = Key<URL>("fileURL", default: fixtureFileURL, suite: suite_)
static let data = Key<Data>("data", default: Data([]), suite: suite_)
static let date = Key<Date>("date", default: fixtureDate, suite: suite_)
static let uuid = Key<UUID?>("uuid", suite: suite_)
static let defaultDynamicDate = Key<Date>("defaultDynamicOptionalDate", suite: suite_) { Date(timeIntervalSince1970: 0) }
static let defaultDynamicOptionalDate = Key<Date?>("defaultDynamicOptionalDate", suite: suite_) { Date(timeIntervalSince1970: 1) }
}
@Suite(.serialized)
final class DefaultsTests {
init() {
Defaults.removeAll(suite: suite_)
}
deinit {
Defaults.removeAll(suite: suite_)
}
@Test
func testKey() {
let key = Defaults.Key<Bool>("independentKey", default: false, suite: suite_)
#expect(!Defaults[key])
Defaults[key] = true
#expect(Defaults[key])
}
@Test
func testValidKeyName() {
let validKey = Defaults.Key<Bool>("test", default: false, suite: suite_)
let containsDotKey = Defaults.Key<Bool>("test.a", default: false, suite: suite_)
let startsWithAtKey = Defaults.Key<Bool>("@test", default: false, suite: suite_)
#expect(Defaults.isValidKeyPath(name: validKey.name))
#expect(!Defaults.isValidKeyPath(name: containsDotKey.name))
#expect(!Defaults.isValidKeyPath(name: startsWithAtKey.name))
}
@Test
func testOptionalKey() {
let key = Defaults.Key<Bool?>("independentOptionalKey", suite: suite_)
let url = Defaults.Key<URL?>("independentOptionalURLKey", suite: suite_)
#expect(Defaults[key] == nil)
#expect(Defaults[url] == nil)
Defaults[key] = true
Defaults[url] = fixtureURL
#expect(Defaults[key] == true)
#expect(Defaults[url] == fixtureURL)
Defaults[key] = nil
Defaults[url] = nil
#expect(Defaults[key] == nil)
#expect(Defaults[url] == nil)
Defaults[key] = false
Defaults[url] = fixtureURL2
#expect(Defaults[key] == false)
#expect(Defaults[url] == fixtureURL2)
}
@Test
func testInitializeDynamicDateKey() {
_ = Defaults.Key<Date>("independentInitializeDynamicDateKey", suite: suite_) {
Issue.record("Init dynamic key should not trigger getter")
return Date()
}
_ = Defaults.Key<Date?>("independentInitializeDynamicOptionalDateKey", suite: suite_) {
Issue.record("Init dynamic optional key should not trigger getter")
return Date()
}
}
@Test
func testKeyRegistersDefault() {
let keyName = "registersDefault"
#expect(!suite_.bool(forKey: keyName))
_ = Defaults.Key<Bool>(keyName, default: true, suite: suite_)
#expect(suite_.bool(forKey: keyName))
let keyName2 = "registersDefault2"
_ = Defaults.Key<String>(keyName2, default: keyName2, suite: suite_)
#expect(suite_.string(forKey: keyName2) == keyName2)
}
@Test
func testKeyWithUserDefaultSubscript() {
let key = Defaults.Key<Bool>("keyWithUserDeaultSubscript", default: false, suite: suite_)
#expect(!suite_[key])
suite_[key] = true
#expect(suite_[key])
}
@Test
func testKeys() {
#expect(!Defaults[.key])
Defaults[.key] = true
#expect(Defaults[.key])
}
@Test
func testUrlType() {
#expect(Defaults[.url] == fixtureURL)
let newUrl = URL(string: "https://twitter.com")!
Defaults[.url] = newUrl
#expect(Defaults[.url] == newUrl)
}
@Test
func testDataType() {
#expect(Defaults[.data] == Data([]))
let newData = Data([0xFF])
Defaults[.data] = newData
#expect(Defaults[.data] == newData)
}
@Test
func testDateType() {
#expect(Defaults[.date] == fixtureDate)
let newDate = Date()
Defaults[.date] = newDate
#expect(Defaults[.date] == newDate)
}
@Test
func testDynamicDateType() {
#expect(Defaults[.defaultDynamicDate] == Date(timeIntervalSince1970: 0))
let next = Date(timeIntervalSince1970: 1)
Defaults[.defaultDynamicDate] = next
#expect(Defaults[.defaultDynamicDate] == next)
#expect(suite_.object(forKey: Defaults.Key<Date>.defaultDynamicDate.name) as! Date == next)
Defaults.Key<Date>.defaultDynamicDate.reset()
#expect(Defaults[.defaultDynamicDate] == Date(timeIntervalSince1970: 0))
}
@Test
func testDynamicOptionalDateType() {
#expect(Defaults[.defaultDynamicOptionalDate] == Date(timeIntervalSince1970: 1))
let next = Date(timeIntervalSince1970: 2)
Defaults[.defaultDynamicOptionalDate] = next
#expect(Defaults[.defaultDynamicOptionalDate] == next)
#expect(suite_.object(forKey: Defaults.Key<Date>.defaultDynamicOptionalDate.name) as! Date == next)
Defaults[.defaultDynamicOptionalDate] = nil
#expect(Defaults[.defaultDynamicOptionalDate] == Date(timeIntervalSince1970: 1))
#expect(suite_.object(forKey: Defaults.Key<Date>.defaultDynamicOptionalDate.name) == nil)
}
@Test
func testFileURLType() {
#expect(Defaults[.file] == fixtureFileURL)
}
@Test
func testUUIDType() {
let fixture = UUID()
Defaults[.uuid] = fixture
#expect(Defaults[.uuid] == fixture)
}
@Test
func testRemoveAll() {
let key = Defaults.Key<Bool>("removeAll", default: false, suite: suite_)
let key2 = Defaults.Key<Bool>("removeAll2", default: false, suite: suite_)
Defaults[key] = true
Defaults[key2] = true
#expect(Defaults[key])
#expect(Defaults[key2])
Defaults.removeAll(suite: suite_)
#expect(!Defaults[key])
#expect(!Defaults[key2])
}
@Test
func testCustomSuite() {
let customSuite = UserDefaults(suiteName: "com.sindresorhus.customSuite")!
let key = Defaults.Key<Bool>("customSuite", default: false, suite: customSuite)
#expect(!customSuite[key])
#expect(!Defaults[key])
Defaults[key] = true
#expect(customSuite[key])
#expect(Defaults[key])
Defaults.removeAll(suite: customSuite)
}
@Test
func testIsDefaultValue() {
let key = Defaults.Key<Bool>("isDefaultValue", default: false, suite: suite_)
#expect(key.isDefaultValue)
Defaults[key].toggle()
#expect(!key.isDefaultValue)
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testObserveKeyCombine() async throws {
let key = Defaults.Key<Bool>("observeKey", default: false, suite: suite_)
var results: [(Bool, Bool)] = []
try await confirmation(expectedCount: 2) { confirmation in
let publisher = Defaults
.publisher(key, options: [])
.map { ($0.oldValue, $0.newValue) }
.sink { value in
results.append(value)
confirmation()
}
await Task.yield()
Defaults[key] = true
Defaults.reset(key)
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
let expectedValues = [(false, true), (true, false)]
// #expect(results == expectedValues)
#expect(results.count == expectedValues.count)
for (index, expected) in expectedValues.enumerated() {
#expect(results[index].0 == expected.0)
#expect(results[index].1 == expected.1)
}
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testObserveOptionalKeyCombine() async throws {
let key = Defaults.Key<Bool?>("observeOptionalKey", suite: suite_)
var results: [(Bool?, Bool?)] = []
try await confirmation(expectedCount: 3) { confirmation in
let publisher = Defaults
.publisher(key, options: [])
.map { ($0.oldValue, $0.newValue) }
.sink { value in
results.append(value)
confirmation()
}
await Task.yield()
Defaults[key] = true
Defaults[key] = false
Defaults.reset(key)
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
let expectedValues: [(Bool?, Bool?)] = [(nil, true), (true, false), (false, nil)]
// #expect(results == expectedValues)
#expect(results.count == expectedValues.count)
for (index, expected) in expectedValues.enumerated() {
#expect(results[index].0 == expected.0)
#expect(results[index].1 == expected.1)
}
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testDynamicOptionalDateTypeCombine() async throws {
let first = Date(timeIntervalSince1970: 0)
let second = Date(timeIntervalSince1970: 1)
let third = Date(timeIntervalSince1970: 2)
let key = Defaults.Key<Date?>("combineDynamicOptionalDateKey", suite: suite_) { first }
var results: [(Date?, Date?)] = []
try await confirmation(expectedCount: 3) { confirmation in
let publisher = Defaults
.publisher(key, options: [])
.map { ($0.oldValue, $0.newValue) }
.sink { value in
results.append(value)
confirmation()
}
await Task.yield()
Defaults[key] = second
Defaults[key] = third
Defaults.reset(key)
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
let expectedValues: [(Date?, Date?)] = [(first, second), (second, third), (third, first)]
// #expect(results == expectedValues)
#expect(results.count == expectedValues.count)
for (index, expected) in expectedValues.enumerated() {
#expect(results[index].0 == expected.0)
#expect(results[index].1 == expected.1)
}
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testObserveMultipleKeysCombine() async throws {
let key1 = Defaults.Key<String>("observeKey1", default: "x", suite: suite_)
let key2 = Defaults.Key<Bool>("observeKey2", default: true, suite: suite_)
var count = 0
try await confirmation(expectedCount: 2) { confirmation in
let publisher = Defaults.publisher(keys: key1, key2, options: [])
.sink { _ in
count += 1
confirmation()
}
await Task.yield()
Defaults[key1] = "y"
Defaults[key2] = false
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
#expect(count == 2)
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testObserveMultipleOptionalKeysCombine() async throws {
let key1 = Defaults.Key<String?>("observeOptionalKey1", suite: suite_)
let key2 = Defaults.Key<Bool?>("observeOptionalKey2", suite: suite_)
var count = 0
try await confirmation(expectedCount: 2) { confirmation in
let publisher = Defaults.publisher(keys: key1, key2, options: [])
.sink { _ in
count += 1
confirmation()
}
await Task.yield()
Defaults[key1] = "x"
Defaults[key2] = false
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
#expect(count == 2)
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testReceiveValueBeforeSubscriptionCombine() async throws {
let key = Defaults.Key<String>("receiveValueBeforeSubscription", default: "hello", suite: suite_)
try await confirmation(expectedCount: 2) { confirmation in
let publisher = Defaults
.publisher(key)
.map(\.newValue)
.sink { value in
confirmation()
print("Received value: \(value)")
}
// Ensure we're subscribed before changing the value
await Task.yield()
// Change the value
Defaults[key] = "world"
// Keep the subscription alive
try await Task.sleep(for: .milliseconds(100))
publisher.cancel()
}
}
@Test
func testObservePreventPropagationCombine() async throws {
let key1 = Defaults.Key<Bool?>("preventPropagation6", default: nil, suite: suite_)
await confirmation() { confirmation in
var wasInside = false
let cancellable = Defaults.publisher(key1, options: []).sink { _ in
#expect(!wasInside)
wasInside = true
Defaults.withoutPropagation {
Defaults[key1] = true
}
confirmation()
}
Defaults[key1] = false
cancellable.cancel()
}
}
@Test
func testObservePreventPropagationMultipleKeysCombine() async throws {
let key1 = Defaults.Key<Bool?>("preventPropagation7", default: nil, suite: suite_)
let key2 = Defaults.Key<Bool?>("preventPropagation8", default: nil, suite: suite_)
await confirmation() { confirmation in
var wasInside = false
let cancellable = Defaults.publisher(keys: key1, key2, options: []).sink { _ in
#expect(!wasInside)
wasInside = true
Defaults.withoutPropagation {
Defaults[key1] = true
}
confirmation()
}
Defaults[key2] = false
cancellable.cancel()
}
}
@Test
func testResetKey() {
let defaultFixture1 = "foo1"
let defaultFixture2 = 0
let newFixture1 = "bar1"
let newFixture2 = 1
let key1 = Defaults.Key<String>("key1", default: defaultFixture1, suite: suite_)
let key2 = Defaults.Key<Int>("key2", default: defaultFixture2, suite: suite_)
Defaults[key1] = newFixture1
Defaults[key2] = newFixture2
Defaults.reset(key1)
#expect(Defaults[key1] == defaultFixture1)
#expect(Defaults[key2] == newFixture2)
}
@Test
func testResetMultipleKeys() {
let defaultFxiture1 = "foo1"
let defaultFixture2 = 0
let defaultFixture3 = "foo3"
let newFixture1 = "bar1"
let newFixture2 = 1
let newFixture3 = "bar3"
let key1 = Defaults.Key<String>("akey1", default: defaultFxiture1, suite: suite_)
let key2 = Defaults.Key<Int>("akey2", default: defaultFixture2, suite: suite_)
let key3 = Defaults.Key<String>("akey3", default: defaultFixture3, suite: suite_)
Defaults[key1] = newFixture1
Defaults[key2] = newFixture2
Defaults[key3] = newFixture3
Defaults.reset(key1, key2)
#expect(Defaults[key1] == defaultFxiture1)
#expect(Defaults[key2] == defaultFixture2)
#expect(Defaults[key3] == newFixture3)
}
@Test
func testResetMultipleOptionalKeys() {
let newFixture1 = "bar1"
let newFixture2 = 1
let newFixture3 = "bar3"
let key1 = Defaults.Key<String?>("aoptionalKey1", suite: suite_)
let key2 = Defaults.Key<Int?>("aoptionalKey2", suite: suite_)
let key3 = Defaults.Key<String?>("aoptionalKey3", suite: suite_)
Defaults[key1] = newFixture1
Defaults[key2] = newFixture2
Defaults[key3] = newFixture3
Defaults.reset(key1, key2)
#expect(Defaults[key1] == nil)
#expect(Defaults[key2] == nil)
#expect(Defaults[key3] == newFixture3)
}
@available(iOS 15.0, macOS 12.0, tvOS 15.0, watchOS 8.0, visionOS 1.0, *)
@Test
func testImmediatelyFinishingPublisherCombine() async throws {
let key = Defaults.Key<Bool>("observeKey", default: false, suite: suite_)
let result: Void? = try await withThrowingTaskGroup(of: Void.self) { group in
let publisher = Defaults
.publisher(key, options: [.initial])
.first()
group.addTask {
for try await _ in publisher.values {
return
}
}
return try await group.next()
}
#expect(result != nil)
}
@available(iOS 15.0, macOS 12.0, tvOS 15.0, watchOS 8.0, visionOS 1.0, *)
@Test
func testImmediatelyFinishingMultiplePublisherCombine() async throws {
let key1 = Defaults.Key<Bool>("observeKey1", default: false, suite: suite_)
let key2 = Defaults.Key<String>("observeKey2", default: "🦄", suite: suite_)
let result: Void? = try await withThrowingTaskGroup(of: Void.self) { group in
let publisher = Defaults
.publisher(keys: [key1, key2], options: [.initial])
.first()
group.addTask {
for try await _ in publisher.values {
return
}
}
return try await group.next()
}
#expect(result != nil)
}
@Test
func testKeyEquatable() {
#expect(Defaults.Key<Bool>("equatableKeyTest", default: false, suite: suite_) == Defaults.Key<Bool>("equatableKeyTest", default: false, suite: suite_))
}
@Test
func testKeyHashable() {
_ = Set([Defaults.Key<Bool>("hashableKeyTest", default: false, suite: suite_)])
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testUpdates() async {
let key = Defaults.Key<Bool>("updatesKey", default: false, suite: suite_)
async let waiter = Defaults.updates(key, initial: false).first { $0 }
try? await Task.sleep(for: .seconds(0.1))
Defaults[key] = true
guard let result = await waiter else {
Issue.record("Failed to get result")
return
}
#expect(result)
}
@available(iOS 16.0, macOS 13.0, tvOS 16.0, watchOS 9.0, visionOS 1.0, *)
@Test
func testUpdatesMultipleKeys() async {
let key1 = Defaults.Key<Bool>("updatesMultipleKey1", default: false, suite: suite_)
let key2 = Defaults.Key<Bool>("updatesMultipleKey2", default: false, suite: suite_)
let counter = Counter()
async let waiter: Void = {
for await _ in Defaults.updates([key1, key2], initial: false) {
await counter.increment()
if await counter.count == 2 {
break
}
}
}()
try? await Task.sleep(for: .seconds(0.1))
Defaults[key1] = true
Defaults[key2] = true
await waiter
let count = await counter.count
#expect(count == 2)
}
}
actor Counter {
private var _count = 0
var count: Int { _count }
func increment() {
_count += 1
}
}
// swiftlint:enable discouraged_optional_boolean