FSTUserDataConverterTests.mm 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. /*
  2. * Copyright 2019 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 "Firestore/Source/API/FSTUserDataConverter.h"
  17. #import <FirebaseFirestore/FIRFieldValue.h>
  18. #import <FirebaseFirestore/FIRGeoPoint.h>
  19. #import <FirebaseFirestore/FIRTimestamp.h>
  20. #import <XCTest/XCTest.h>
  21. #import "Firestore/Example/Tests/Util/FSTHelpers.h"
  22. #import "Firestore/Source/API/converters.h"
  23. #include "Firestore/core/src/model/database_id.h"
  24. #include "Firestore/core/src/model/field_value.h"
  25. #include "Firestore/core/src/model/patch_mutation.h"
  26. #include "Firestore/core/src/model/set_mutation.h"
  27. #include "Firestore/core/src/model/transform_operation.h"
  28. #include "Firestore/core/src/nanopb/nanopb_util.h"
  29. #include "Firestore/core/test/unit/testutil/testutil.h"
  30. namespace util = firebase::firestore::util;
  31. using firebase::firestore::api::MakeGeoPoint;
  32. using firebase::firestore::api::MakeTimestamp;
  33. using firebase::firestore::model::ArrayTransform;
  34. using firebase::firestore::model::DatabaseId;
  35. using firebase::firestore::model::FieldPath;
  36. using firebase::firestore::model::FieldValue;
  37. using firebase::firestore::model::FieldTransform;
  38. using firebase::firestore::model::ObjectValue;
  39. using firebase::firestore::model::PatchMutation;
  40. using firebase::firestore::model::TransformOperation;
  41. using firebase::firestore::model::SetMutation;
  42. using firebase::firestore::nanopb::MakeNSData;
  43. using firebase::firestore::testutil::Field;
  44. @interface FSTUserDataConverterTests : XCTestCase
  45. @end
  46. @implementation FSTUserDataConverterTests
  47. - (void)testConvertsIntegers {
  48. NSArray<NSNumber *> *values = @[
  49. @(INT_MIN), @(-1), @0, @1, @2, @(UCHAR_MAX), @(INT_MAX), // Standard integers
  50. @(LONG_MIN), @(LONG_MAX), @(LLONG_MIN), @(LLONG_MAX) // Larger values
  51. ];
  52. for (NSNumber *value in values) {
  53. FieldValue wrapped = FSTTestFieldValue(value);
  54. XCTAssertEqual(wrapped.type(), FieldValue::Type::Integer);
  55. XCTAssertEqual(wrapped.integer_value(), [value longLongValue]);
  56. }
  57. }
  58. - (void)testConvertsDoubles {
  59. // Note that 0x1.0p-1074 is a hex floating point literal representing the minimum subnormal
  60. // number: <https://en.wikipedia.org/wiki/Denormal_number>.
  61. NSArray<NSNumber *> *values = @[
  62. @(-INFINITY), @(-DBL_MAX), @(LLONG_MIN * -1.0), @(-1.1), @(-0x1.0p-1074), @(-0.0), @(0.0),
  63. @(0x1.0p-1074), @(DBL_MIN), @(1.1), @(LLONG_MAX * 1.0), @(DBL_MAX), @(INFINITY)
  64. ];
  65. for (NSNumber *value in values) {
  66. FieldValue wrapped = FSTTestFieldValue(value);
  67. XCTAssertEqual(wrapped.type(), FieldValue::Type::Double);
  68. XCTAssertEqual(wrapped.double_value(), [value doubleValue]);
  69. }
  70. }
  71. - (void)testConvertsNilAndNSNull {
  72. FieldValue nullValue = FieldValue::Null();
  73. XCTAssertEqual(nullValue.type(), FieldValue::Type::Null);
  74. XCTAssertEqual(FSTTestFieldValue(nil), nullValue);
  75. XCTAssertEqual(FSTTestFieldValue([NSNull null]), nullValue);
  76. }
  77. - (void)testConvertsBooleans {
  78. NSArray<NSNumber *> *values = @[ @YES, @NO ];
  79. for (NSNumber *value in values) {
  80. FieldValue wrapped = FSTTestFieldValue(value);
  81. XCTAssertEqual(wrapped.type(), FieldValue::Type::Boolean);
  82. XCTAssertEqual(wrapped.boolean_value(), [value boolValue]);
  83. }
  84. }
  85. - (void)testConvertsUnsignedCharToInteger {
  86. // See comments in FSTUserDataConverter regarding handling of signed char. Essentially, signed
  87. // char has to be treated as boolean. Unsigned chars could conceivably be handled consistently
  88. // with signed chars but on arm64 these end up being stored as signed shorts. This forces us to
  89. // choose, and it's more useful to support shorts as Integers than it is to treat unsigned char as
  90. // Boolean.
  91. FieldValue wrapped = FSTTestFieldValue([NSNumber numberWithUnsignedChar:1]);
  92. XCTAssertEqual(wrapped, FieldValue::FromInteger(1));
  93. }
  94. union DoubleBits {
  95. double d;
  96. uint64_t bits;
  97. };
  98. - (void)testConvertsStrings {
  99. NSArray<NSString *> *values = @[ @"", @"abc" ];
  100. for (id value in values) {
  101. FieldValue wrapped = FSTTestFieldValue(value);
  102. XCTAssertEqual(wrapped.type(), FieldValue::Type::String);
  103. XCTAssertEqual(wrapped.string_value(), util::MakeString(value));
  104. }
  105. }
  106. - (void)testConvertsDates {
  107. NSArray<NSDate *> *values =
  108. @[ FSTTestDate(1900, 12, 1, 1, 20, 30), FSTTestDate(2017, 4, 24, 13, 20, 30) ];
  109. for (NSDate *value in values) {
  110. FieldValue wrapped = FSTTestFieldValue(value);
  111. XCTAssertEqual(wrapped.type(), FieldValue::Type::Timestamp);
  112. XCTAssertEqual(wrapped.timestamp_value(), MakeTimestamp(value));
  113. }
  114. }
  115. - (void)testConvertsGeoPoints {
  116. NSArray<FIRGeoPoint *> *values = @[ FSTTestGeoPoint(1.24, 4.56), FSTTestGeoPoint(-20, 100) ];
  117. for (FIRGeoPoint *value in values) {
  118. FieldValue wrapped = FSTTestFieldValue(value);
  119. XCTAssertEqual(wrapped.type(), FieldValue::Type::GeoPoint);
  120. XCTAssertEqual(wrapped.geo_point_value(), MakeGeoPoint(value));
  121. }
  122. }
  123. - (void)testConvertsBlobs {
  124. NSArray<NSData *> *values = @[ FSTTestData(1, 2, 3, -1), FSTTestData(1, 2, -1) ];
  125. for (NSData *value in values) {
  126. FieldValue wrapped = FSTTestFieldValue(value);
  127. XCTAssertEqual(wrapped.type(), FieldValue::Type::Blob);
  128. XCTAssertEqualObjects(MakeNSData(wrapped.blob_value()), value);
  129. }
  130. }
  131. - (void)testConvertsResourceNames {
  132. NSArray<FSTDocumentKeyReference *> *values = @[
  133. FSTTestRef("project", DatabaseId::kDefault, @"foo/bar"),
  134. FSTTestRef("project", DatabaseId::kDefault, @"foo/baz")
  135. ];
  136. for (FSTDocumentKeyReference *value in values) {
  137. FieldValue wrapped = FSTTestFieldValue(value);
  138. XCTAssertEqual(wrapped.type(), FieldValue::Type::Reference);
  139. XCTAssertEqual(wrapped.reference_value().key(), value.key);
  140. XCTAssertTrue(wrapped.reference_value().database_id() == value.databaseID);
  141. }
  142. }
  143. - (void)testConvertsEmptyObjects {
  144. XCTAssertEqual(ObjectValue(FSTTestFieldValue(@{})), ObjectValue::Empty());
  145. XCTAssertEqual(FSTTestFieldValue(@{}).type(), FieldValue::Type::Object);
  146. }
  147. - (void)testConvertsSimpleObjects {
  148. ObjectValue actual =
  149. FSTTestObjectValue(@{@"a" : @"foo", @"b" : @(1L), @"c" : @YES, @"d" : [NSNull null]});
  150. ObjectValue expected = ObjectValue::FromMap({{"a", FieldValue::FromString("foo")},
  151. {"b", FieldValue::FromInteger(1)},
  152. {"c", FieldValue::True()},
  153. {"d", FieldValue::Null()}});
  154. XCTAssertEqual(actual, expected);
  155. XCTAssertEqual(actual.AsFieldValue().type(), FieldValue::Type::Object);
  156. }
  157. - (void)testConvertsNestedObjects {
  158. ObjectValue actual = FSTTestObjectValue(@{@"a" : @{@"b" : @{@"c" : @"foo"}, @"d" : @YES}});
  159. ObjectValue expected = ObjectValue::FromMap({
  160. {"a",
  161. ObjectValue::FromMap({{"b", ObjectValue::FromMap({{"c", FieldValue::FromString("foo")}})},
  162. {"d", FieldValue::True()}})},
  163. });
  164. XCTAssertEqual(actual, expected);
  165. XCTAssertEqual(actual.AsFieldValue().type(), FieldValue::Type::Object);
  166. }
  167. - (void)testConvertsArrays {
  168. FieldValue expected = FieldValue::FromArray({
  169. FieldValue::FromString("value"),
  170. FieldValue::True(),
  171. });
  172. FieldValue actual = (FieldValue)FSTTestFieldValue(@[ @"value", @YES ]);
  173. XCTAssertEqual(actual, expected);
  174. XCTAssertEqual(actual.type(), FieldValue::Type::Array);
  175. }
  176. - (void)testNSDatesAreConvertedToTimestamps {
  177. NSDate *date = [NSDate date];
  178. id input = @{@"array" : @[ @1, date ], @"obj" : @{@"date" : date, @"string" : @"hi"}};
  179. ObjectValue value = FSTTestObjectValue(input);
  180. {
  181. auto array = value.Get(Field("array"));
  182. XCTAssertTrue(array.has_value());
  183. XCTAssertEqual(array->type(), FieldValue::Type::Array);
  184. const FieldValue &actual = array->array_value()[1];
  185. XCTAssertEqual(actual.type(), FieldValue::Type::Timestamp);
  186. XCTAssertEqual(actual.timestamp_value(), MakeTimestamp(date));
  187. }
  188. {
  189. auto found = value.Get(Field("obj.date"));
  190. XCTAssertTrue(found.has_value());
  191. XCTAssertEqual(found->type(), FieldValue::Type::Timestamp);
  192. XCTAssertEqual(found->timestamp_value(), MakeTimestamp(date));
  193. }
  194. }
  195. - (void)testCreatesArrayUnionTransforms {
  196. PatchMutation patchMutation = FSTTestPatchMutation(@"collection/key", @{
  197. @"foo" : [FIRFieldValue fieldValueForArrayUnion:@[ @"tag" ]],
  198. @"bar.baz" :
  199. [FIRFieldValue fieldValueForArrayUnion:@[ @YES, @{@"nested" : @{@"a" : @[ @1, @2 ]}} ]]
  200. },
  201. {});
  202. XCTAssertEqual(patchMutation.field_transforms().size(), 2u);
  203. SetMutation setMutation = FSTTestSetMutation(@"collection/key", @{
  204. @"foo" : [FIRFieldValue fieldValueForArrayUnion:@[ @"tag" ]],
  205. @"bar" : [FIRFieldValue fieldValueForArrayUnion:@[ @YES, @{@"nested" : @{@"a" : @[ @1, @2 ]}} ]]
  206. });
  207. XCTAssertEqual(setMutation.field_transforms().size(), 2u);
  208. const FieldTransform &patchFirst = patchMutation.field_transforms()[0];
  209. XCTAssertEqual(patchFirst.path(), FieldPath({"foo"}));
  210. const FieldTransform &setFirst = setMutation.field_transforms()[0];
  211. XCTAssertEqual(setFirst.path(), FieldPath({"foo"}));
  212. {
  213. std::vector<FieldValue> expectedElements{FSTTestFieldValue(@"tag")};
  214. ArrayTransform expected(TransformOperation::Type::ArrayUnion, expectedElements);
  215. XCTAssertEqual(static_cast<const ArrayTransform &>(patchFirst.transformation()), expected);
  216. XCTAssertEqual(static_cast<const ArrayTransform &>(setFirst.transformation()), expected);
  217. }
  218. const FieldTransform &patchSecond = patchMutation.field_transforms()[1];
  219. XCTAssertEqual(patchSecond.path(), FieldPath({"bar", "baz"}));
  220. const FieldTransform &setSecond = setMutation.field_transforms()[1];
  221. XCTAssertEqual(setSecond.path(), FieldPath({"bar"}));
  222. {
  223. std::vector<FieldValue> expectedElements {
  224. FSTTestFieldValue(@YES), FSTTestFieldValue(@{@"nested" : @{@"a" : @[ @1, @2 ]}})
  225. };
  226. ArrayTransform expected(TransformOperation::Type::ArrayUnion, expectedElements);
  227. XCTAssertEqual(static_cast<const ArrayTransform &>(patchSecond.transformation()), expected);
  228. XCTAssertEqual(static_cast<const ArrayTransform &>(setSecond.transformation()), expected);
  229. }
  230. }
  231. - (void)testCreatesArrayRemoveTransforms {
  232. PatchMutation patchMutation = FSTTestPatchMutation(@"collection/key", @{
  233. @"foo" : [FIRFieldValue fieldValueForArrayRemove:@[ @"tag" ]],
  234. },
  235. {});
  236. XCTAssertEqual(patchMutation.field_transforms().size(), 1u);
  237. SetMutation setMutation = FSTTestSetMutation(@"collection/key", @{
  238. @"foo" : [FIRFieldValue fieldValueForArrayRemove:@[ @"tag" ]],
  239. });
  240. XCTAssertEqual(patchMutation.field_transforms().size(), 1u);
  241. const FieldTransform &patchFirst = patchMutation.field_transforms()[0];
  242. XCTAssertEqual(patchFirst.path(), FieldPath({"foo"}));
  243. const FieldTransform &setFirst = setMutation.field_transforms()[0];
  244. XCTAssertEqual(setFirst.path(), FieldPath({"foo"}));
  245. {
  246. std::vector<FieldValue> expectedElements{FSTTestFieldValue(@"tag")};
  247. const ArrayTransform expected(TransformOperation::Type::ArrayRemove, expectedElements);
  248. XCTAssertEqual(static_cast<const ArrayTransform &>(patchFirst.transformation()), expected);
  249. XCTAssertEqual(static_cast<const ArrayTransform &>(setFirst.transformation()), expected);
  250. }
  251. }
  252. @end