| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544 |
- /*
- * Copyright 2018 Google
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #ifndef FIRESTORE_CORE_TEST_UNIT_TESTUTIL_TESTUTIL_H_
- #define FIRESTORE_CORE_TEST_UNIT_TESTUTIL_TESTUTIL_H_
- #include <algorithm>
- #include <memory>
- #include <string>
- #include <unordered_map>
- #include <utility>
- #include <vector>
- #include "Firestore/Protos/nanopb/google/firestore/v1/document.nanopb.h"
- #include "Firestore/core/src/core/core_fwd.h"
- #include "Firestore/core/src/core/direction.h"
- #include "Firestore/core/src/model/document_key.h"
- #include "Firestore/core/src/model/document_set.h"
- #include "Firestore/core/src/model/field_index.h"
- #include "Firestore/core/src/model/model_fwd.h"
- #include "Firestore/core/src/model/mutation.h"
- #include "Firestore/core/src/model/precondition.h"
- #include "Firestore/core/src/model/value_util.h"
- #include "Firestore/core/src/nanopb/byte_string.h"
- #include "Firestore/core/src/nanopb/message.h"
- #include "Firestore/core/src/nanopb/nanopb_util.h"
- #include "absl/strings/string_view.h"
- namespace firebase {
- class Timestamp;
- namespace firestore {
- class GeoPoint;
- namespace remote {
- class RemoteEvent;
- } // namespace remote
- namespace model {
- class MutableDocument;
- } // namespace model
- namespace nanopb {
- class ByteString;
- } // namespace nanopb
- namespace core {
- class FieldFilter;
- class CompositeFilter;
- } // namespace core
- namespace testutil {
- namespace details {
- nanopb::Message<google_firestore_v1_Value> BlobValue(
- std::initializer_list<uint8_t>);
- } // namespace details
- // A bit pattern for our canonical NaN value.
- ABSL_CONST_INIT extern const uint64_t kCanonicalNanBits;
- // Convenience methods for creating instances for tests.
- nanopb::ByteString Bytes(std::initializer_list<uint8_t>);
- nanopb::Message<google_firestore_v1_Value> Value(std::nullptr_t);
- /**
- * A type definition that evaluates to type V only if T is exactly type `bool`.
- */
- template <typename T, typename V>
- using EnableForExactlyBool =
- typename std::enable_if<std::is_same<bool, T>::value, V>::type;
- /**
- * A type definition that evaluates to type V only if T is an integral type but
- * not `bool`.
- */
- template <typename T, typename V>
- using EnableForInts = typename std::enable_if<std::is_integral<T>::value &&
- !std::is_same<bool, T>::value,
- V>::type;
- /**
- * Creates a boolean FieldValue.
- *
- * @tparam T A type that must be exactly bool. Any T that is not bool causes
- * this declaration to be disabled.
- * @param bool_value A boolean value that disallows implicit conversions.
- */
- template <typename T>
- EnableForExactlyBool<T, nanopb::Message<google_firestore_v1_Value>> Value(
- T bool_value) {
- nanopb::Message<google_firestore_v1_Value> result;
- result->which_value_type = google_firestore_v1_Value_boolean_value_tag;
- result->boolean_value = bool_value;
- return result;
- }
- /**
- * Creates an integer FieldValue.
- *
- * This is defined as a template to capture all integer literals. Just defining
- * this as taking int64_t would make integer literals ambiguous because int64_t
- * and double are equally good choices according to the standard.
- *
- * @tparam T Any integral type (but not bool). Types larger than int64_t will
- * likely generate a warning.
- * @param value An integer value.
- */
- template <typename T>
- EnableForInts<T, nanopb::Message<google_firestore_v1_Value>> Value(T value) {
- nanopb::Message<google_firestore_v1_Value> result;
- result->which_value_type = google_firestore_v1_Value_integer_value_tag;
- result->integer_value = value;
- return result;
- }
- nanopb::Message<google_firestore_v1_Value> Value(double value);
- nanopb::Message<google_firestore_v1_Value> Value(Timestamp value);
- nanopb::Message<google_firestore_v1_Value> Value(const char* value);
- nanopb::Message<google_firestore_v1_Value> Value(const std::string& value);
- nanopb::Message<google_firestore_v1_Value> Value(
- const nanopb::ByteString& value);
- nanopb::Message<google_firestore_v1_Value> Value(const GeoPoint& value);
- template <typename... Ints>
- nanopb::Message<google_firestore_v1_Value> BlobValue(Ints... octets) {
- return details::BlobValue({static_cast<uint8_t>(octets)...});
- }
- nanopb::Message<google_firestore_v1_Value> Value(
- nanopb::Message<google_firestore_v1_Value> value);
- nanopb::Message<google_firestore_v1_Value> Value(
- nanopb::Message<google_firestore_v1_ArrayValue> value);
- nanopb::Message<google_firestore_v1_Value> Value(
- nanopb::Message<google_firestore_v1_MapValue> value);
- nanopb::Message<google_firestore_v1_Value> Value(
- const model::ObjectValue& value);
- using OverlayTypeMap = std::unordered_map<model::DocumentKey,
- model::Mutation::Type,
- model::DocumentKeyHash>;
- namespace details {
- /**
- * Recursive base case for AddPairs, below. Returns the map.
- */
- inline nanopb::Message<google_firestore_v1_Value> AddPairs(
- nanopb::Message<google_firestore_v1_Value> prior) {
- return prior;
- }
- /**
- * Inserts the given key-value pair into the map, and then recursively calls
- * AddPairs to add any remaining arguments.
- *
- * @param prior A map into which the values should be inserted.
- * @param key The key naming the field to insert.
- * @param value A value to wrap with a call to Value(), above.
- * @param rest Any remaining arguments
- *
- * @return The resulting map.
- */
- template <typename ValueType, typename... Args>
- nanopb::Message<google_firestore_v1_Value> AddPairs(
- nanopb::Message<google_firestore_v1_Value> prior,
- const std::string& key,
- ValueType value,
- Args... rest) {
- nanopb::Message<google_firestore_v1_Value> result = std::move(prior);
- result->which_value_type = google_firestore_v1_Value_map_value_tag;
- size_t new_count = result->map_value.fields_count + 1;
- result->map_value.fields_count = nanopb::CheckedSize(new_count);
- result->map_value.fields =
- nanopb::ResizeArray<google_firestore_v1_MapValue_FieldsEntry>(
- result->map_value.fields, new_count);
- result->map_value.fields[new_count - 1].key = nanopb::MakeBytesArray(key);
- result->map_value.fields[new_count - 1].value =
- *Value(std::move(value)).release();
- return AddPairs(std::move(result), std::forward<Args>(rest)...);
- }
- /**
- * Creates an immutable sorted map from the given key/value pairs.
- *
- * @param key_value_pairs Alternating strings naming keys and values that can
- * be passed to Value().
- */
- template <typename... Args>
- nanopb::Message<google_firestore_v1_Value> MakeMap(Args... key_value_pairs) {
- nanopb::Message<google_firestore_v1_Value> map_value;
- map_value->which_value_type = google_firestore_v1_Value_map_value_tag;
- map_value->map_value = {};
- return AddPairs(std::move(map_value), std::forward<Args>(key_value_pairs)...);
- }
- /**
- * Recursive base case for AddElements, below.
- */
- inline void AddElements(nanopb::Message<google_firestore_v1_ArrayValue>&,
- pb_size_t) {
- }
- /**
- * Inserts the given element into the array, and then recursively calls
- * AddElement to add any remaining arguments.
- *
- * @param array_value An array into which the values should be inserted.
- * @param pos The index of the next element.
- * @param value The element to insert.
- * @param rest Any remaining arguments
- */
- template <typename ValueType, typename... Args>
- void AddElements(nanopb::Message<google_firestore_v1_ArrayValue>& array_value,
- pb_size_t pos,
- ValueType value,
- Args&&... rest) {
- array_value->values[pos] = *Value(std::move(value)).release();
- AddElements(array_value, ++pos, std::forward<Args>(rest)...);
- }
- /**
- * Inserts the elements into the given array.
- */
- template <typename... Args>
- nanopb::Message<google_firestore_v1_ArrayValue> MakeArray(Args&&... values) {
- nanopb::Message<google_firestore_v1_ArrayValue> array_value;
- array_value->values_count = nanopb::CheckedSize(sizeof...(Args));
- array_value->values =
- nanopb::MakeArray<google_firestore_v1_Value>(array_value->values_count);
- AddElements(array_value, 0, std::forward<Args>(values)...);
- return array_value;
- }
- } // namespace details
- template <typename... Args>
- nanopb::Message<google_firestore_v1_ArrayValue> Array(Args&&... values) {
- return details::MakeArray(std::move(values)...);
- }
- /** Wraps an immutable sorted map into an ObjectValue. */
- model::ObjectValue WrapObject(nanopb::Message<google_firestore_v1_Value> value);
- /**
- * Creates an ObjectValue from the given key/value pairs.
- *
- * @param key_value_pairs Alternating strings naming keys and values that can
- * be passed to Value().
- */
- template <typename... Args>
- model::ObjectValue WrapObject(Args... key_value_pairs) {
- return WrapObject(details::MakeMap(std::move(key_value_pairs)...));
- }
- /**
- * Creates an ObjectValue from the given key/value pairs with Type::Object.
- *
- * @param key_value_pairs Alternating strings naming keys and values that can
- * be passed to Value().
- */
- template <typename... Args>
- nanopb::Message<google_firestore_v1_Value> Map(Args... key_value_pairs) {
- return details::MakeMap(std::move(key_value_pairs)...);
- }
- template <typename... Args>
- nanopb::Message<google_firestore_v1_Value> VectorType(Args&&... values) {
- return Map("__type__", "__vector__", "value",
- details::MakeArray(std::move(values)...));
- }
- model::DocumentKey Key(absl::string_view path);
- model::FieldPath Field(absl::string_view field);
- model::DatabaseId DbId(std::string project = "project/(default)");
- nanopb::Message<google_firestore_v1_Value> Ref(std::string project,
- absl::string_view path);
- model::ResourcePath Resource(absl::string_view field);
- /**
- * Creates a snapshot version from the given version timestamp.
- *
- * @param version a timestamp in microseconds since the epoch.
- */
- model::SnapshotVersion Version(int64_t version);
- model::SnapshotVersion Version(int64_t seconds, int32_t nanoseconds);
- model::MutableDocument Doc(absl::string_view key, int64_t version = 0);
- model::MutableDocument Doc(absl::string_view key,
- int64_t version,
- nanopb::Message<google_firestore_v1_Value> data);
- /** A convenience method for creating deleted docs for tests. */
- model::MutableDocument DeletedDoc(absl::string_view key, int64_t version = 0);
- /** A convenience method for creating deleted docs for tests. */
- model::MutableDocument DeletedDoc(model::DocumentKey key, int64_t version = 0);
- /** A convenience method for creating unknown docs for tests. */
- model::MutableDocument UnknownDoc(absl::string_view key, int64_t version);
- /** A convenience method for creating invalid (missing) docs for tests. */
- model::MutableDocument InvalidDoc(absl::string_view key);
- /**
- * Creates a DocumentComparator that will compare Documents by the given
- * field_path string then by key.
- */
- model::DocumentComparator DocComparator(absl::string_view field_path);
- /**
- * Creates a DocumentSet based on the given comparator, initially containing the
- * given documents.
- */
- model::DocumentSet DocSet(model::DocumentComparator comp,
- std::vector<model::Document> docs);
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- nanopb::Message<google_firestore_v1_Value> value);
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- nanopb::Message<google_firestore_v1_ArrayValue> value);
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- std::nullptr_t);
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- const char* value);
- template <typename T>
- EnableForExactlyBool<T, core::FieldFilter> Filter(absl::string_view key,
- absl::string_view op,
- T value) {
- return Filter(key, op, Value(value));
- }
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- int value);
- core::FieldFilter Filter(absl::string_view key,
- absl::string_view op,
- double value);
- core::CompositeFilter AndFilters(std::vector<core::Filter> filters);
- core::CompositeFilter OrFilters(std::vector<core::Filter> filters);
- core::Direction Direction(absl::string_view direction);
- core::OrderBy OrderBy(absl::string_view key,
- absl::string_view direction = "asc");
- core::OrderBy OrderBy(model::FieldPath field_path, core::Direction direction);
- core::Query Query(absl::string_view path);
- core::Query CollectionGroupQuery(absl::string_view collection_id);
- remote::RemoteEvent AddedRemoteEvent(
- const std::vector<model::MutableDocument>& docs,
- const std::vector<model::TargetId>& added_to_targets);
- remote::RemoteEvent AddedRemoteEvent(
- const model::MutableDocument& doc,
- const std::vector<model::TargetId>& added_to_targets);
- remote::RemoteEvent UpdateRemoteEvent(
- const model::MutableDocument& doc,
- const std::vector<model::TargetId>& updated_in_targets,
- const std::vector<model::TargetId>& removed_from_targets);
- remote::RemoteEvent UpdateRemoteEventWithLimboTargets(
- const model::MutableDocument& doc,
- const std::vector<model::TargetId>& updated_in_targets,
- const std::vector<model::TargetId>& removed_from_targets,
- const std::vector<model::TargetId>& limbo_targets);
- model::SetMutation SetMutation(
- absl::string_view path,
- nanopb::Message<google_firestore_v1_Value> values,
- std::vector<std::pair<std::string, model::TransformOperation>> transforms =
- {});
- model::PatchMutation PatchMutation(
- absl::string_view path,
- nanopb::Message<google_firestore_v1_Value> values,
- std::vector<std::pair<std::string, model::TransformOperation>> transforms =
- {});
- model::PatchMutation PatchMutation(
- absl::string_view path,
- nanopb::Message<google_firestore_v1_Value> values,
- const std::vector<model::FieldPath>& update_mask,
- std::vector<std::pair<std::string, model::TransformOperation>> transforms =
- {});
- model::PatchMutation MergeMutation(
- absl::string_view path,
- nanopb::Message<google_firestore_v1_Value> values,
- const std::vector<model::FieldPath>& update_mask,
- std::vector<std::pair<std::string, model::TransformOperation>> transforms =
- {});
- model::PatchMutation PatchMutationHelper(
- absl::string_view path,
- nanopb::Message<google_firestore_v1_Value> values,
- std::vector<std::pair<std::string, model::TransformOperation>> transforms,
- model::Precondition precondition,
- const absl::optional<std::vector<model::FieldPath>>& update_mask);
- /**
- * Creates a pair of field name, TransformOperation that represents a
- * server-generated timestamp, suitable for passing to TransformMutation,
- * above.
- */
- std::pair<std::string, model::TransformOperation> ServerTimestamp(
- std::string field);
- /**
- * Creates a pair of field name, TransformOperation that represents a numeric
- * increment on the given field, suitable for passing to TransformMutation,
- * above.
- */
- std::pair<std::string, model::TransformOperation> Increment(
- std::string field, nanopb::Message<google_firestore_v1_Value> operand);
- /**
- * Creates a pair of field name, TransformOperation that represents an array
- * union on the given field, suitable for passing to TransformMutation,
- * above.
- */
- std::pair<std::string, model::TransformOperation> ArrayUnion(
- std::string field,
- const std::vector<nanopb::Message<google_firestore_v1_Value>>& operands);
- model::DeleteMutation DeleteMutation(absl::string_view path);
- model::VerifyMutation VerifyMutation(absl::string_view path, int64_t version);
- model::MutationResult MutationResult(int64_t version);
- nanopb::ByteString ResumeToken(int64_t snapshot_version);
- template <typename T, typename... Ts>
- std::vector<T> Vector(T&& arg1, Ts&&... args) {
- return {std::forward<T>(arg1), std::forward<Ts>(args)...};
- }
- // Degenerate case to end recursion of `MoveIntoVector`.
- template <typename T>
- void MoveIntoVector(std::vector<std::unique_ptr<T>>*) {
- }
- template <typename T, typename Head, typename... Tail>
- void MoveIntoVector(std::vector<std::unique_ptr<T>>* result,
- Head head,
- Tail... tail) {
- result->push_back(std::move(head));
- MoveIntoVector(result, std::move(tail)...);
- }
- // Works around the fact that move-only types (in this case, `unique_ptr`) don't
- // work with `initializer_list`. Desired (doesn't work):
- //
- // std::unique_ptr<int> x, y;
- // std::vector<std::unique_ptr>> foo{std::move(x), std::move(y)};
- //
- // Actual:
- //
- // std::unique_ptr<int> x, y;
- // std::vector<std::unique_ptr<int>> foo = Changes(std::move(x),
- // std::move(y));
- template <typename T, typename... Elems>
- std::vector<std::unique_ptr<T>> VectorOfUniquePtrs(Elems... elems) {
- std::vector<std::unique_ptr<T>> result;
- MoveIntoVector<T>(&result, std::move(elems)...);
- return result;
- }
- model::FieldIndex MakeFieldIndex(const std::string& collection_group);
- model::FieldIndex MakeFieldIndex(const std::string& collection_group,
- int32_t index_id,
- model::IndexState state);
- model::FieldIndex MakeFieldIndex(const std::string& collection_group,
- const std::string& field,
- model::Segment::Kind kind);
- model::FieldIndex MakeFieldIndex(const std::string& collection_group,
- const std::string& field_1,
- model::Segment::Kind kind_1,
- const std::string& field_2,
- model::Segment::Kind kind_2);
- model::FieldIndex MakeFieldIndex(const std::string& collection_group,
- const std::string& field_1,
- model::Segment::Kind kind_1,
- const std::string& field_2,
- model::Segment::Kind kind_2,
- const std::string& field_3,
- model::Segment::Kind kind_3);
- model::FieldIndex MakeFieldIndex(const std::string& collection_group,
- int32_t index_id,
- model::IndexState state,
- const std::string& field_1,
- model::Segment::Kind kind_1);
- } // namespace testutil
- } // namespace firestore
- } // namespace firebase
- #endif // FIRESTORE_CORE_TEST_UNIT_TESTUTIL_TESTUTIL_H_
|