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

RCOCOA-2271: Collections in Mixed #8546

Merged
merged 12 commits into from
Jun 6, 2024
Merged

Conversation

dianaafanador3
Copy link
Contributor

@dianaafanador3 dianaafanador3 commented Apr 12, 2024

  • Added support for storing nested collections (List and Map not ManagedSet) in a AnyRealmValue.

    class MixedObject: Object {
      @Persisted var anyValue: AnyRealmValue
    }
    
    // You can build a AnyRealmValue from a Swift's Dictionary.
    let dictionary: Dictionary<String, AnyRealmValue> = ["key1": .string("hello"), "key2": .bool(false)]
    
    // You can build a AnyRealmValue from a Swift's Map
    // and nest a collection within another collection.
    let list: Array<AnyRealmValue> = [.int(12), .double(14.17), AnyRealmValue.fromDictionary(dictionary)]
    
    let realm = realmWithTestPath()
    try realm.write {
      let obj = MixedObject()
      obj.anyValue = AnyRealmValue.fromArray(list)
      realm.add(o)
    }
  • Added new operators to Swift's Query API for supporting querying nested collections.

    realm.objects(MixedObject.self).where { $0.anyValue[0][0][1] == .double(76.54) }

    The .all operator allows looking up in all keys or indexes.

    realm.objects(MixedObject.self).where { $0.anyValue["key"].all == .bool(false) }

@cla-bot cla-bot bot added the cla: yes label Apr 12, 2024
@dianaafanador3 dianaafanador3 marked this pull request as draft April 12, 2024 16:16
@dianaafanador3 dianaafanador3 force-pushed the dp/collections_in_mixed branch 8 times, most recently from b60f46f to 3386d7a Compare April 22, 2024 07:36
@dianaafanador3 dianaafanador3 marked this pull request as ready for review April 22, 2024 11:15
@dianaafanador3
Copy link
Contributor Author

This is waiting for this PR to be merged to core. Also, there is one failing test which is working locally, taking a look at it, but putting this on review on the meantime.

@dianaafanador3 dianaafanador3 changed the title Collections in Mixed RCOCOA-2271: Collections in Mixed Apr 22, 2024
@dianaafanador3 dianaafanador3 force-pushed the dp/collections_in_mixed branch 9 times, most recently from ed91d6b to 45a17bd Compare April 23, 2024 14:44
Copy link
Member

@tgoyne tgoyne left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A first pass over things.

Realm/RLMDictionary.h Outdated Show resolved Hide resolved
Realm/RLMAccessor.mm Outdated Show resolved Hide resolved
Realm/RLMAccessor.mm Outdated Show resolved Hide resolved
Realm/RLMArray_Private.hpp Show resolved Hide resolved
Realm/RLMConstants.h Outdated Show resolved Hide resolved
Realm/RLMAccessor.mm Outdated Show resolved Hide resolved
RealmSwift/AnyRealmValue.swift Outdated Show resolved Hide resolved
RealmSwift/List.swift Outdated Show resolved Hide resolved
@@ -779,6 +779,14 @@ public final class Map<Key: _MapKey, Value: RealmCollectionValue>: RLMSwiftColle
}
}

// MARK: - Hashable

extension Map: Hashable {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Similarly, this is not at all a valid Hashable implementation.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Using ObjectIdentifier given that we already conform to Equatable

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ObjectIdentifier is not a valid definition either. Two managed dictionary objects which correspond to the same field are equal to each other but have different ObjectIdentifiers. All of our accessor types are fundamentally incompatible with Hashable due to that their identity changes when you promote an unmanaged object.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@tgoyne we do need to conform to Hashable to be able to include Map or List as a an associated value in AnyRealmValue if we want to go that approach.
I can think of a few approaches for this:

  • Use ObjectIdentifer and sacrifice equality for managed and unmanaged object
  • Have associated values as Swift's native types Array/Dictionary, this will clearly will go bad with nested collections which will need to be instantiated completely to convert it (Discarded)
  • Store some reference to the collection as the associated value and then and have some static function that returns the collection (Not need to conform to hashable, but worse developer experience).

I'm open to discuss different solutions

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can override the hash function on AnyRealmValue and return a constant value for the Map and List cases.

RealmSwift/Tests/ObjectCreationTests.swift Outdated Show resolved Hide resolved
Realm/RLMAccessor.mm Outdated Show resolved Hide resolved
Realm/RLMAccessor.mm Show resolved Hide resolved
@@ -890,15 +914,17 @@ void RLMSetSwiftPropertyAny(__unsafe_unretained RLMObjectBase *const obj, uint16
}

id RLMAccessorContext::box(realm::Mixed v) {
return RLMMixedToObjc(v, _realm, &_info);
auto property = (currentProperty) ? currentProperty : _info.propertyForTableColumn(_colKey);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why would currentProperty be nil here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For any other mixed wrapped types besides list and dictionary the accessor is constructed without the current property because it is not needed to box the native value

Realm/RLMCollection.mm Outdated Show resolved Hide resolved
Realm/RLMCollection.mm Outdated Show resolved Hide resolved
Realm/RLMValue.h Outdated Show resolved Hide resolved
Realm/Tests/KVOTests.mm Outdated Show resolved Hide resolved
Realm/Tests/NotificationTests.m Show resolved Hide resolved
Realm/Tests/ObjectTests.m Outdated Show resolved Hide resolved
RealmSwift/AnyRealmValue.swift Outdated Show resolved Hide resolved
@tgoyne tgoyne mentioned this pull request May 21, 2024
Copy link
Member

@tgoyne tgoyne left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There are still several new warnings. I'm not sure why the implicit casts from id to RLMArray are producing warnings here but not on master, but the other warnings seem straightforward.

CHANGELOG.md Outdated Show resolved Hide resolved
Realm/RLMQueryUtil.mm Outdated Show resolved Hide resolved
Realm/RLMQueryUtil.mm Outdated Show resolved Hide resolved
Realm/RLMQueryUtil.mm Outdated Show resolved Hide resolved
Realm/RLMQueryUtil.mm Outdated Show resolved Hide resolved
}
/// :nodoc:
public subscript(key: String) -> Query<AnyRealmValue> {
.init(appendKeyPath("['\(key)']", options: [.isPath]))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you do $0.anyCol["'"] == 0 to query on the dictionary key "'" this will produce invalid syntax.

RealmSwift/Query.swift Outdated Show resolved Hide resolved
RealmSwift/RealmCollection.swift Outdated Show resolved Hide resolved
RealmSwift/ObjectiveCSupport+AnyRealmValue.swift Outdated Show resolved Hide resolved
@dianaafanador3 dianaafanador3 force-pushed the dp/collections_in_mixed branch 3 times, most recently from 3683365 to 98a39c3 Compare May 22, 2024 23:36
@dianaafanador3 dianaafanador3 force-pushed the dp/collections_in_mixed branch from 98a39c3 to 2a3861f Compare May 22, 2024 23:43
@dianaafanador3 dianaafanador3 requested a review from tgoyne May 22, 2024 23:43
@dianaafanador3
Copy link
Contributor Author

@tgoyne solved PR comments and removed the warnings

Realm/RLMSwiftValueStorage.mm Outdated Show resolved Hide resolved
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
- (RLMPropertyType)rlm_valueType {
REALM_UNREACHABLE();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Returning RLMPropertyTypeAny seems like the best option here.

REALM_UNREACHABLE() marks a place that should be logically impossible to hit and if we get there it always indicates a bug in our code.

Realm/Tests/NotificationTests.m Outdated Show resolved Hide resolved
Realm/Tests/QueryTests.m Outdated Show resolved Hide resolved
RealmSwift/AnyRealmValue.swift Outdated Show resolved Hide resolved
RealmSwift/Object.swift Outdated Show resolved Hide resolved
}
/// :nodoc:
public subscript(key: String) -> Query<AnyRealmValue> {
.init(appendKeyPath("['\(key)']", options: [.isPath]))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know what you mean. any["'"] == %@ (or any['\''] == %@) is a perfectly valid query, which this currently fails to generate a valid string for.

RealmSwift/RealmCollection.swift Outdated Show resolved Hide resolved
RealmSwift/Tests/MixedCollectionTest.swift Outdated Show resolved Hide resolved
@dianaafanador3 dianaafanador3 requested a review from tgoyne May 27, 2024 22:42
Realm/RLMQueryUtil.mm Outdated Show resolved Hide resolved
@dianaafanador3 dianaafanador3 requested a review from tgoyne May 30, 2024 06:56
RealmSwift/Query.swift Outdated Show resolved Hide resolved
@dianaafanador3 dianaafanador3 requested a review from tgoyne June 4, 2024 09:00
public func hash(into hasher: inout Hasher) {
switch self {
case let .int(i):
hasher.combine(i.hashValue)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These should all be hasher.combine(i) without the .hashValue. This currently is hashing the values and then rehashing the hash.

@dianaafanador3 dianaafanador3 merged commit 023f713 into master Jun 6, 2024
117 of 123 checks passed
@dianaafanador3 dianaafanador3 deleted the dp/collections_in_mixed branch June 6, 2024 09:58
@github-actions github-actions bot locked as resolved and limited conversation to collaborators Jul 6, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants