diff --git a/src/js_object_accessor.hpp b/src/js_object_accessor.hpp index 37448925..cccf6e23 100644 --- a/src/js_object_accessor.hpp +++ b/src/js_object_accessor.hpp @@ -120,10 +120,14 @@ struct NativeAccessor { auto child = realm::Object::create(ctx, realm, *object_schema, static_cast(object), try_update); return child.row().get_index(); } - static size_t to_existing_object_index(ContextType ctx, ValueType &value) { + static size_t to_existing_object_index(ContextType ctx, SharedRealm realm, ValueType &value) { ObjectType object = Value::validated_to_object(ctx, value); if (Object::template is_instance>(ctx, object)) { - return get_internal>(object)->row().get_index(); + auto realm_object = get_internal>(object); + if (realm_object->realm() == realm) { + return realm_object->row().get_index(); + } + throw std::runtime_error("Realm object is from another Realm"); } throw std::runtime_error("object is not a Realm Object"); } diff --git a/src/js_results.hpp b/src/js_results.hpp index 640d67fb..97459bbd 100644 --- a/src/js_results.hpp +++ b/src/js_results.hpp @@ -126,7 +126,7 @@ typename T::Object ResultsClass::create_filtered(ContextType ctx, const U &co } parser::Predicate predicate = parser::parse(query_string); - query_builder::ArgumentConverter converter(ctx, args); + query_builder::ArgumentConverter converter(ctx, realm, args); query_builder::apply_predicate(query, predicate, converter, *realm->config().schema, object_schema.name); return create_instance(ctx, realm, object_schema, std::move(query)); diff --git a/src/object-store/src/object_accessor.hpp b/src/object-store/src/object_accessor.hpp index c20282b6..61360f10 100644 --- a/src/object-store/src/object_accessor.hpp +++ b/src/object-store/src/object_accessor.hpp @@ -101,7 +101,7 @@ namespace realm { static ValueType from_object(ContextType ctx, Object); // object index for an existing object - static size_t to_existing_object_index(ContextType ctx, ValueType &val); + static size_t to_existing_object_index(ContextType ctx, SharedRealm realm, ValueType &val); // list value acessors static size_t list_size(ContextType ctx, ValueType &val); diff --git a/src/object-store/src/parser/query_builder.hpp b/src/object-store/src/parser/query_builder.hpp index f9dba1bd..ae491072 100644 --- a/src/object-store/src/parser/query_builder.hpp +++ b/src/object-store/src/parser/query_builder.hpp @@ -50,7 +50,8 @@ class Arguments { template class ArgumentConverter : public Arguments { public: - ArgumentConverter(ContextType context, std::vector arguments) : m_arguments(arguments), m_ctx(context) {} + ArgumentConverter(ContextType context, SharedRealm realm, std::vector arguments) + : m_arguments(arguments), m_ctx(context), m_realm(std::move(realm)) {} using Accessor = realm::NativeAccessor; virtual bool bool_for_argument(size_t argument_index) { return Accessor::to_bool(m_ctx, argument_at(argument_index)); } @@ -60,12 +61,13 @@ class ArgumentConverter : public Arguments { virtual std::string string_for_argument(size_t argument_index) { return Accessor::to_string(m_ctx, argument_at(argument_index)); } virtual std::string binary_for_argument(size_t argument_index) { return Accessor::to_binary(m_ctx, argument_at(argument_index)); } virtual Timestamp timestamp_for_argument(size_t argument_index) { return Accessor::to_timestamp(m_ctx, argument_at(argument_index)); } - virtual size_t object_index_for_argument(size_t argument_index) { return Accessor::to_existing_object_index(m_ctx, argument_at(argument_index)); } + virtual size_t object_index_for_argument(size_t argument_index) { return Accessor::to_existing_object_index(m_ctx, m_realm, argument_at(argument_index)); } virtual bool is_argument_null(size_t argument_index) { return Accessor::is_null(m_ctx, argument_at(argument_index)); } private: std::vector m_arguments; ContextType m_ctx; + SharedRealm m_realm; ValueType &argument_at(size_t index) { if (index >= m_arguments.size()) { diff --git a/tests/js/results-tests.js b/tests/js/results-tests.js index 24b3822f..ad3636eb 100644 --- a/tests/js/results-tests.js +++ b/tests/js/results-tests.js @@ -130,6 +130,7 @@ module.exports = { testResultsFiltered: function() { var realm = new Realm({schema: [schemas.PersonObject, schemas.DefaultValues, schemas.TestObject]}); + realm.write(function() { realm.create('PersonObject', {name: 'Ari', age: 10}); realm.create('PersonObject', {name: 'Tim', age: 11}); @@ -155,9 +156,13 @@ module.exports = { TestCase.assertEqual(realm.objects('PersonObject').filtered('name = $0', 'Tim').length, 1); TestCase.assertEqual(realm.objects('PersonObject').filtered('age > $1 && age < $0', 13, 10).length, 3); + TestCase.assertThrows(function() { realm.objects('PersonObject').filtered('age > $2 && age < $0', 13, 10) }); + TestCase.assertThrows(function() { + realm.objects('PersonObject').filtered("invalidQuery"); + }); realm.write(function() { realm.create('DefaultValuesObject', {'dateCol': new Date(3)}); @@ -167,9 +172,19 @@ module.exports = { TestCase.assertEqual(realm.objects('DefaultValuesObject').filtered('dateCol > $0', new Date(4)).length, 1); TestCase.assertEqual(realm.objects('DefaultValuesObject').filtered('dateCol <= $0', new Date(4)).length, 2); + }, + + testResultsFilteredByForeignObject: function() { + var realm = new Realm({schema: [schemas.LinkTypes, schemas.TestObject]}); + var realm2 = new Realm({path: '2.realm', schema: realm.schema}); + var object; + + realm2.write(function() { + object = realm2.create('TestObject', {doubleCol: 1}); + }); TestCase.assertThrows(function() { - realm.objects('PersonObject').filtered("invalidQuery"); + realm.objects('LinkTypesObject').filtered('objectCol = $0', object); }); },