mirror of
https://github.com/Mauler125/r5sdk.git
synced 2025-02-09 19:15:03 +01:00
390 lines
15 KiB
C++
390 lines
15 KiB
C++
// 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.
|
|
|
|
#ifndef GOOGLE_PROTOBUF_UTIL_INTERNAL_PROTO_WRITER_H__
|
|
#define GOOGLE_PROTOBUF_UTIL_INTERNAL_PROTO_WRITER_H__
|
|
|
|
#include <cstdint>
|
|
#include <deque>
|
|
#include <string>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#include <thirdparty/protobuf/stubs/common.h>
|
|
#include <thirdparty/protobuf/type.pb.h>
|
|
#include <thirdparty/protobuf/io/coded_stream.h>
|
|
#include <thirdparty/protobuf/io/zero_copy_stream_impl.h>
|
|
#include <thirdparty/protobuf/descriptor.h>
|
|
#include <thirdparty/protobuf/stubs/bytestream.h>
|
|
#include <thirdparty/protobuf/stubs/status.h>
|
|
#include <thirdparty/protobuf/util/internal/datapiece.h>
|
|
#include <thirdparty/protobuf/util/internal/error_listener.h>
|
|
#include <thirdparty/protobuf/util/internal/structured_objectwriter.h>
|
|
#include <thirdparty/protobuf/util/internal/type_info.h>
|
|
#include <thirdparty/protobuf/util/type_resolver.h>
|
|
#include <thirdparty/protobuf/stubs/hash.h>
|
|
#include <thirdparty/protobuf/stubs/status.h>
|
|
|
|
// Must be included last.
|
|
#include <thirdparty/protobuf/port_def.inc>
|
|
|
|
namespace google {
|
|
namespace protobuf {
|
|
namespace util {
|
|
namespace converter {
|
|
|
|
|
|
class ObjectLocationTracker;
|
|
|
|
// An ObjectWriter that can write protobuf bytes directly from writer events.
|
|
// This class does not support special types like Struct or Map. However, since
|
|
// this class supports raw protobuf, it can be used to provide support for
|
|
// special types by inheriting from it or by wrapping it.
|
|
//
|
|
// It also supports streaming.
|
|
class PROTOBUF_EXPORT ProtoWriter : public StructuredObjectWriter {
|
|
public:
|
|
// Constructor. Does not take ownership of any parameter passed in.
|
|
ProtoWriter(TypeResolver* type_resolver, const google::protobuf::Type& type,
|
|
strings::ByteSink* output, ErrorListener* listener);
|
|
~ProtoWriter() override;
|
|
|
|
// ObjectWriter methods.
|
|
ProtoWriter* StartObject(StringPiece name) override;
|
|
ProtoWriter* EndObject() override;
|
|
ProtoWriter* StartList(StringPiece name) override;
|
|
ProtoWriter* EndList() override;
|
|
ProtoWriter* RenderBool(StringPiece name, bool value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderInt32(StringPiece name, int32_t value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderUint32(StringPiece name, uint32_t value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderInt64(StringPiece name, int64_t value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderUint64(StringPiece name, uint64_t value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderDouble(StringPiece name, double value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderFloat(StringPiece name, float value) override {
|
|
return RenderDataPiece(name, DataPiece(value));
|
|
}
|
|
ProtoWriter* RenderString(StringPiece name,
|
|
StringPiece value) override {
|
|
return RenderDataPiece(name,
|
|
DataPiece(value, use_strict_base64_decoding()));
|
|
}
|
|
|
|
ProtoWriter* RenderBytes(StringPiece name, StringPiece value) override {
|
|
return RenderDataPiece(
|
|
name, DataPiece(value, false, use_strict_base64_decoding()));
|
|
}
|
|
|
|
ProtoWriter* RenderNull(StringPiece name) override {
|
|
return RenderDataPiece(name, DataPiece::NullData());
|
|
}
|
|
|
|
|
|
// Renders a DataPiece 'value' into a field whose wire type is determined
|
|
// from the given field 'name'.
|
|
virtual ProtoWriter* RenderDataPiece(StringPiece name,
|
|
const DataPiece& data);
|
|
|
|
|
|
// Returns the location tracker to use for tracking locations for errors.
|
|
const LocationTrackerInterface& location() {
|
|
return element_ != nullptr ? *element_ : *tracker_;
|
|
}
|
|
|
|
// When true, we finished writing to output a complete message.
|
|
bool done() override { return done_; }
|
|
|
|
// Returns the proto stream object.
|
|
io::CodedOutputStream* stream() { return stream_.get(); }
|
|
|
|
// Getters and mutators of invalid_depth_.
|
|
void IncrementInvalidDepth() { ++invalid_depth_; }
|
|
void DecrementInvalidDepth() { --invalid_depth_; }
|
|
int invalid_depth() { return invalid_depth_; }
|
|
|
|
ErrorListener* listener() { return listener_; }
|
|
|
|
const TypeInfo* typeinfo() { return typeinfo_; }
|
|
|
|
void set_ignore_unknown_fields(bool ignore_unknown_fields) {
|
|
ignore_unknown_fields_ = ignore_unknown_fields;
|
|
}
|
|
|
|
bool ignore_unknown_fields() { return ignore_unknown_fields_; }
|
|
|
|
void set_ignore_unknown_enum_values(bool ignore_unknown_enum_values) {
|
|
ignore_unknown_enum_values_ = ignore_unknown_enum_values;
|
|
}
|
|
|
|
void set_use_lower_camel_for_enums(bool use_lower_camel_for_enums) {
|
|
use_lower_camel_for_enums_ = use_lower_camel_for_enums;
|
|
}
|
|
|
|
void set_case_insensitive_enum_parsing(bool case_insensitive_enum_parsing) {
|
|
case_insensitive_enum_parsing_ = case_insensitive_enum_parsing;
|
|
}
|
|
|
|
void set_use_json_name_in_missing_fields(
|
|
bool use_json_name_in_missing_fields) {
|
|
use_json_name_in_missing_fields_ = use_json_name_in_missing_fields;
|
|
}
|
|
|
|
protected:
|
|
class PROTOBUF_EXPORT ProtoElement : public BaseElement,
|
|
public LocationTrackerInterface {
|
|
public:
|
|
// Constructor for the root element. No parent nor field.
|
|
ProtoElement(const TypeInfo* typeinfo, const google::protobuf::Type& type,
|
|
ProtoWriter* enclosing);
|
|
|
|
// Constructor for a field of an element.
|
|
ProtoElement(ProtoElement* parent, const google::protobuf::Field* field,
|
|
const google::protobuf::Type& type, bool is_list);
|
|
|
|
~ProtoElement() override {}
|
|
|
|
// Called just before the destructor for clean up:
|
|
// - reports any missing required fields
|
|
// - computes the space needed by the size field, and augment the
|
|
// length of all parent messages by this additional space.
|
|
// - releases and returns the parent pointer.
|
|
ProtoElement* pop();
|
|
|
|
// Accessors
|
|
// parent_field() may be nullptr if we are at root.
|
|
const google::protobuf::Field* parent_field() const {
|
|
return parent_field_;
|
|
}
|
|
const google::protobuf::Type& type() const { return type_; }
|
|
|
|
// Registers field for accounting required fields.
|
|
void RegisterField(const google::protobuf::Field* field);
|
|
|
|
// To report location on error messages.
|
|
std::string ToString() const override;
|
|
|
|
ProtoElement* parent() const override {
|
|
return static_cast<ProtoElement*>(BaseElement::parent());
|
|
}
|
|
|
|
// Returns true if the index is already taken by a preceding oneof input.
|
|
bool IsOneofIndexTaken(int32_t index);
|
|
|
|
// Marks the oneof 'index' as taken. Future inputs to this oneof will
|
|
// generate an error.
|
|
void TakeOneofIndex(int32_t index);
|
|
|
|
bool proto3() { return proto3_; }
|
|
|
|
private:
|
|
// Used for access to variables of the enclosing instance of
|
|
// ProtoWriter.
|
|
ProtoWriter* ow_;
|
|
|
|
// Describes the element as a field in the parent message.
|
|
// parent_field_ is nullptr if and only if this element is the root element.
|
|
const google::protobuf::Field* parent_field_;
|
|
|
|
// TypeInfo to lookup types.
|
|
const TypeInfo* typeinfo_;
|
|
|
|
// Whether the type_ is proto3 or not.
|
|
bool proto3_;
|
|
|
|
// Additional variables if this element is a message:
|
|
// (Root element is always a message).
|
|
// type_ : the type of this element.
|
|
// required_fields_ : set of required fields.
|
|
// size_index_ : index into ProtoWriter::size_insert_
|
|
// for later insertion of serialized message length.
|
|
const google::protobuf::Type& type_;
|
|
std::unordered_set<const google::protobuf::Field*> required_fields_;
|
|
const int size_index_;
|
|
|
|
// Tracks position in repeated fields, needed for LocationTrackerInterface.
|
|
int array_index_;
|
|
|
|
// Set of oneof indices already seen for the type_. Used to validate
|
|
// incoming messages so no more than one oneof is set.
|
|
std::vector<bool> oneof_indices_;
|
|
|
|
GOOGLE_DISALLOW_IMPLICIT_CONSTRUCTORS(ProtoElement);
|
|
};
|
|
|
|
// Container for inserting 'size' information at the 'pos' position.
|
|
struct SizeInfo {
|
|
const int pos;
|
|
int size;
|
|
};
|
|
|
|
ProtoWriter(const TypeInfo* typeinfo, const google::protobuf::Type& type,
|
|
strings::ByteSink* output, ErrorListener* listener);
|
|
|
|
ProtoElement* element() override { return element_.get(); }
|
|
|
|
// Helper methods for calling ErrorListener. See error_listener.h.
|
|
void InvalidName(StringPiece unknown_name, StringPiece message);
|
|
void InvalidValue(StringPiece type_name, StringPiece value);
|
|
void MissingField(StringPiece missing_name);
|
|
|
|
// Common code for BeginObject() and BeginList() that does invalid_depth_
|
|
// bookkeeping associated with name lookup.
|
|
const google::protobuf::Field* BeginNamed(StringPiece name,
|
|
bool is_list);
|
|
|
|
// Lookup the field in the current element. Looks in the base descriptor
|
|
// and in any extension. This will report an error if the field cannot be
|
|
// found when ignore_unknown_names_ is false or if multiple matching
|
|
// extensions are found.
|
|
const google::protobuf::Field* Lookup(StringPiece name);
|
|
|
|
// Lookup the field type in the type descriptor. Returns nullptr if the type
|
|
// is not known.
|
|
const google::protobuf::Type* LookupType(
|
|
const google::protobuf::Field* field);
|
|
|
|
// Write serialized output to the final output ByteSink, inserting all
|
|
// the size information for nested messages that are missing from the
|
|
// intermediate Cord buffer.
|
|
void WriteRootMessage();
|
|
|
|
// Helper method to write proto tags based on the given field.
|
|
void WriteTag(const google::protobuf::Field& field);
|
|
|
|
|
|
// Returns true if the field for type_ can be set as a oneof. If field is not
|
|
// a oneof type, this function does nothing and returns true.
|
|
// If another field for this oneof is already set, this function returns
|
|
// false. It also calls the appropriate error callback.
|
|
// unnormalized_name is used for error string.
|
|
bool ValidOneof(const google::protobuf::Field& field,
|
|
StringPiece unnormalized_name);
|
|
|
|
// Returns true if the field is repeated.
|
|
bool IsRepeated(const google::protobuf::Field& field);
|
|
|
|
// Starts an object given the field and the enclosing type.
|
|
ProtoWriter* StartObjectField(const google::protobuf::Field& field,
|
|
const google::protobuf::Type& type);
|
|
|
|
// Starts a list given the field and the enclosing type.
|
|
ProtoWriter* StartListField(const google::protobuf::Field& field,
|
|
const google::protobuf::Type& type);
|
|
|
|
// Renders a primitive field given the field and the enclosing type.
|
|
ProtoWriter* RenderPrimitiveField(const google::protobuf::Field& field,
|
|
const google::protobuf::Type& type,
|
|
const DataPiece& data);
|
|
|
|
private:
|
|
// Writes an ENUM field, including tag, to the stream.
|
|
static util::Status WriteEnum(int field_number, const DataPiece& data,
|
|
const google::protobuf::Enum* enum_type,
|
|
io::CodedOutputStream* stream,
|
|
bool use_lower_camel_for_enums,
|
|
bool case_insensitive_enum_parsing,
|
|
bool ignore_unknown_values);
|
|
|
|
// Variables for describing the structure of the input tree:
|
|
// master_type_: descriptor for the whole protobuf message.
|
|
// typeinfo_ : the TypeInfo object to lookup types.
|
|
const google::protobuf::Type& master_type_;
|
|
const TypeInfo* typeinfo_;
|
|
// Whether we own the typeinfo_ object.
|
|
bool own_typeinfo_;
|
|
|
|
// Indicates whether we finished writing root message completely.
|
|
bool done_;
|
|
|
|
// If true, don't report unknown field names to the listener.
|
|
bool ignore_unknown_fields_;
|
|
|
|
// If true, don't report unknown enum values to the listener.
|
|
bool ignore_unknown_enum_values_;
|
|
|
|
// If true, check if enum name in camel case or without underscore matches the
|
|
// field name.
|
|
bool use_lower_camel_for_enums_;
|
|
|
|
// If true, check if enum name in UPPER_CASE matches the field name.
|
|
bool case_insensitive_enum_parsing_;
|
|
|
|
// If true, use the json name in missing fields errors.
|
|
bool use_json_name_in_missing_fields_;
|
|
|
|
// Variable for internal state processing:
|
|
// element_ : the current element.
|
|
// size_insert_: sizes of nested messages.
|
|
// pos - position to insert the size field.
|
|
// size - size value to be inserted.
|
|
std::unique_ptr<ProtoElement> element_;
|
|
std::deque<SizeInfo> size_insert_;
|
|
|
|
// Variables for output generation:
|
|
// output_ : pointer to an external ByteSink for final user-visible output.
|
|
// buffer_ : buffer holding partial message before being ready for output_.
|
|
// adapter_ : internal adapter between CodedOutputStream and buffer_.
|
|
// stream_ : wrapper for writing tags and other encodings in wire format.
|
|
strings::ByteSink* output_;
|
|
std::string buffer_;
|
|
io::StringOutputStream adapter_;
|
|
std::unique_ptr<io::CodedOutputStream> stream_;
|
|
|
|
// Variables for error tracking and reporting:
|
|
// listener_ : a place to report any errors found.
|
|
// invalid_depth_: number of enclosing invalid nested messages.
|
|
// tracker_ : the root location tracker interface.
|
|
ErrorListener* listener_;
|
|
int invalid_depth_;
|
|
std::unique_ptr<LocationTrackerInterface> tracker_;
|
|
|
|
GOOGLE_DISALLOW_IMPLICIT_CONSTRUCTORS(ProtoWriter);
|
|
};
|
|
|
|
} // namespace converter
|
|
} // namespace util
|
|
} // namespace protobuf
|
|
} // namespace google
|
|
|
|
#include <thirdparty/protobuf/port_undef.inc>
|
|
|
|
#endif // GOOGLE_PROTOBUF_UTIL_INTERNAL_PROTO_WRITER_H__
|