// Protocol Buffers - Google's data interchange format // Copyright 2008 Google Inc. All rights reserved. // https://developers.google.com/protocol-buffers/ // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following disclaimer // in the documentation and/or other materials provided with the // distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived from // this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // Author: [email protected] (Joseph Schorr) // Based on original Protocol Buffers design by // Sanjay Ghemawat, Jeff Dean, and others. // // This file defines static methods and classes for comparing Protocol // Messages. // // Aug. 2008: Added Unknown Fields Comparison for messages. // Aug. 2009: Added different options to compare repeated fields. // Apr. 2010: Moved field comparison to FieldComparator // Sep. 2020: Added option to output map keys in path #ifndef GOOGLE_PROTOBUF_UTIL_MESSAGE_DIFFERENCER_H__ #define GOOGLE_PROTOBUF_UTIL_MESSAGE_DIFFERENCER_H__ #include <functional> #include <map> #include <memory> #include <set> #include <string> #include <vector> #include <google/protobuf/descriptor.h> // FieldDescriptor #include <google/protobuf/message.h> // Message #include <google/protobuf/unknown_field_set.h> #include <google/protobuf/util/field_comparator.h> // Always include as last one, otherwise it can break compilation #include <google/protobuf/port_def.inc> namespace google { namespace protobuf { class DynamicMessageFactory; class FieldDescriptor; namespace io { class ZeroCopyOutputStream; class Printer; } // namespace io namespace util { class DefaultFieldComparator; class FieldContext; // declared below MessageDifferencer // Defines a collection of field descriptors. // In case of internal google codebase we are using absl::FixedArray instead // of vector. It significantly speeds up proto comparison (by ~30%) by // reducing the number of malloc/free operations FieldDescriptorArray; // A basic differencer that can be used to determine // the differences between two specified Protocol Messages. If any differences // are found, the Compare method will return false, and any differencer reporter // specified via ReportDifferencesTo will have its reporting methods called (see // below for implementation of the report). Based off of the original // ProtocolDifferencer implementation in //net/proto/protocol-differencer.h // (Thanks Todd!). // // MessageDifferencer REQUIRES that compared messages be the same type, defined // as messages that share the same descriptor. If not, the behavior of this // class is undefined. // // People disagree on what MessageDifferencer should do when asked to compare // messages with different descriptors. Some people think it should always // return false. Others expect it to try to look for similar fields and // compare them anyway -- especially if the descriptors happen to be identical. // If we chose either of these behaviors, some set of people would find it // surprising, and could end up writing code expecting the other behavior // without realizing their error. Therefore, we forbid that usage. // // This class is implemented based on the proto2 reflection. The performance // should be good enough for normal usages. However, for places where the // performance is extremely sensitive, there are several alternatives: // - Comparing serialized string // Downside: false negatives (there are messages that are the same but their // serialized strings are different). // - Equals code generator by compiler plugin (net/proto2/contrib/equals_plugin) // Downside: more generated code; maintenance overhead for the additional rule // (must be in sync with the original proto_library). // // Note on handling of google.protobuf.Any: MessageDifferencer automatically // unpacks Any::value into a Message and compares its individual fields. // Messages encoded in a repeated Any cannot be compared using TreatAsMap. // // Note on thread-safety: MessageDifferencer is *not* thread-safe. You need to // guard it with a lock to use the same MessageDifferencer instance from // multiple threads. Note that it's fine to call static comparison methods // (like MessageDifferencer::Equals) concurrently, but it's not recommended for // performance critical code as it leads to extra allocations. class PROTOBUF_EXPORT MessageDifferencer { … }; // This class provides extra information to the FieldComparator::Compare // function. class PROTOBUF_EXPORT FieldContext { … }; } // namespace util } // namespace protobuf } // namespace google #include <google/protobuf/port_undef.inc> #endif // GOOGLE_PROTOBUF_UTIL_MESSAGE_DIFFERENCER_H__