FIRQuery.m 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. /*
  2. * Copyright 2017 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import "FIRQuery.h"
  17. #import "FIRDocumentReference.h"
  18. #import "Firestore/Source/API/FIRDocumentReference+Internal.h"
  19. #import "Firestore/Source/API/FIRDocumentSnapshot+Internal.h"
  20. #import "Firestore/Source/API/FIRFieldPath+Internal.h"
  21. #import "Firestore/Source/API/FIRFirestore+Internal.h"
  22. #import "Firestore/Source/API/FIRListenerRegistration+Internal.h"
  23. #import "Firestore/Source/API/FIRQuery+Internal.h"
  24. #import "Firestore/Source/API/FIRQuerySnapshot+Internal.h"
  25. #import "Firestore/Source/API/FIRQuery_Init.h"
  26. #import "Firestore/Source/API/FIRSnapshotMetadata+Internal.h"
  27. #import "Firestore/Source/API/FSTUserDataConverter.h"
  28. #import "Firestore/Source/Core/FSTEventManager.h"
  29. #import "Firestore/Source/Core/FSTFirestoreClient.h"
  30. #import "Firestore/Source/Core/FSTQuery.h"
  31. #import "Firestore/Source/Model/FSTDocument.h"
  32. #import "Firestore/Source/Model/FSTDocumentKey.h"
  33. #import "Firestore/Source/Model/FSTFieldValue.h"
  34. #import "Firestore/Source/Model/FSTPath.h"
  35. #import "Firestore/Source/Util/FSTAssert.h"
  36. #import "Firestore/Source/Util/FSTAsyncQueryListener.h"
  37. #import "Firestore/Source/Util/FSTUsageValidation.h"
  38. NS_ASSUME_NONNULL_BEGIN
  39. @interface FIRQueryListenOptions ()
  40. - (instancetype)initWithIncludeQueryMetadataChanges:(BOOL)includeQueryMetadataChanges
  41. includeDocumentMetadataChanges:(BOOL)includeDocumentMetadataChanges
  42. NS_DESIGNATED_INITIALIZER;
  43. @end
  44. @implementation FIRQueryListenOptions
  45. + (instancetype)options {
  46. return [[FIRQueryListenOptions alloc] init];
  47. }
  48. - (instancetype)initWithIncludeQueryMetadataChanges:(BOOL)includeQueryMetadataChanges
  49. includeDocumentMetadataChanges:(BOOL)includeDocumentMetadataChanges {
  50. if (self = [super init]) {
  51. _includeQueryMetadataChanges = includeQueryMetadataChanges;
  52. _includeDocumentMetadataChanges = includeDocumentMetadataChanges;
  53. }
  54. return self;
  55. }
  56. - (instancetype)init {
  57. return [self initWithIncludeQueryMetadataChanges:NO includeDocumentMetadataChanges:NO];
  58. }
  59. - (instancetype)includeQueryMetadataChanges:(BOOL)includeQueryMetadataChanges {
  60. return [[FIRQueryListenOptions alloc]
  61. initWithIncludeQueryMetadataChanges:includeQueryMetadataChanges
  62. includeDocumentMetadataChanges:_includeDocumentMetadataChanges];
  63. }
  64. - (instancetype)includeDocumentMetadataChanges:(BOOL)includeDocumentMetadataChanges {
  65. return [[FIRQueryListenOptions alloc]
  66. initWithIncludeQueryMetadataChanges:_includeQueryMetadataChanges
  67. includeDocumentMetadataChanges:includeDocumentMetadataChanges];
  68. }
  69. @end
  70. @interface FIRQuery ()
  71. @property(nonatomic, strong, readonly) FSTQuery *query;
  72. @end
  73. @implementation FIRQuery (Internal)
  74. + (instancetype)referenceWithQuery:(FSTQuery *)query firestore:(FIRFirestore *)firestore {
  75. return [[FIRQuery alloc] initWithQuery:query firestore:firestore];
  76. }
  77. @end
  78. @implementation FIRQuery
  79. #pragma mark - Constructor Methods
  80. - (instancetype)initWithQuery:(FSTQuery *)query firestore:(FIRFirestore *)firestore {
  81. if (self = [super init]) {
  82. _query = query;
  83. _firestore = firestore;
  84. }
  85. return self;
  86. }
  87. #pragma mark - NSObject Methods
  88. - (BOOL)isEqual:(nullable id)other {
  89. if (other == self) return YES;
  90. if (![[other class] isEqual:[self class]]) return NO;
  91. return [self isEqualToQuery:other];
  92. }
  93. - (BOOL)isEqualToQuery:(nullable FIRQuery *)query {
  94. if (self == query) return YES;
  95. if (query == nil) return NO;
  96. if (self.firestore != query.firestore && ![self.firestore isEqual:query.firestore]) return NO;
  97. if (self.query != query.query && ![self.query isEqual:query.query]) return NO;
  98. return YES;
  99. }
  100. - (NSUInteger)hash {
  101. NSUInteger hash = [self.firestore hash];
  102. hash = hash * 31u + [self.query hash];
  103. return hash;
  104. }
  105. #pragma mark - Public Methods
  106. - (void)getDocumentsWithCompletion:(void (^)(FIRQuerySnapshot *_Nullable snapshot,
  107. NSError *_Nullable error))completion {
  108. FSTListenOptions *options = [[FSTListenOptions alloc] initWithIncludeQueryMetadataChanges:YES
  109. includeDocumentMetadataChanges:YES
  110. waitForSyncWhenOnline:YES];
  111. dispatch_semaphore_t registered = dispatch_semaphore_create(0);
  112. __block id<FIRListenerRegistration> listenerRegistration;
  113. FIRQuerySnapshotBlock listener = ^(FIRQuerySnapshot *snapshot, NSError *error) {
  114. if (error) {
  115. completion(nil, error);
  116. return;
  117. }
  118. // Remove query first before passing event to user to avoid user actions affecting the
  119. // now stale query.
  120. dispatch_semaphore_wait(registered, DISPATCH_TIME_FOREVER);
  121. [listenerRegistration remove];
  122. completion(snapshot, nil);
  123. };
  124. listenerRegistration = [self addSnapshotListenerInternalWithOptions:options listener:listener];
  125. dispatch_semaphore_signal(registered);
  126. }
  127. - (id<FIRListenerRegistration>)addSnapshotListener:(FIRQuerySnapshotBlock)listener {
  128. return [self addSnapshotListenerWithOptions:nil listener:listener];
  129. }
  130. - (id<FIRListenerRegistration>)addSnapshotListenerWithOptions:
  131. (nullable FIRQueryListenOptions *)options
  132. listener:(FIRQuerySnapshotBlock)listener {
  133. return [self addSnapshotListenerInternalWithOptions:[self internalOptions:options]
  134. listener:listener];
  135. }
  136. - (id<FIRListenerRegistration>)
  137. addSnapshotListenerInternalWithOptions:(FSTListenOptions *)internalOptions
  138. listener:(FIRQuerySnapshotBlock)listener {
  139. FIRFirestore *firestore = self.firestore;
  140. FSTQuery *query = self.query;
  141. FSTViewSnapshotHandler snapshotHandler = ^(FSTViewSnapshot *snapshot, NSError *error) {
  142. if (error) {
  143. listener(nil, error);
  144. return;
  145. }
  146. FIRSnapshotMetadata *metadata =
  147. [FIRSnapshotMetadata snapshotMetadataWithPendingWrites:snapshot.hasPendingWrites
  148. fromCache:snapshot.fromCache];
  149. listener([FIRQuerySnapshot snapshotWithFirestore:firestore
  150. originalQuery:query
  151. snapshot:snapshot
  152. metadata:metadata],
  153. nil);
  154. };
  155. FSTAsyncQueryListener *asyncListener =
  156. [[FSTAsyncQueryListener alloc] initWithDispatchQueue:self.firestore.client.userDispatchQueue
  157. snapshotHandler:snapshotHandler];
  158. FSTQueryListener *internalListener =
  159. [firestore.client listenToQuery:query
  160. options:internalOptions
  161. viewSnapshotHandler:[asyncListener asyncSnapshotHandler]];
  162. return [[FSTListenerRegistration alloc] initWithClient:self.firestore.client
  163. asyncListener:asyncListener
  164. internalListener:internalListener];
  165. }
  166. - (FIRQuery *)queryWhereField:(NSString *)field isEqualTo:(id)value {
  167. return [self queryWithFilterOperator:FSTRelationFilterOperatorEqual field:field value:value];
  168. }
  169. - (FIRQuery *)queryWhereFieldPath:(FIRFieldPath *)path isEqualTo:(id)value {
  170. return [self queryWithFilterOperator:FSTRelationFilterOperatorEqual
  171. path:path.internalValue
  172. value:value];
  173. }
  174. - (FIRQuery *)queryWhereField:(NSString *)field isLessThan:(id)value {
  175. return [self queryWithFilterOperator:FSTRelationFilterOperatorLessThan field:field value:value];
  176. }
  177. - (FIRQuery *)queryWhereFieldPath:(FIRFieldPath *)path isLessThan:(id)value {
  178. return [self queryWithFilterOperator:FSTRelationFilterOperatorLessThan
  179. path:path.internalValue
  180. value:value];
  181. }
  182. - (FIRQuery *)queryWhereField:(NSString *)field isLessThanOrEqualTo:(id)value {
  183. return [self queryWithFilterOperator:FSTRelationFilterOperatorLessThanOrEqual
  184. field:field
  185. value:value];
  186. }
  187. - (FIRQuery *)queryWhereFieldPath:(FIRFieldPath *)path isLessThanOrEqualTo:(id)value {
  188. return [self queryWithFilterOperator:FSTRelationFilterOperatorLessThanOrEqual
  189. path:path.internalValue
  190. value:value];
  191. }
  192. - (FIRQuery *)queryWhereField:(NSString *)field isGreaterThan:(id)value {
  193. return
  194. [self queryWithFilterOperator:FSTRelationFilterOperatorGreaterThan field:field value:value];
  195. }
  196. - (FIRQuery *)queryWhereFieldPath:(FIRFieldPath *)path isGreaterThan:(id)value {
  197. return [self queryWithFilterOperator:FSTRelationFilterOperatorGreaterThan
  198. path:path.internalValue
  199. value:value];
  200. }
  201. - (FIRQuery *)queryWhereField:(NSString *)field isGreaterThanOrEqualTo:(id)value {
  202. return [self queryWithFilterOperator:FSTRelationFilterOperatorGreaterThanOrEqual
  203. field:field
  204. value:value];
  205. }
  206. - (FIRQuery *)queryWhereFieldPath:(FIRFieldPath *)path isGreaterThanOrEqualTo:(id)value {
  207. return [self queryWithFilterOperator:FSTRelationFilterOperatorGreaterThanOrEqual
  208. path:path.internalValue
  209. value:value];
  210. }
  211. - (FIRQuery *)queryFilteredUsingComparisonPredicate:(NSPredicate *)predicate {
  212. NSComparisonPredicate *comparison = (NSComparisonPredicate *)predicate;
  213. if (comparison.comparisonPredicateModifier != NSDirectPredicateModifier) {
  214. FSTThrowInvalidArgument(@"Invalid query. Predicate cannot have an aggregate modifier.");
  215. }
  216. NSString *path;
  217. id value = nil;
  218. if ([comparison.leftExpression expressionType] == NSKeyPathExpressionType &&
  219. [comparison.rightExpression expressionType] == NSConstantValueExpressionType) {
  220. path = comparison.leftExpression.keyPath;
  221. value = comparison.rightExpression.constantValue;
  222. switch (comparison.predicateOperatorType) {
  223. case NSEqualToPredicateOperatorType:
  224. return [self queryWhereField:path isEqualTo:value];
  225. case NSLessThanPredicateOperatorType:
  226. return [self queryWhereField:path isLessThan:value];
  227. case NSLessThanOrEqualToPredicateOperatorType:
  228. return [self queryWhereField:path isLessThanOrEqualTo:value];
  229. case NSGreaterThanPredicateOperatorType:
  230. return [self queryWhereField:path isGreaterThan:value];
  231. case NSGreaterThanOrEqualToPredicateOperatorType:
  232. return [self queryWhereField:path isGreaterThanOrEqualTo:value];
  233. default:; // Fallback below to throw assertion.
  234. }
  235. } else if ([comparison.leftExpression expressionType] == NSConstantValueExpressionType &&
  236. [comparison.rightExpression expressionType] == NSKeyPathExpressionType) {
  237. path = comparison.rightExpression.keyPath;
  238. value = comparison.leftExpression.constantValue;
  239. switch (comparison.predicateOperatorType) {
  240. case NSEqualToPredicateOperatorType:
  241. return [self queryWhereField:path isEqualTo:value];
  242. case NSLessThanPredicateOperatorType:
  243. return [self queryWhereField:path isGreaterThan:value];
  244. case NSLessThanOrEqualToPredicateOperatorType:
  245. return [self queryWhereField:path isGreaterThanOrEqualTo:value];
  246. case NSGreaterThanPredicateOperatorType:
  247. return [self queryWhereField:path isLessThan:value];
  248. case NSGreaterThanOrEqualToPredicateOperatorType:
  249. return [self queryWhereField:path isLessThanOrEqualTo:value];
  250. default:; // Fallback below to throw assertion.
  251. }
  252. } else {
  253. FSTThrowInvalidArgument(
  254. @"Invalid query. Predicate comparisons must include a key path and a constant.");
  255. }
  256. // Fallback cases of unsupported comparison operator.
  257. switch (comparison.predicateOperatorType) {
  258. case NSCustomSelectorPredicateOperatorType:
  259. FSTThrowInvalidArgument(@"Invalid query. Custom predicate filters are not supported.");
  260. break;
  261. default:
  262. FSTThrowInvalidArgument(@"Invalid query. Operator type %lu is not supported.",
  263. (unsigned long)comparison.predicateOperatorType);
  264. }
  265. }
  266. - (FIRQuery *)queryFilteredUsingCompoundPredicate:(NSPredicate *)predicate {
  267. NSCompoundPredicate *compound = (NSCompoundPredicate *)predicate;
  268. if (compound.compoundPredicateType != NSAndPredicateType || compound.subpredicates.count == 0) {
  269. FSTThrowInvalidArgument(@"Invalid query. Only compound queries using AND are supported.");
  270. }
  271. FIRQuery *query = self;
  272. for (NSPredicate *pred in compound.subpredicates) {
  273. query = [query queryFilteredUsingPredicate:pred];
  274. }
  275. return query;
  276. }
  277. - (FIRQuery *)queryFilteredUsingPredicate:(NSPredicate *)predicate {
  278. if ([predicate isKindOfClass:[NSComparisonPredicate class]]) {
  279. return [self queryFilteredUsingComparisonPredicate:predicate];
  280. } else if ([predicate isKindOfClass:[NSCompoundPredicate class]]) {
  281. return [self queryFilteredUsingCompoundPredicate:predicate];
  282. } else if ([predicate isKindOfClass:[[NSPredicate
  283. predicateWithBlock:^BOOL(id obj, NSDictionary *bindings) {
  284. return true;
  285. }] class]]) {
  286. FSTThrowInvalidArgument(
  287. @"Invalid query. Block-based predicates are not "
  288. "supported. Please use predicateWithFormat to "
  289. "create predicates instead.");
  290. } else {
  291. FSTThrowInvalidArgument(
  292. @"Invalid query. Expect comparison or compound of "
  293. "comparison predicate. Please use "
  294. "predicateWithFormat to create predicates.");
  295. }
  296. }
  297. - (FIRQuery *)queryOrderedByField:(NSString *)field {
  298. return
  299. [self queryOrderedByFieldPath:[FIRFieldPath pathWithDotSeparatedString:field] descending:NO];
  300. }
  301. - (FIRQuery *)queryOrderedByFieldPath:(FIRFieldPath *)fieldPath {
  302. return [self queryOrderedByFieldPath:fieldPath descending:NO];
  303. }
  304. - (FIRQuery *)queryOrderedByField:(NSString *)field descending:(BOOL)descending {
  305. return [self queryOrderedByFieldPath:[FIRFieldPath pathWithDotSeparatedString:field]
  306. descending:descending];
  307. }
  308. - (FIRQuery *)queryOrderedByFieldPath:(FIRFieldPath *)fieldPath descending:(BOOL)descending {
  309. [self validateNewOrderByPath:fieldPath.internalValue];
  310. if (self.query.startAt) {
  311. FSTThrowInvalidUsage(
  312. @"InvalidQueryException",
  313. @"Invalid query. You must not specify a starting point before specifying the order by.");
  314. }
  315. if (self.query.endAt) {
  316. FSTThrowInvalidUsage(
  317. @"InvalidQueryException",
  318. @"Invalid query. You must not specify an ending point before specifying the order by.");
  319. }
  320. FSTSortOrder *sortOrder =
  321. [FSTSortOrder sortOrderWithFieldPath:fieldPath.internalValue ascending:!descending];
  322. return [FIRQuery referenceWithQuery:[self.query queryByAddingSortOrder:sortOrder]
  323. firestore:self.firestore];
  324. }
  325. - (FIRQuery *)queryLimitedTo:(NSInteger)limit {
  326. if (limit <= 0) {
  327. FSTThrowInvalidArgument(@"Invalid Query. Query limit (%ld) is invalid. Limit must be positive.",
  328. (long)limit);
  329. }
  330. return [FIRQuery referenceWithQuery:[self.query queryBySettingLimit:limit] firestore:_firestore];
  331. }
  332. - (FIRQuery *)queryStartingAtDocument:(FIRDocumentSnapshot *)snapshot {
  333. FSTBound *bound = [self boundFromSnapshot:snapshot isBefore:YES];
  334. return [FIRQuery referenceWithQuery:[self.query queryByAddingStartAt:bound]
  335. firestore:self.firestore];
  336. }
  337. - (FIRQuery *)queryStartingAtValues:(NSArray *)fieldValues {
  338. FSTBound *bound = [self boundFromFieldValues:fieldValues isBefore:YES];
  339. return [FIRQuery referenceWithQuery:[self.query queryByAddingStartAt:bound]
  340. firestore:self.firestore];
  341. }
  342. - (FIRQuery *)queryStartingAfterDocument:(FIRDocumentSnapshot *)snapshot {
  343. FSTBound *bound = [self boundFromSnapshot:snapshot isBefore:NO];
  344. return [FIRQuery referenceWithQuery:[self.query queryByAddingStartAt:bound]
  345. firestore:self.firestore];
  346. }
  347. - (FIRQuery *)queryStartingAfterValues:(NSArray *)fieldValues {
  348. FSTBound *bound = [self boundFromFieldValues:fieldValues isBefore:NO];
  349. return [FIRQuery referenceWithQuery:[self.query queryByAddingStartAt:bound]
  350. firestore:self.firestore];
  351. }
  352. - (FIRQuery *)queryEndingBeforeDocument:(FIRDocumentSnapshot *)snapshot {
  353. FSTBound *bound = [self boundFromSnapshot:snapshot isBefore:YES];
  354. return
  355. [FIRQuery referenceWithQuery:[self.query queryByAddingEndAt:bound] firestore:self.firestore];
  356. }
  357. - (FIRQuery *)queryEndingBeforeValues:(NSArray *)fieldValues {
  358. FSTBound *bound = [self boundFromFieldValues:fieldValues isBefore:YES];
  359. return
  360. [FIRQuery referenceWithQuery:[self.query queryByAddingEndAt:bound] firestore:self.firestore];
  361. }
  362. - (FIRQuery *)queryEndingAtDocument:(FIRDocumentSnapshot *)snapshot {
  363. FSTBound *bound = [self boundFromSnapshot:snapshot isBefore:NO];
  364. return
  365. [FIRQuery referenceWithQuery:[self.query queryByAddingEndAt:bound] firestore:self.firestore];
  366. }
  367. - (FIRQuery *)queryEndingAtValues:(NSArray *)fieldValues {
  368. FSTBound *bound = [self boundFromFieldValues:fieldValues isBefore:NO];
  369. return
  370. [FIRQuery referenceWithQuery:[self.query queryByAddingEndAt:bound] firestore:self.firestore];
  371. }
  372. #pragma mark - Private Methods
  373. /** Private helper for all of the queryWhereField: methods. */
  374. - (FIRQuery *)queryWithFilterOperator:(FSTRelationFilterOperator)filterOperator
  375. field:(NSString *)field
  376. value:(id)value {
  377. return [self queryWithFilterOperator:filterOperator
  378. path:[FIRFieldPath pathWithDotSeparatedString:field].internalValue
  379. value:value];
  380. }
  381. - (FIRQuery *)queryWithFilterOperator:(FSTRelationFilterOperator)filterOperator
  382. path:(FSTFieldPath *)fieldPath
  383. value:(id)value {
  384. FSTFieldValue *fieldValue;
  385. if ([fieldPath isKeyFieldPath]) {
  386. if ([value isKindOfClass:[NSString class]]) {
  387. NSString *documentKey = (NSString *)value;
  388. if ([documentKey containsString:@"/"]) {
  389. FSTThrowInvalidArgument(
  390. @"Invalid query. When querying by document ID you must provide "
  391. "a valid document ID, but '%@' contains a '/' character.",
  392. documentKey);
  393. } else if (documentKey.length == 0) {
  394. FSTThrowInvalidArgument(
  395. @"Invalid query. When querying by document ID you must provide "
  396. "a valid document ID, but it was an empty string.");
  397. }
  398. FSTResourcePath *path = [self.query.path pathByAppendingSegment:documentKey];
  399. fieldValue = [FSTReferenceValue referenceValue:[FSTDocumentKey keyWithPath:path]
  400. databaseID:self.firestore.databaseID];
  401. } else if ([value isKindOfClass:[FIRDocumentReference class]]) {
  402. FIRDocumentReference *ref = (FIRDocumentReference *)value;
  403. fieldValue = [FSTReferenceValue referenceValue:ref.key databaseID:self.firestore.databaseID];
  404. } else {
  405. FSTThrowInvalidArgument(
  406. @"Invalid query. When querying by document ID you must provide a "
  407. "valid string or DocumentReference, but it was of type: %@",
  408. NSStringFromClass([value class]));
  409. }
  410. } else {
  411. fieldValue = [self.firestore.dataConverter parsedQueryValue:value];
  412. }
  413. id<FSTFilter> filter;
  414. if ([fieldValue isEqual:[FSTNullValue nullValue]]) {
  415. if (filterOperator != FSTRelationFilterOperatorEqual) {
  416. FSTThrowInvalidUsage(@"InvalidQueryException",
  417. @"Invalid Query. You can only perform equality comparisons on nil / "
  418. "NSNull.");
  419. }
  420. filter = [[FSTNullFilter alloc] initWithField:fieldPath];
  421. } else if ([fieldValue isEqual:[FSTDoubleValue nanValue]]) {
  422. if (filterOperator != FSTRelationFilterOperatorEqual) {
  423. FSTThrowInvalidUsage(@"InvalidQueryException",
  424. @"Invalid Query. You can only perform equality comparisons on NaN.");
  425. }
  426. filter = [[FSTNanFilter alloc] initWithField:fieldPath];
  427. } else {
  428. filter = [FSTRelationFilter filterWithField:fieldPath
  429. filterOperator:filterOperator
  430. value:fieldValue];
  431. [self validateNewRelationFilter:filter];
  432. }
  433. return [FIRQuery referenceWithQuery:[self.query queryByAddingFilter:filter]
  434. firestore:self.firestore];
  435. }
  436. - (void)validateNewRelationFilter:(FSTRelationFilter *)filter {
  437. if ([filter isInequality]) {
  438. FSTFieldPath *existingField = [self.query inequalityFilterField];
  439. if (existingField && ![existingField isEqual:filter.field]) {
  440. FSTThrowInvalidUsage(
  441. @"InvalidQueryException",
  442. @"Invalid Query. All where filters with an inequality "
  443. "(lessThan, lessThanOrEqual, greaterThan, or greaterThanOrEqual) must be on the same "
  444. "field. But you have inequality filters on '%@' and '%@'",
  445. existingField, filter.field);
  446. }
  447. FSTFieldPath *firstOrderByField = [self.query firstSortOrderField];
  448. if (firstOrderByField) {
  449. [self validateOrderByField:firstOrderByField matchesInequalityField:filter.field];
  450. }
  451. }
  452. }
  453. - (void)validateNewOrderByPath:(FSTFieldPath *)fieldPath {
  454. if (![self.query firstSortOrderField]) {
  455. // This is the first order by. It must match any inequality.
  456. FSTFieldPath *inequalityField = [self.query inequalityFilterField];
  457. if (inequalityField) {
  458. [self validateOrderByField:fieldPath matchesInequalityField:inequalityField];
  459. }
  460. }
  461. }
  462. - (void)validateOrderByField:(FSTFieldPath *)orderByField
  463. matchesInequalityField:(FSTFieldPath *)inequalityField {
  464. if (!([orderByField isEqual:inequalityField])) {
  465. FSTThrowInvalidUsage(
  466. @"InvalidQueryException",
  467. @"Invalid query. You have a where filter with an "
  468. "inequality (lessThan, lessThanOrEqual, greaterThan, or greaterThanOrEqual) on field '%@' "
  469. "and so you must also use '%@' as your first queryOrderedBy field, but your first "
  470. "queryOrderedBy is currently on field '%@' instead.",
  471. inequalityField, inequalityField, orderByField);
  472. }
  473. }
  474. /**
  475. * Create a FSTBound from a query given the document.
  476. *
  477. * Note that the FSTBound will always include the key of the document and the position will be
  478. * unambiguous.
  479. *
  480. * Will throw if the document does not contain all fields of the order by of the query.
  481. */
  482. - (FSTBound *)boundFromSnapshot:(FIRDocumentSnapshot *)snapshot isBefore:(BOOL)isBefore {
  483. if (![snapshot exists]) {
  484. FSTThrowInvalidUsage(@"InvalidQueryException",
  485. @"Invalid query. You are trying to start or end a query using a document "
  486. @"that doesn't exist.");
  487. }
  488. FSTDocument *document = snapshot.internalDocument;
  489. NSMutableArray<FSTFieldValue *> *components = [NSMutableArray array];
  490. // Because people expect to continue/end a query at the exact document provided, we need to
  491. // use the implicit sort order rather than the explicit sort order, because it's guaranteed to
  492. // contain the document key. That way the position becomes unambiguous and the query
  493. // continues/ends exactly at the provided document. Without the key (by using the explicit sort
  494. // orders), multiple documents could match the position, yielding duplicate results.
  495. for (FSTSortOrder *sortOrder in self.query.sortOrders) {
  496. if ([sortOrder.field isEqual:[FSTFieldPath keyFieldPath]]) {
  497. [components addObject:[FSTReferenceValue referenceValue:document.key
  498. databaseID:self.firestore.databaseID]];
  499. } else {
  500. FSTFieldValue *value = [document fieldForPath:sortOrder.field];
  501. if (value != nil) {
  502. [components addObject:value];
  503. } else {
  504. FSTThrowInvalidUsage(@"InvalidQueryException",
  505. @"Invalid query. You are trying to start or end a query using a "
  506. "document for which the field '%@' (used as the order by) "
  507. "does not exist.",
  508. sortOrder.field.canonicalString);
  509. }
  510. }
  511. }
  512. return [FSTBound boundWithPosition:components isBefore:isBefore];
  513. }
  514. /** Converts a list of field values to an FSTBound. */
  515. - (FSTBound *)boundFromFieldValues:(NSArray<id> *)fieldValues isBefore:(BOOL)isBefore {
  516. // Use explicit sort order because it has to match the query the user made
  517. NSArray<FSTSortOrder *> *explicitSortOrders = self.query.explicitSortOrders;
  518. if (fieldValues.count > explicitSortOrders.count) {
  519. FSTThrowInvalidUsage(@"InvalidQueryException",
  520. @"Invalid query. You are trying to start or end a query using more values "
  521. @"than were specified in the order by.");
  522. }
  523. NSMutableArray<FSTFieldValue *> *components = [NSMutableArray array];
  524. [fieldValues enumerateObjectsUsingBlock:^(id rawValue, NSUInteger idx, BOOL *stop) {
  525. FSTSortOrder *sortOrder = explicitSortOrders[idx];
  526. if ([sortOrder.field isEqual:[FSTFieldPath keyFieldPath]]) {
  527. if (![rawValue isKindOfClass:[NSString class]]) {
  528. FSTThrowInvalidUsage(@"InvalidQueryException",
  529. @"Invalid query. Expected a string for the document ID.");
  530. }
  531. NSString *documentID = (NSString *)rawValue;
  532. if ([documentID containsString:@"/"]) {
  533. FSTThrowInvalidUsage(@"InvalidQueryException",
  534. @"Invalid query. Document ID '%@' contains a slash.", documentID);
  535. }
  536. FSTDocumentKey *key =
  537. [FSTDocumentKey keyWithPath:[self.query.path pathByAppendingSegment:documentID]];
  538. [components
  539. addObject:[FSTReferenceValue referenceValue:key databaseID:self.firestore.databaseID]];
  540. } else {
  541. FSTFieldValue *fieldValue = [self.firestore.dataConverter parsedQueryValue:rawValue];
  542. [components addObject:fieldValue];
  543. }
  544. }];
  545. return [FSTBound boundWithPosition:components isBefore:isBefore];
  546. }
  547. /** Converts the public API options object to the internal options object. */
  548. - (FSTListenOptions *)internalOptions:(nullable FIRQueryListenOptions *)options {
  549. return [[FSTListenOptions alloc]
  550. initWithIncludeQueryMetadataChanges:options.includeQueryMetadataChanges
  551. includeDocumentMetadataChanges:options.includeDocumentMetadataChanges
  552. waitForSyncWhenOnline:NO];
  553. }
  554. @end
  555. NS_ASSUME_NONNULL_END