Migrate `const std::string&` to alternatives:

- In most cases, migrate to `absl::string_view`.
 - When capturing a temporary string in a variable, use `const std::string` without the ref.

We avoid changing virtual function parameters in this CL, since that is a breaking change.

PiperOrigin-RevId: 647686085
pull/17249/head
Protobuf Team Bot 9 months ago committed by Copybara-Service
parent 99a335e79f
commit 10c00b82af
  1. 88
      conformance/binary_json_conformance_suite.cc
  2. 84
      conformance/binary_json_conformance_suite.h
  3. 43
      conformance/conformance_test.cc
  4. 30
      conformance/conformance_test.h
  5. 22
      conformance/text_format_conformance_suite.cc
  6. 23
      conformance/text_format_conformance_suite.h
  7. 14
      src/google/protobuf/compiler/annotation_test_util.cc
  8. 17
      src/google/protobuf/compiler/annotation_test_util.h
  9. 221
      src/google/protobuf/compiler/command_line_interface.cc
  10. 31
      src/google/protobuf/compiler/command_line_interface.h
  11. 5
      src/google/protobuf/compiler/command_line_interface_tester.cc
  12. 14
      src/google/protobuf/compiler/command_line_interface_tester.h
  13. 117
      src/google/protobuf/compiler/command_line_interface_unittest.cc
  14. 1
      src/google/protobuf/compiler/cpp/BUILD.bazel
  15. 8
      src/google/protobuf/compiler/cpp/bootstrap_unittest.cc
  16. 6
      src/google/protobuf/compiler/cpp/message.cc
  17. 6
      src/google/protobuf/compiler/cpp/metadata_test.cc
  18. 5
      src/google/protobuf/compiler/cpp/namespace_printer.cc
  19. 5
      src/google/protobuf/compiler/csharp/csharp_bootstrap_unittest.cc
  20. 1
      src/google/protobuf/compiler/csharp/csharp_field_base.h
  21. 7
      src/google/protobuf/compiler/importer.cc
  22. 4
      src/google/protobuf/compiler/importer.h
  23. 15
      src/google/protobuf/compiler/importer_unittest.cc
  24. 5
      src/google/protobuf/compiler/java/context.cc
  25. 4
      src/google/protobuf/compiler/java/context.h
  26. 8
      src/google/protobuf/compiler/java/doc_comment.cc
  27. 3
      src/google/protobuf/compiler/java/doc_comment.h
  28. 15
      src/google/protobuf/compiler/java/file.cc
  29. 5
      src/google/protobuf/compiler/java/file.h
  30. 1
      src/google/protobuf/compiler/java/full/message_field.cc
  31. 4
      src/google/protobuf/compiler/java/helpers.cc
  32. 4
      src/google/protobuf/compiler/java/helpers.h
  33. 3
      src/google/protobuf/compiler/java/shared_code_generator.cc
  34. 7
      src/google/protobuf/compiler/objectivec/field.cc
  35. 4
      src/google/protobuf/compiler/objectivec/field.h
  36. 14
      src/google/protobuf/compiler/objectivec/import_writer.cc
  37. 12
      src/google/protobuf/compiler/objectivec/import_writer.h
  38. 2
      src/google/protobuf/compiler/objectivec/message.cc
  39. 3
      src/google/protobuf/compiler/objectivec/message.h
  40. 23
      src/google/protobuf/compiler/objectivec/tf_decode_data.cc
  41. 10
      src/google/protobuf/compiler/objectivec/tf_decode_data.h
  42. 6
      src/google/protobuf/compiler/parser.cc
  43. 4
      src/google/protobuf/compiler/parser.h
  44. 9
      src/google/protobuf/compiler/parser_unittest.cc
  45. 12
      src/google/protobuf/compiler/php/names.cc
  46. 9
      src/google/protobuf/compiler/php/names.h
  47. 2
      src/google/protobuf/compiler/php/php_generator.cc
  48. 8
      src/google/protobuf/compiler/python/pyi_generator.cc
  49. 3
      src/google/protobuf/compiler/python/pyi_generator.h
  50. 10
      src/google/protobuf/compiler/subprocess.cc
  51. 3
      src/google/protobuf/compiler/subprocess.h
  52. 10
      src/google/protobuf/compiler/zip_writer.cc
  53. 3
      src/google/protobuf/compiler/zip_writer.h
  54. 174
      src/google/protobuf/descriptor.cc
  55. 14
      src/google/protobuf/io/coded_stream.cc
  56. 16
      src/google/protobuf/io/coded_stream.h
  57. 4
      src/google/protobuf/json/internal/descriptor_traits.h
  58. 4
      src/google/protobuf/json/internal/parser.cc
  59. 2
      src/google/protobuf/json/internal/parser.h
  60. 4
      src/google/protobuf/json/internal/parser_traits.h
  61. 4
      src/google/protobuf/json/internal/unparser.cc
  62. 2
      src/google/protobuf/json/internal/unparser.h
  63. 12
      src/google/protobuf/json/json.cc
  64. 20
      src/google/protobuf/json/json.h
  65. 38
      src/google/protobuf/testing/file.cc
  66. 24
      src/google/protobuf/testing/file.h
  67. 6
      src/google/protobuf/util/message_differencer.cc
  68. 3
      src/google/protobuf/util/message_differencer.h

@ -124,7 +124,7 @@ std::string fixed64(void* data) {
return std::string(reinterpret_cast<char*>(&data_le), 8); return std::string(reinterpret_cast<char*>(&data_le), 8);
} }
std::string delim(const std::string& buf) { std::string delim(absl::string_view buf) {
return absl::StrCat(varint(buf.size()), buf); return absl::StrCat(varint(buf.size()), buf);
} }
std::string u32(uint32_t u32) { return fixed32(&u32); } std::string u32(uint32_t u32) { return fixed32(&u32); }
@ -239,7 +239,7 @@ std::string UpperCase(std::string str) {
} }
bool IsProto3Default(FieldDescriptor::Type type, bool IsProto3Default(FieldDescriptor::Type type,
const std::string& binary_data) { absl::string_view binary_data) {
switch (type) { switch (type) {
case FieldDescriptor::TYPE_DOUBLE: case FieldDescriptor::TYPE_DOUBLE:
return binary_data == dbl(0); return binary_data == dbl(0);
@ -299,7 +299,7 @@ bool BinaryAndJsonConformanceSuite::ParseResponse(
const ConformanceRequestSetting& setting, Message* test_message) { const ConformanceRequestSetting& setting, Message* test_message) {
const ConformanceRequest& request = setting.GetRequest(); const ConformanceRequest& request = setting.GetRequest();
WireFormat requested_output = request.requested_output_format(); WireFormat requested_output = request.requested_output_format();
const std::string& test_name = setting.GetTestName(); const std::string test_name = setting.GetTestName();
ConformanceLevel level = setting.GetLevel(); ConformanceLevel level = setting.GetLevel();
switch (response.result_case()) { switch (response.result_case()) {
@ -421,8 +421,8 @@ void BinaryAndJsonConformanceSuite::RunDelimitedFieldTests() {
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
ExpectParseFailureForProtoWithProtoVersion(const std::string& proto, ExpectParseFailureForProtoWithProtoVersion(absl::string_view proto,
const std::string& test_name, absl::string_view test_name,
ConformanceLevel level) { ConformanceLevel level) {
MessageType prototype; MessageType prototype;
// We don't expect output, but if the program erroneously accepts the protobuf // We don't expect output, but if the program erroneously accepts the protobuf
@ -451,7 +451,7 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
// Expect that this precise protobuf will cause a parse error. // Expect that this precise protobuf will cause a parse error.
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForProto( void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForProto(
const std::string& proto, const std::string& test_name, absl::string_view proto, absl::string_view test_name,
ConformanceLevel level) { ConformanceLevel level) {
ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level); ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level);
} }
@ -463,16 +463,16 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForProto(
// TODO: implement the second of these. // TODO: implement the second of these.
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl< void BinaryAndJsonConformanceSuiteImpl<
MessageType>::ExpectHardParseFailureForProto(const std::string& proto, MessageType>::ExpectHardParseFailureForProto(absl::string_view proto,
const std::string& test_name, absl::string_view test_name,
ConformanceLevel level) { ConformanceLevel level) {
return ExpectParseFailureForProto(proto, test_name, level); return ExpectParseFailureForProto(proto, test_name, level);
} }
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidJsonTest( void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidJsonTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_json, const std::string& equivalent_text_format) { absl::string_view input_json, absl::string_view equivalent_text_format) {
MessageType prototype; MessageType prototype;
RunValidJsonTestWithMessage(test_name, level, input_json, RunValidJsonTestWithMessage(test_name, level, input_json,
equivalent_text_format, prototype); equivalent_text_format, prototype);
@ -480,10 +480,10 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidJsonTest(
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
RunValidJsonTestWithMessage(const std::string& test_name, RunValidJsonTestWithMessage(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const std::string& equivalent_text_format, absl::string_view equivalent_text_format,
const Message& prototype) { const Message& prototype) {
ConformanceRequestSetting setting1( ConformanceRequestSetting setting1(
level, conformance::JSON, conformance::PROTOBUF, conformance::JSON_TEST, level, conformance::JSON, conformance::PROTOBUF, conformance::JSON_TEST,
@ -498,8 +498,8 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
RunValidJsonTestWithProtobufInput( RunValidJsonTestWithProtobufInput(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const MessageType& input, const std::string& equivalent_text_format) { const MessageType& input, absl::string_view equivalent_text_format) {
ConformanceRequestSetting setting( ConformanceRequestSetting setting(
level, conformance::PROTOBUF, conformance::JSON, conformance::JSON_TEST, level, conformance::PROTOBUF, conformance::JSON, conformance::JSON_TEST,
input, test_name, input.SerializeAsString()); input, test_name, input.SerializeAsString());
@ -508,10 +508,10 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
RunValidJsonIgnoreUnknownTest(const std::string& test_name, RunValidJsonIgnoreUnknownTest(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting setting( ConformanceRequestSetting setting(
level, conformance::JSON, conformance::PROTOBUF, level, conformance::JSON, conformance::PROTOBUF,
@ -522,9 +522,9 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuite::RunValidBinaryProtobufTest( void BinaryAndJsonConformanceSuite::RunValidBinaryProtobufTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_protobuf, absl::string_view input_protobuf,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting binary_to_binary( ConformanceRequestSetting binary_to_binary(
@ -535,9 +535,9 @@ void BinaryAndJsonConformanceSuite::RunValidBinaryProtobufTest(
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuite::RunValidProtobufTest( void BinaryAndJsonConformanceSuite::RunValidProtobufTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_protobuf, absl::string_view input_protobuf,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting binary_to_binary( ConformanceRequestSetting binary_to_binary(
@ -553,24 +553,24 @@ void BinaryAndJsonConformanceSuite::RunValidProtobufTest(
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidProtobufTest( void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidProtobufTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_protobuf, absl::string_view input_protobuf,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
suite_.RunValidProtobufTest<MessageType>(test_name, level, input_protobuf, suite_.RunValidProtobufTest<MessageType>(test_name, level, input_protobuf,
equivalent_text_format); equivalent_text_format);
} }
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidBinaryProtobufTest( void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidBinaryProtobufTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_protobuf) { absl::string_view input_protobuf) {
RunValidBinaryProtobufTest(test_name, level, input_protobuf, input_protobuf); RunValidBinaryProtobufTest(test_name, level, input_protobuf, input_protobuf);
} }
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidBinaryProtobufTest( void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidBinaryProtobufTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_protobuf, const std::string& expected_protobuf) { absl::string_view input_protobuf, absl::string_view expected_protobuf) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting setting( ConformanceRequestSetting setting(
level, conformance::PROTOBUF, conformance::PROTOBUF, level, conformance::PROTOBUF, conformance::PROTOBUF,
@ -580,8 +580,8 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::RunValidBinaryProtobufTest(
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
RunBinaryPerformanceMergeMessageWithField(const std::string& test_name, RunBinaryPerformanceMergeMessageWithField(absl::string_view test_name,
const std::string& field_proto) { absl::string_view field_proto) {
std::string message_tag = tag(27, WireFormatLite::WIRETYPE_LENGTH_DELIMITED); std::string message_tag = tag(27, WireFormatLite::WIRETYPE_LENGTH_DELIMITED);
std::string message_proto = absl::StrCat(message_tag, delim(field_proto)); std::string message_proto = absl::StrCat(message_tag, delim(field_proto));
@ -592,7 +592,7 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
std::string multiple_repeated_field_proto; std::string multiple_repeated_field_proto;
for (size_t i = 0; i < kPerformanceRepeatCount; i++) { for (size_t i = 0; i < kPerformanceRepeatCount; i++) {
multiple_repeated_field_proto.append(field_proto); multiple_repeated_field_proto.append(std::string(field_proto));
} }
std::string expected_proto = std::string expected_proto =
absl::StrCat(message_tag, delim(multiple_repeated_field_proto)); absl::StrCat(message_tag, delim(multiple_repeated_field_proto));
@ -602,10 +602,10 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<MessageType>::
RunValidProtobufTestWithMessage(const std::string& test_name, RunValidProtobufTestWithMessage(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const Message* input, const Message* input,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
RunValidProtobufTest(test_name, level, input->SerializeAsString(), RunValidProtobufTest(test_name, level, input->SerializeAsString(),
equivalent_text_format); equivalent_text_format);
} }
@ -617,9 +617,9 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::
template <typename MessageType> // the JSON output directly. template <typename MessageType> // the JSON output directly.
void BinaryAndJsonConformanceSuiteImpl< void BinaryAndJsonConformanceSuiteImpl<
MessageType>::RunValidJsonTestWithValidator(const std::string& test_name, MessageType>::RunValidJsonTestWithValidator(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const Validator& validator) { const Validator& validator) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting setting(level, conformance::JSON, conformance::JSON, ConformanceRequestSetting setting(level, conformance::JSON, conformance::JSON,
@ -663,8 +663,8 @@ void BinaryAndJsonConformanceSuiteImpl<
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForJson( void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForJson(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_json) { absl::string_view input_json) {
MessageType prototype; MessageType prototype;
// We don't expect output, but if the program erroneously accepts the protobuf // We don't expect output, but if the program erroneously accepts the protobuf
// we let it send its response as this. We must not leave it unspecified. // we let it send its response as this. We must not leave it unspecified.
@ -689,10 +689,10 @@ void BinaryAndJsonConformanceSuiteImpl<MessageType>::ExpectParseFailureForJson(
} }
template <typename MessageType> template <typename MessageType>
void BinaryAndJsonConformanceSuiteImpl<MessageType>:: void BinaryAndJsonConformanceSuiteImpl<
ExpectSerializeFailureForJson(const std::string& test_name, MessageType>::ExpectSerializeFailureForJson(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& text_format) { absl::string_view text_format) {
MessageType payload_message; MessageType payload_message;
ABSL_CHECK(TextFormat::ParseFromString(text_format, &payload_message)) ABSL_CHECK(TextFormat::ParseFromString(text_format, &payload_message))
<< "Failed to parse: " << text_format; << "Failed to parse: " << text_format;

@ -39,16 +39,15 @@ class BinaryAndJsonConformanceSuite : public ConformanceTestSuite {
} }
template <typename MessageType> template <typename MessageType>
void RunValidBinaryProtobufTest(const std::string& test_name, void RunValidBinaryProtobufTest(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_protobuf, absl::string_view input_protobuf,
const std::string& equivalent_text_format); absl::string_view equivalent_text_format);
template <typename MessageType> template <typename MessageType>
void RunValidProtobufTest(const std::string& test_name, void RunValidProtobufTest(absl::string_view test_name, ConformanceLevel level,
ConformanceLevel level, absl::string_view input_protobuf,
const std::string& input_protobuf, absl::string_view equivalent_text_format);
const std::string& equivalent_text_format);
void RunDelimitedFieldTests(); void RunDelimitedFieldTests();
@ -88,58 +87,57 @@ class BinaryAndJsonConformanceSuiteImpl {
void RunJsonTestsForValue(); void RunJsonTestsForValue();
void RunJsonTestsForAny(); void RunJsonTestsForAny();
void RunJsonTestsForUnknownEnumStringValues(); void RunJsonTestsForUnknownEnumStringValues();
void RunValidJsonTest(const std::string& test_name, ConformanceLevel level, void RunValidJsonTest(absl::string_view test_name, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const std::string& equivalent_text_format); absl::string_view equivalent_text_format);
void RunValidJsonTestWithMessage(const std::string& test_name, void RunValidJsonTestWithMessage(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const std::string& equivalent_text_forma, absl::string_view equivalent_text_forma,
const Message& prototype); const Message& prototype);
void RunValidJsonTestWithProtobufInput( void RunValidJsonTestWithProtobufInput(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const MessageType& input, const std::string& equivalent_text_format); const MessageType& input, absl::string_view equivalent_text_format);
void RunValidJsonIgnoreUnknownTest(const std::string& test_name, void RunValidJsonIgnoreUnknownTest(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const std::string& equivalent_text_format); absl::string_view equivalent_text_format);
void RunValidProtobufTest(const std::string& test_name, void RunValidProtobufTest(absl::string_view test_name, ConformanceLevel level,
ConformanceLevel level, absl::string_view input_protobuf,
const std::string& input_protobuf, absl::string_view equivalent_text_format);
const std::string& equivalent_text_format); void RunValidBinaryProtobufTest(absl::string_view test_name,
void RunValidBinaryProtobufTest(const std::string& test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_protobuf); absl::string_view input_protobuf);
void RunValidBinaryProtobufTest(const std::string& test_name, void RunValidBinaryProtobufTest(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_protobuf, absl::string_view input_protobuf,
const std::string& expected_protobuf); absl::string_view expected_protobuf);
void RunBinaryPerformanceMergeMessageWithField( void RunBinaryPerformanceMergeMessageWithField(absl::string_view test_name,
const std::string& test_name, const std::string& field_proto); absl::string_view field_proto);
void RunValidProtobufTestWithMessage( void RunValidProtobufTestWithMessage(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level, const Message* input,
const Message* input, const std::string& equivalent_text_format); absl::string_view equivalent_text_format);
typedef std::function<bool(const Json::Value&)> Validator; typedef std::function<bool(const Json::Value&)> Validator;
void RunValidJsonTestWithValidator(const std::string& test_name, void RunValidJsonTestWithValidator(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json, absl::string_view input_json,
const Validator& validator); const Validator& validator);
void ExpectParseFailureForJson(const std::string& test_name, void ExpectParseFailureForJson(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_json); absl::string_view input_json);
void ExpectSerializeFailureForJson(const std::string& test_name, void ExpectSerializeFailureForJson(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& text_format); absl::string_view text_format);
void ExpectParseFailureForProtoWithProtoVersion(const std::string& proto, void ExpectParseFailureForProtoWithProtoVersion(absl::string_view proto,
const std::string& test_name, absl::string_view test_name,
ConformanceLevel level); ConformanceLevel level);
void ExpectParseFailureForProto(const std::string& proto, void ExpectParseFailureForProto(absl::string_view proto,
const std::string& test_name, absl::string_view test_name,
ConformanceLevel level); ConformanceLevel level);
void ExpectHardParseFailureForProto(const std::string& proto, void ExpectHardParseFailureForProto(absl::string_view proto,
const std::string& test_name, absl::string_view test_name,
ConformanceLevel level); ConformanceLevel level);
void TestPrematureEOFForType(google::protobuf::FieldDescriptor::Type type); void TestPrematureEOFForType(google::protobuf::FieldDescriptor::Type type);
void TestIllegalTags(); void TestIllegalTags();

@ -37,7 +37,7 @@ using std::string;
namespace { namespace {
static std::string ToOctString(const std::string& binary_string) { static std::string ToOctString(absl::string_view binary_string) {
std::string oct_string; std::string oct_string;
for (size_t i = 0; i < binary_string.size(); i++) { for (size_t i = 0; i < binary_string.size(); i++) {
uint8_t c = binary_string.at(i); uint8_t c = binary_string.at(i);
@ -98,7 +98,7 @@ ConformanceTestSuite::ConformanceRequestSetting::ConformanceRequestSetting(
ConformanceLevel level, conformance::WireFormat input_format, ConformanceLevel level, conformance::WireFormat input_format,
conformance::WireFormat output_format, conformance::WireFormat output_format,
conformance::TestCategory test_category, const Message& prototype_message, conformance::TestCategory test_category, const Message& prototype_message,
const std::string& test_name, const std::string& input) absl::string_view test_name, absl::string_view input)
: level_(level), : level_(level),
input_format_(input_format), input_format_(input_format),
output_format_(output_format), output_format_(output_format),
@ -265,19 +265,19 @@ ConformanceResponse ConformanceTestSuite::TruncateResponse(
return debug_response; return debug_response;
} }
void ConformanceTestSuite::ReportSuccess(const std::string& test_name) { void ConformanceTestSuite::ReportSuccess(absl::string_view test_name) {
if (expected_to_fail_.erase(test_name) != 0) { if (expected_to_fail_.erase(test_name) != 0) {
absl::StrAppendFormat( absl::StrAppendFormat(
&output_, &output_,
"ERROR: test %s is in the failure list, but test succeeded. " "ERROR: test %s is in the failure list, but test succeeded. "
"Remove it from the failure list.\n", "Remove it from the failure list.\n",
test_name); test_name);
unexpected_succeeding_tests_.insert(test_name); unexpected_succeeding_tests_.insert(std::string(test_name));
} }
successes_++; successes_++;
} }
void ConformanceTestSuite::ReportFailure(const std::string& test_name, void ConformanceTestSuite::ReportFailure(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const ConformanceRequest& request, const ConformanceRequest& request,
const ConformanceResponse& response, const ConformanceResponse& response,
@ -289,7 +289,7 @@ void ConformanceTestSuite::ReportFailure(const std::string& test_name,
absl::StrAppendFormat(&output_, "WARNING, test=%s: ", test_name); absl::StrAppendFormat(&output_, "WARNING, test=%s: ", test_name);
} else { } else {
absl::StrAppendFormat(&output_, "ERROR, test=%s: ", test_name); absl::StrAppendFormat(&output_, "ERROR, test=%s: ", test_name);
unexpected_failing_tests_.insert(test_name); unexpected_failing_tests_.insert(std::string(test_name));
} }
absl::StrAppendFormat(&output_, "%s, request=%s, response=%s\n", message, absl::StrAppendFormat(&output_, "%s, request=%s, response=%s\n", message,
@ -297,7 +297,7 @@ void ConformanceTestSuite::ReportFailure(const std::string& test_name,
TruncateResponse(response).ShortDebugString()); TruncateResponse(response).ShortDebugString());
} }
void ConformanceTestSuite::ReportSkip(const std::string& test_name, void ConformanceTestSuite::ReportSkip(absl::string_view test_name,
const ConformanceRequest& request, const ConformanceRequest& request,
const ConformanceResponse& response) { const ConformanceResponse& response) {
if (verbose_) { if (verbose_) {
@ -305,12 +305,12 @@ void ConformanceTestSuite::ReportSkip(const std::string& test_name,
&output_, "SKIPPED, test=%s request=%s, response=%s\n", test_name, &output_, "SKIPPED, test=%s request=%s, response=%s\n", test_name,
request.ShortDebugString(), response.ShortDebugString()); request.ShortDebugString(), response.ShortDebugString());
} }
skipped_.insert(test_name); skipped_.insert(std::string(test_name));
} }
void ConformanceTestSuite::RunValidInputTest( void ConformanceTestSuite::RunValidInputTest(
const ConformanceRequestSetting& setting, const ConformanceRequestSetting& setting,
const std::string& equivalent_text_format) { absl::string_view equivalent_text_format) {
std::unique_ptr<Message> reference_message(setting.NewTestMessage()); std::unique_ptr<Message> reference_message(setting.NewTestMessage());
ABSL_CHECK(TextFormat::ParseFromString(equivalent_text_format, ABSL_CHECK(TextFormat::ParseFromString(equivalent_text_format,
reference_message.get())) reference_message.get()))
@ -323,7 +323,7 @@ void ConformanceTestSuite::RunValidInputTest(
void ConformanceTestSuite::RunValidBinaryInputTest( void ConformanceTestSuite::RunValidBinaryInputTest(
const ConformanceRequestSetting& setting, const ConformanceRequestSetting& setting,
const std::string& equivalent_wire_format, bool require_same_wire_format) { absl::string_view equivalent_wire_format, bool require_same_wire_format) {
const ConformanceRequest& request = setting.GetRequest(); const ConformanceRequest& request = setting.GetRequest();
ConformanceResponse response; ConformanceResponse response;
RunTest(setting.GetTestName(), request, &response); RunTest(setting.GetTestName(), request, &response);
@ -333,12 +333,12 @@ void ConformanceTestSuite::RunValidBinaryInputTest(
void ConformanceTestSuite::VerifyResponse( void ConformanceTestSuite::VerifyResponse(
const ConformanceRequestSetting& setting, const ConformanceRequestSetting& setting,
const std::string& equivalent_wire_format, absl::string_view equivalent_wire_format,
const ConformanceResponse& response, bool need_report_success, const ConformanceResponse& response, bool need_report_success,
bool require_same_wire_format) { bool require_same_wire_format) {
std::unique_ptr<Message> test_message(setting.NewTestMessage()); std::unique_ptr<Message> test_message(setting.NewTestMessage());
const ConformanceRequest& request = setting.GetRequest(); const ConformanceRequest& request = setting.GetRequest();
const std::string& test_name = setting.GetTestName(); const std::string test_name = setting.GetTestName();
ConformanceLevel level = setting.GetLevel(); ConformanceLevel level = setting.GetLevel();
std::unique_ptr<Message> reference_message = setting.NewTestMessage(); std::unique_ptr<Message> reference_message = setting.NewTestMessage();
@ -379,7 +379,7 @@ void ConformanceTestSuite::VerifyResponse(
if (require_same_wire_format) { if (require_same_wire_format) {
ABSL_DCHECK_EQ(response.result_case(), ABSL_DCHECK_EQ(response.result_case(),
ConformanceResponse::kProtobufPayload); ConformanceResponse::kProtobufPayload);
const std::string& protobuf_payload = response.protobuf_payload(); absl::string_view protobuf_payload = response.protobuf_payload();
check = equivalent_wire_format == protobuf_payload; check = equivalent_wire_format == protobuf_payload;
differences = absl::StrCat("Expect: ", ToOctString(equivalent_wire_format), differences = absl::StrCat("Expect: ", ToOctString(equivalent_wire_format),
", but got: ", ToOctString(protobuf_payload)); ", but got: ", ToOctString(protobuf_payload));
@ -399,10 +399,10 @@ void ConformanceTestSuite::VerifyResponse(
} }
} }
void ConformanceTestSuite::RunTest(const std::string& test_name, void ConformanceTestSuite::RunTest(absl::string_view test_name,
const ConformanceRequest& request, const ConformanceRequest& request,
ConformanceResponse* response) { ConformanceResponse* response) {
if (test_names_.insert(test_name).second == false) { if (test_names_.insert(std::string(test_name)).second == false) {
ABSL_LOG(FATAL) << "Duplicated test name: " << test_name; ABSL_LOG(FATAL) << "Duplicated test name: " << test_name;
} }
@ -410,7 +410,8 @@ void ConformanceTestSuite::RunTest(const std::string& test_name,
std::string serialized_response; std::string serialized_response;
request.SerializeToString(&serialized_request); request.SerializeToString(&serialized_request);
runner_->RunTest(test_name, serialized_request, &serialized_response); runner_->RunTest(std::string(test_name), serialized_request,
&serialized_response);
if (!response->ParseFromString(serialized_response)) { if (!response->ParseFromString(serialized_response)) {
response->Clear(); response->Clear();
@ -443,13 +444,13 @@ std::string ConformanceTestSuite::WireFormatToString(WireFormat wire_format) {
return ""; return "";
} }
void ConformanceTestSuite::AddExpectedFailedTest(const std::string& test_name) { void ConformanceTestSuite::AddExpectedFailedTest(absl::string_view test_name) {
expected_to_fail_.insert(test_name); expected_to_fail_.insert(std::string(test_name));
} }
bool ConformanceTestSuite::RunSuite(ConformanceTestRunner* runner, bool ConformanceTestSuite::RunSuite(ConformanceTestRunner* runner,
std::string* output, std::string* output,
const std::string& filename, absl::string_view filename,
conformance::FailureSet* failure_list) { conformance::FailureSet* failure_list) {
runner_ = runner; runner_ = runner;
successes_ = 0; successes_ = 0;
@ -461,9 +462,9 @@ bool ConformanceTestSuite::RunSuite(ConformanceTestRunner* runner,
output_ = "\nCONFORMANCE TEST BEGIN ====================================\n\n"; output_ = "\nCONFORMANCE TEST BEGIN ====================================\n\n";
failure_list_filename_ = filename; failure_list_filename_ = std::string(filename);
expected_to_fail_.clear(); expected_to_fail_.clear();
for (const std::string& failure : failure_list->failure()) { for (absl::string_view failure : failure_list->failure()) {
AddExpectedFailedTest(failure); AddExpectedFailedTest(failure);
} }
RunSuiteImpl(); RunSuiteImpl();

@ -67,7 +67,7 @@ class ForkPipeRunner : public ConformanceTestRunner {
static int Run(int argc, char* argv[], static int Run(int argc, char* argv[],
const std::vector<ConformanceTestSuite*>& suites); const std::vector<ConformanceTestSuite*>& suites);
ForkPipeRunner(const std::string& executable, ForkPipeRunner(absl::string_view executable,
const std::vector<std::string>& executable_args, const std::vector<std::string>& executable_args,
bool performance) bool performance)
: child_pid_(-1), : child_pid_(-1),
@ -75,7 +75,7 @@ class ForkPipeRunner : public ConformanceTestRunner {
executable_args_(executable_args), executable_args_(executable_args),
performance_(performance) {} performance_(performance) {}
explicit ForkPipeRunner(const std::string& executable) explicit ForkPipeRunner(absl::string_view executable)
: child_pid_(-1), executable_(executable) {} : child_pid_(-1), executable_(executable) {}
virtual ~ForkPipeRunner() {} virtual ~ForkPipeRunner() {}
@ -156,8 +156,8 @@ class ConformanceTestSuite {
// By default, this would return --failure_list // By default, this would return --failure_list
std::string GetFailureListFlagName() { return failure_list_flag_name_; } std::string GetFailureListFlagName() { return failure_list_flag_name_; }
void SetFailureListFlagName(const std::string& failure_list_flag_name) { void SetFailureListFlagName(absl::string_view failure_list_flag_name) {
failure_list_flag_name_ = failure_list_flag_name; failure_list_flag_name_ = std::string(failure_list_flag_name);
} }
// Sets the path of the output directory. // Sets the path of the output directory.
@ -171,7 +171,7 @@ class ConformanceTestSuite {
// The filename here is *only* used to create/format useful error messages for // The filename here is *only* used to create/format useful error messages for
// how to update the failure list. We do NOT read this file at all. // how to update the failure list. We do NOT read this file at all.
bool RunSuite(ConformanceTestRunner* runner, std::string* output, bool RunSuite(ConformanceTestRunner* runner, std::string* output,
const std::string& filename, absl::string_view filename,
conformance::FailureSet* failure_list); conformance::FailureSet* failure_list);
protected: protected:
@ -199,8 +199,8 @@ class ConformanceTestSuite {
conformance::WireFormat output_format, conformance::WireFormat output_format,
conformance::TestCategory test_category, conformance::TestCategory test_category,
const Message& prototype_message, const Message& prototype_message,
const std::string& test_name, absl::string_view test_name,
const std::string& input); absl::string_view input);
virtual ~ConformanceRequestSetting() {} virtual ~ConformanceRequestSetting() {}
std::unique_ptr<Message> NewTestMessage() const; std::unique_ptr<Message> NewTestMessage() const;
@ -249,7 +249,7 @@ class ConformanceTestSuite {
Message* test_message) = 0; Message* test_message) = 0;
void VerifyResponse(const ConformanceRequestSetting& setting, void VerifyResponse(const ConformanceRequestSetting& setting,
const std::string& equivalent_wire_format, absl::string_view equivalent_wire_format,
const conformance::ConformanceResponse& response, const conformance::ConformanceResponse& response,
bool need_report_success, bool require_same_wire_format); bool need_report_success, bool require_same_wire_format);
@ -259,26 +259,26 @@ class ConformanceTestSuite {
conformance::ConformanceResponse TruncateResponse( conformance::ConformanceResponse TruncateResponse(
const conformance::ConformanceResponse& response); const conformance::ConformanceResponse& response);
void ReportSuccess(const std::string& test_name); void ReportSuccess(absl::string_view test_name);
void ReportFailure(const std::string& test_name, ConformanceLevel level, void ReportFailure(absl::string_view test_name, ConformanceLevel level,
const conformance::ConformanceRequest& request, const conformance::ConformanceRequest& request,
const conformance::ConformanceResponse& response, const conformance::ConformanceResponse& response,
absl::string_view message); absl::string_view message);
void ReportSkip(const std::string& test_name, void ReportSkip(absl::string_view test_name,
const conformance::ConformanceRequest& request, const conformance::ConformanceRequest& request,
const conformance::ConformanceResponse& response); const conformance::ConformanceResponse& response);
void RunValidInputTest(const ConformanceRequestSetting& setting, void RunValidInputTest(const ConformanceRequestSetting& setting,
const std::string& equivalent_text_format); absl::string_view equivalent_text_format);
void RunValidBinaryInputTest(const ConformanceRequestSetting& setting, void RunValidBinaryInputTest(const ConformanceRequestSetting& setting,
const std::string& equivalent_wire_format, absl::string_view equivalent_wire_format,
bool require_same_wire_format = false); bool require_same_wire_format = false);
void RunTest(const std::string& test_name, void RunTest(absl::string_view test_name,
const conformance::ConformanceRequest& request, const conformance::ConformanceRequest& request,
conformance::ConformanceResponse* response); conformance::ConformanceResponse* response);
void AddExpectedFailedTest(const std::string& test_name); void AddExpectedFailedTest(absl::string_view test_name);
virtual void RunSuiteImpl() = 0; virtual void RunSuiteImpl() = 0;

@ -158,8 +158,8 @@ TextFormatConformanceTestSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl<MessageType>::ExpectParseFailure( void TextFormatConformanceTestSuiteImpl<MessageType>::ExpectParseFailure(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input) { absl::string_view input) {
MessageType prototype; MessageType prototype;
// We don't expect output, but if the program erroneously accepts the protobuf // We don't expect output, but if the program erroneously accepts the protobuf
// we let it send its response as this. We must not leave it unspecified. // we let it send its response as this. We must not leave it unspecified.
@ -185,17 +185,17 @@ void TextFormatConformanceTestSuiteImpl<MessageType>::ExpectParseFailure(
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl<MessageType>::RunValidTextFormatTest( void TextFormatConformanceTestSuiteImpl<MessageType>::RunValidTextFormatTest(
const std::string& test_name, ConformanceLevel level, absl::string_view test_name, ConformanceLevel level,
const std::string& input_text) { absl::string_view input_text) {
MessageType prototype; MessageType prototype;
RunValidTextFormatTestWithMessage(test_name, level, input_text, prototype); RunValidTextFormatTestWithMessage(test_name, level, input_text, prototype);
} }
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl<MessageType>:: void TextFormatConformanceTestSuiteImpl<MessageType>::
RunValidTextFormatTestWithMessage(const std::string& test_name, RunValidTextFormatTestWithMessage(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_text, absl::string_view input_text,
const Message& message) { const Message& message) {
ConformanceRequestSetting setting1( ConformanceRequestSetting setting1(
level, conformance::TEXT_FORMAT, conformance::PROTOBUF, level, conformance::TEXT_FORMAT, conformance::PROTOBUF,
@ -209,10 +209,10 @@ void TextFormatConformanceTestSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl<MessageType>:: void TextFormatConformanceTestSuiteImpl<MessageType>::
RunValidTextFormatTestWithExpected(const std::string& test_name, RunValidTextFormatTestWithExpected(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_text, absl::string_view input_text,
const std::string& expected_text) { absl::string_view expected_text) {
MessageType prototype; MessageType prototype;
ConformanceRequestSetting setting1( ConformanceRequestSetting setting1(
level, conformance::TEXT_FORMAT, conformance::PROTOBUF, level, conformance::TEXT_FORMAT, conformance::PROTOBUF,
@ -226,7 +226,7 @@ void TextFormatConformanceTestSuiteImpl<MessageType>::
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl< void TextFormatConformanceTestSuiteImpl<
MessageType>::RunValidUnknownTextFormatTest(const std::string& test_name, MessageType>::RunValidUnknownTextFormatTest(absl::string_view test_name,
const Message& message) { const Message& message) {
std::string serialized_input; std::string serialized_input;
message.SerializeToString(&serialized_input); message.SerializeToString(&serialized_input);
@ -836,7 +836,7 @@ void TextFormatConformanceTestSuiteImpl<
template <typename MessageType> template <typename MessageType>
void TextFormatConformanceTestSuiteImpl<MessageType>:: void TextFormatConformanceTestSuiteImpl<MessageType>::
TestTextFormatPerformanceMergeMessageWithRepeatedField( TestTextFormatPerformanceMergeMessageWithRepeatedField(
const std::string& test_type_name, const std::string& message_field) { absl::string_view test_type_name, absl::string_view message_field) {
std::string recursive_message = std::string recursive_message =
absl::StrCat("recursive_message { ", message_field, " }"); absl::StrCat("recursive_message { ", message_field, " }");

@ -10,6 +10,7 @@
#include <string> #include <string>
#include "absl/strings/string_view.h"
#include "conformance_test.h" #include "conformance_test.h"
#include "google/protobuf/message.h" #include "google/protobuf/message.h"
@ -54,22 +55,22 @@ class TextFormatConformanceTestSuiteImpl {
void RunAnyTests(); void RunAnyTests();
void RunTextFormatPerformanceTests(); void RunTextFormatPerformanceTests();
void RunValidTextFormatTest(const std::string& test_name, void RunValidTextFormatTest(absl::string_view test_name,
ConformanceLevel level, const std::string& input); ConformanceLevel level, absl::string_view input);
void RunValidTextFormatTestWithExpected(const std::string& test_name, void RunValidTextFormatTestWithExpected(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_text, absl::string_view input_text,
const std::string& expected_text); absl::string_view expected_text);
void RunValidUnknownTextFormatTest(const std::string& test_name, void RunValidUnknownTextFormatTest(absl::string_view test_name,
const Message& message); const Message& message);
void RunValidTextFormatTestWithMessage(const std::string& test_name, void RunValidTextFormatTestWithMessage(absl::string_view test_name,
ConformanceLevel level, ConformanceLevel level,
const std::string& input_text, absl::string_view input_text,
const Message& message); const Message& message);
void ExpectParseFailure(const std::string& test_name, ConformanceLevel level, void ExpectParseFailure(absl::string_view test_name, ConformanceLevel level,
const std::string& input); absl::string_view input);
void TestTextFormatPerformanceMergeMessageWithRepeatedField( void TestTextFormatPerformanceMergeMessageWithRepeatedField(
const std::string& test_type_name, const std::string& message_field); absl::string_view test_type_name, absl::string_view message_field);
TextFormatConformanceTestSuite& suite_; TextFormatConformanceTestSuite& suite_;
}; };

@ -71,7 +71,7 @@ bool RunProtoCompiler(const std::string& filename,
return cli->Run(5, argv) == 0; return cli->Run(5, argv) == 0;
} }
bool DecodeMetadata(const std::string& path, GeneratedCodeInfo* info) { bool DecodeMetadata(absl::string_view path, GeneratedCodeInfo* info) {
std::string data; std::string data;
ABSL_CHECK_OK(File::GetContents(path, &data, true)); ABSL_CHECK_OK(File::GetContents(path, &data, true));
io::ArrayInputStream input(data.data(), data.size()); io::ArrayInputStream input(data.data(), data.size());
@ -79,7 +79,7 @@ bool DecodeMetadata(const std::string& path, GeneratedCodeInfo* info) {
} }
void FindAnnotationsOnPath( void FindAnnotationsOnPath(
const GeneratedCodeInfo& info, const std::string& source_file, const GeneratedCodeInfo& info, absl::string_view source_file,
const std::vector<int>& path, const std::vector<int>& path,
std::vector<const GeneratedCodeInfo::Annotation*>* annotations) { std::vector<const GeneratedCodeInfo::Annotation*>* annotations) {
for (int i = 0; i < info.annotation_size(); ++i) { for (int i = 0; i < info.annotation_size(); ++i) {
@ -101,7 +101,7 @@ void FindAnnotationsOnPath(
} }
const GeneratedCodeInfo::Annotation* FindAnnotationOnPath( const GeneratedCodeInfo::Annotation* FindAnnotationOnPath(
const GeneratedCodeInfo& info, const std::string& source_file, const GeneratedCodeInfo& info, absl::string_view source_file,
const std::vector<int>& path) { const std::vector<int>& path) {
std::vector<const GeneratedCodeInfo::Annotation*> annotations; std::vector<const GeneratedCodeInfo::Annotation*> annotations;
FindAnnotationsOnPath(info, source_file, path, &annotations); FindAnnotationsOnPath(info, source_file, path, &annotations);
@ -112,9 +112,9 @@ const GeneratedCodeInfo::Annotation* FindAnnotationOnPath(
} }
bool AtLeastOneAnnotationMatchesSubstring( bool AtLeastOneAnnotationMatchesSubstring(
const std::string& file_content, absl::string_view file_content,
const std::vector<const GeneratedCodeInfo::Annotation*>& annotations, const std::vector<const GeneratedCodeInfo::Annotation*>& annotations,
const std::string& expected_text, absl::string_view expected_text,
absl::optional<GeneratedCodeInfo::Annotation::Semantic> semantic) { absl::optional<GeneratedCodeInfo::Annotation::Semantic> semantic) {
for (std::vector<const GeneratedCodeInfo::Annotation*>::const_iterator for (std::vector<const GeneratedCodeInfo::Annotation*>::const_iterator
i = annotations.begin(), i = annotations.begin(),
@ -133,9 +133,9 @@ bool AtLeastOneAnnotationMatchesSubstring(
return false; return false;
} }
bool AnnotationMatchesSubstring(const std::string& file_content, bool AnnotationMatchesSubstring(absl::string_view file_content,
const GeneratedCodeInfo::Annotation* annotation, const GeneratedCodeInfo::Annotation* annotation,
const std::string& expected_text) { absl::string_view expected_text) {
std::vector<const GeneratedCodeInfo::Annotation*> annotations; std::vector<const GeneratedCodeInfo::Annotation*> annotations;
annotations.push_back(annotation); annotations.push_back(annotation);
return AtLeastOneAnnotationMatchesSubstring(file_content, annotations, return AtLeastOneAnnotationMatchesSubstring(file_content, annotations,

@ -31,8 +31,7 @@ struct ExpectedOutput {
std::string file_path; std::string file_path;
std::string file_content; std::string file_content;
GeneratedCodeInfo file_info; GeneratedCodeInfo file_info;
explicit ExpectedOutput(const std::string& file_path) explicit ExpectedOutput(absl::string_view file_path) : file_path(file_path) {}
: file_path(file_path) {}
}; };
// Creates a file with name `filename` and content `data` in temp test // Creates a file with name `filename` and content `data` in temp test
@ -54,13 +53,13 @@ bool RunProtoCompiler(const std::string& filename,
const std::string& plugin_specific_args, const std::string& plugin_specific_args,
CommandLineInterface* cli, FileDescriptorProto* file); CommandLineInterface* cli, FileDescriptorProto* file);
bool DecodeMetadata(const std::string& path, GeneratedCodeInfo* info); bool DecodeMetadata(absl::string_view path, GeneratedCodeInfo* info);
// Finds all of the Annotations for a given source file and path. // Finds all of the Annotations for a given source file and path.
// See Location.path in https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/descriptor.proto for // See Location.path in https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/descriptor.proto for
// explanation of what path vector is. // explanation of what path vector is.
void FindAnnotationsOnPath( void FindAnnotationsOnPath(
const GeneratedCodeInfo& info, const std::string& source_file, const GeneratedCodeInfo& info, absl::string_view source_file,
const std::vector<int>& path, const std::vector<int>& path,
std::vector<const GeneratedCodeInfo::Annotation*>* annotations); std::vector<const GeneratedCodeInfo::Annotation*>* annotations);
@ -70,23 +69,23 @@ void FindAnnotationsOnPath(
// https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/descriptor.proto for explanation of what path // https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/descriptor.proto for explanation of what path
// vector is. // vector is.
const GeneratedCodeInfo::Annotation* FindAnnotationOnPath( const GeneratedCodeInfo::Annotation* FindAnnotationOnPath(
const GeneratedCodeInfo& info, const std::string& source_file, const GeneratedCodeInfo& info, absl::string_view source_file,
const std::vector<int>& path); const std::vector<int>& path);
// Returns true if at least one of the provided annotations covers a given // Returns true if at least one of the provided annotations covers a given
// substring with the given semantic in file_content. // substring with the given semantic in file_content.
bool AtLeastOneAnnotationMatchesSubstring( bool AtLeastOneAnnotationMatchesSubstring(
const std::string& file_content, absl::string_view file_content,
const std::vector<const GeneratedCodeInfo::Annotation*>& annotations, const std::vector<const GeneratedCodeInfo::Annotation*>& annotations,
const std::string& expected_text, absl::string_view expected_text,
absl::optional<GeneratedCodeInfo::Annotation::Semantic> expected_semantic = absl::optional<GeneratedCodeInfo::Annotation::Semantic> expected_semantic =
absl::nullopt); absl::nullopt);
// Returns true if the provided annotation covers a given substring in // Returns true if the provided annotation covers a given substring in
// file_content. // file_content.
bool AnnotationMatchesSubstring(const std::string& file_content, bool AnnotationMatchesSubstring(absl::string_view file_content,
const GeneratedCodeInfo::Annotation* annotation, const GeneratedCodeInfo::Annotation* annotation,
const std::string& expected_text); absl::string_view expected_text);
// Returns the text spanned by the annotation if the span is valid; otherwise // Returns the text spanned by the annotation if the span is valid; otherwise
// returns nullopt. // returns nullopt.

@ -11,32 +11,17 @@
#include "google/protobuf/compiler/command_line_interface.h" #include "google/protobuf/compiler/command_line_interface.h"
#include <sys/types.h>
#include <algorithm>
#include <cstdint> #include <cstdint>
#include <cstdio>
#include <cstdlib> #include <cstdlib>
#include <cstring> #include <cstring>
#include <fstream>
#include <iostream>
#include <ostream>
#include "absl/algorithm/container.h"
#include "absl/base/attributes.h"
#include "absl/base/log_severity.h"
#include "absl/container/btree_map.h"
#include "absl/container/btree_set.h"
#include "absl/container/flat_hash_map.h"
#include "absl/log/globals.h"
#include "absl/status/status.h"
#include "absl/status/statusor.h"
#include "absl/strings/str_cat.h"
#include "absl/types/span.h"
#include "google/protobuf/compiler/versions.h"
#include "google/protobuf/descriptor_database.h"
#include "google/protobuf/descriptor_visitor.h"
#include "google/protobuf/feature_resolver.h"
#include "google/protobuf/io/zero_copy_stream_impl_lite.h"
#include "google/protobuf/stubs/platform_macros.h"
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#ifdef major #ifdef major
#undef major #undef major
#endif #endif
@ -50,9 +35,6 @@
#endif #endif
#include <errno.h> #include <errno.h>
#include <fstream>
#include <iostream>
#include <limits.h> // For PATH_MAX #include <limits.h> // For PATH_MAX
#include <memory> #include <memory>
@ -66,30 +48,49 @@
#include <sys/sysctl.h> #include <sys/sysctl.h>
#endif #endif
#include "absl/algorithm/container.h"
#include "absl/base/attributes.h"
#include "absl/base/log_severity.h"
#include "absl/container/btree_map.h"
#include "absl/container/btree_set.h"
#include "absl/container/flat_hash_map.h"
#include "absl/container/flat_hash_set.h"
#include "absl/log/absl_check.h" #include "absl/log/absl_check.h"
#include "absl/log/absl_log.h" #include "absl/log/absl_log.h"
#include "absl/container/flat_hash_set.h" #include "absl/log/globals.h"
#include "absl/status/status.h"
#include "absl/status/statusor.h"
#include "absl/strings/match.h" #include "absl/strings/match.h"
#include "absl/strings/str_cat.h"
#include "absl/strings/str_format.h" #include "absl/strings/str_format.h"
#include "absl/strings/str_replace.h" #include "absl/strings/str_replace.h"
#include "absl/strings/str_split.h" #include "absl/strings/str_split.h"
#include "absl/strings/string_view.h" #include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "absl/types/span.h"
#include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/code_generator.h"
#include "google/protobuf/compiler/importer.h" #include "google/protobuf/compiler/importer.h"
#include "google/protobuf/compiler/plugin.pb.h" #include "google/protobuf/compiler/plugin.pb.h"
#include "google/protobuf/compiler/retention.h" #include "google/protobuf/compiler/retention.h"
#include "google/protobuf/compiler/subprocess.h" #include "google/protobuf/compiler/subprocess.h"
#include "google/protobuf/compiler/versions.h"
#include "google/protobuf/compiler/zip_writer.h" #include "google/protobuf/compiler/zip_writer.h"
#include "google/protobuf/descriptor.h" #include "google/protobuf/descriptor.h"
#include "google/protobuf/descriptor.pb.h" #include "google/protobuf/descriptor.pb.h"
#include "google/protobuf/descriptor_database.h"
#include "google/protobuf/descriptor_visitor.h"
#include "google/protobuf/dynamic_message.h" #include "google/protobuf/dynamic_message.h"
#include "google/protobuf/feature_resolver.h"
#include "google/protobuf/io/coded_stream.h" #include "google/protobuf/io/coded_stream.h"
#include "google/protobuf/io/printer.h" #include "google/protobuf/io/printer.h"
#include "google/protobuf/io/zero_copy_stream_impl.h" #include "google/protobuf/io/zero_copy_stream_impl.h"
#include "google/protobuf/io/zero_copy_stream_impl_lite.h"
#include "google/protobuf/text_format.h" #include "google/protobuf/text_format.h"
#include "google/protobuf/stubs/platform_macros.h"
#ifdef _WIN32 #ifdef _WIN32
#include "google/protobuf/io/io_win32.h" #include "google/protobuf/io/io_win32.h"
#endif #endif
@ -131,7 +132,7 @@ static const char* kDefaultDirectDependenciesViolationMsg =
// Returns true if the text looks like a Windows-style absolute path, starting // Returns true if the text looks like a Windows-style absolute path, starting
// with a drive letter. Example: "C:\foo". TODO: Share this with // with a drive letter. Example: "C:\foo". TODO: Share this with
// copy in importer.cc? // copy in importer.cc?
static bool IsWindowsAbsolutePath(const std::string& text) { static bool IsWindowsAbsolutePath(absl::string_view text) {
#if defined(_WIN32) || defined(__CYGWIN__) #if defined(_WIN32) || defined(__CYGWIN__)
return text.size() >= 3 && text[1] == ':' && absl::ascii_isalpha(text[0]) && return text.size() >= 3 && text[1] == ':' && absl::ascii_isalpha(text[0]) &&
(text[2] == '/' || text[2] == '\\') && text.find_last_of(':') == 1; (text[2] == '/' || text[2] == '\\') && text.find_last_of(':') == 1;
@ -167,10 +168,10 @@ void AddTrailingSlash(std::string* path) {
} }
} }
bool VerifyDirectoryExists(const std::string& path) { bool VerifyDirectoryExists(absl::string_view path) {
if (path.empty()) return true; if (path.empty()) return true;
if (access(path.c_str(), F_OK) == -1) { if (access(std::string(path).c_str(), F_OK) == -1) {
std::cerr << path << ": " << strerror(errno) << std::endl; std::cerr << path << ": " << strerror(errno) << std::endl;
return false; return false;
} else { } else {
@ -182,13 +183,13 @@ bool VerifyDirectoryExists(const std::string& path) {
// parent if necessary, and so on. The full file name is actually // parent if necessary, and so on. The full file name is actually
// (prefix + filename), but we assume |prefix| already exists and only create // (prefix + filename), but we assume |prefix| already exists and only create
// directories listed in |filename|. // directories listed in |filename|.
bool TryCreateParentDirectory(const std::string& prefix, bool TryCreateParentDirectory(absl::string_view prefix,
const std::string& filename) { absl::string_view filename) {
// Recursively create parent directories to the output file. // Recursively create parent directories to the output file.
// On Windows, both '/' and '\' are valid path separators. // On Windows, both '/' and '\' are valid path separators.
std::vector<std::string> parts = std::vector<std::string> parts =
absl::StrSplit(filename, absl::ByAnyChar("/\\"), absl::SkipEmpty()); absl::StrSplit(filename, absl::ByAnyChar("/\\"), absl::SkipEmpty());
std::string path_so_far = prefix; std::string path_so_far = std::string(prefix);
for (size_t i = 0; i < parts.size() - 1; ++i) { for (size_t i = 0; i < parts.size() - 1; ++i) {
path_so_far += parts[i]; path_so_far += parts[i];
if (mkdir(path_so_far.c_str(), 0777) != 0) { if (mkdir(path_so_far.c_str(), 0777) != 0) {
@ -297,7 +298,7 @@ std::string PluginName(absl::string_view plugin_prefix,
directive.substr(2, directive.size() - 6)); directive.substr(2, directive.size() - 6));
} }
bool GetBootstrapParam(const std::string& parameter) { bool GetBootstrapParam(absl::string_view parameter) {
std::vector<std::string> parts = absl::StrSplit(parameter, ','); std::vector<std::string> parts = absl::StrSplit(parameter, ',');
for (const auto& part : parts) { for (const auto& part : parts) {
if (part == "bootstrap") { if (part == "bootstrap") {
@ -447,11 +448,11 @@ class CommandLineInterface::GeneratorContextImpl : public GeneratorContext {
// Write all files in the directory to disk at the given output location, // Write all files in the directory to disk at the given output location,
// which must end in a '/'. // which must end in a '/'.
bool WriteAllToDisk(const std::string& prefix); bool WriteAllToDisk(absl::string_view prefix);
// Write the contents of this directory to a ZIP-format archive with the // Write the contents of this directory to a ZIP-format archive with the
// given name. // given name.
bool WriteAllToZip(const std::string& filename); bool WriteAllToZip(absl::string_view filename);
// Add a boilerplate META-INF/MANIFEST.MF file as required by the Java JAR // Add a boilerplate META-INF/MANIFEST.MF file as required by the Java JAR
// format, unless one has already been written. // format, unless one has already been written.
@ -486,13 +487,13 @@ class CommandLineInterface::MemoryOutputStream
: public io::ZeroCopyOutputStream { : public io::ZeroCopyOutputStream {
public: public:
MemoryOutputStream(GeneratorContextImpl* directory, MemoryOutputStream(GeneratorContextImpl* directory,
const std::string& filename, bool append_mode); absl::string_view filename, bool append_mode);
MemoryOutputStream(GeneratorContextImpl* directory, MemoryOutputStream(GeneratorContextImpl* directory,
const std::string& filename, absl::string_view filename,
const std::string& insertion_point); absl::string_view insertion_point);
MemoryOutputStream(GeneratorContextImpl* directory, MemoryOutputStream(GeneratorContextImpl* directory,
const std::string& filename, absl::string_view filename,
const std::string& insertion_point, absl::string_view insertion_point,
const google::protobuf::GeneratedCodeInfo& info); const google::protobuf::GeneratedCodeInfo& info);
~MemoryOutputStream() override; ~MemoryOutputStream() override;
@ -511,14 +512,14 @@ class CommandLineInterface::MemoryOutputStream
// insertion_offset and indent_length. We assume that insertions will not // insertion_offset and indent_length. We assume that insertions will not
// occur within any given annotated span of text. insertion_content must end // occur within any given annotated span of text. insertion_content must end
// with an endline. // with an endline.
void UpdateMetadata(const std::string& insertion_content, void UpdateMetadata(absl::string_view insertion_content,
size_t insertion_offset, size_t insertion_length, size_t insertion_offset, size_t insertion_length,
size_t indent_length); size_t indent_length);
// Inserts info_to_insert_ into target_info, assuming that the relevant // Inserts info_to_insert_ into target_info, assuming that the relevant
// insertion was made at insertion_offset in file_content with the given // insertion was made at insertion_offset in file_content with the given
// indent_length. insertion_content must end with an endline. // indent_length. insertion_content must end with an endline.
void InsertShiftedInfo(const std::string& insertion_content, void InsertShiftedInfo(absl::string_view insertion_content,
size_t insertion_offset, size_t indent_length, size_t insertion_offset, size_t indent_length,
google::protobuf::GeneratedCodeInfo& target_info); google::protobuf::GeneratedCodeInfo& target_info);
@ -547,7 +548,7 @@ CommandLineInterface::GeneratorContextImpl::GeneratorContextImpl(
: parsed_files_(parsed_files), had_error_(false) {} : parsed_files_(parsed_files), had_error_(false) {}
bool CommandLineInterface::GeneratorContextImpl::WriteAllToDisk( bool CommandLineInterface::GeneratorContextImpl::WriteAllToDisk(
const std::string& prefix) { absl::string_view prefix) {
if (had_error_) { if (had_error_) {
return false; return false;
} }
@ -557,14 +558,14 @@ bool CommandLineInterface::GeneratorContextImpl::WriteAllToDisk(
} }
for (const auto& pair : files_) { for (const auto& pair : files_) {
const std::string& relative_filename = pair.first; absl::string_view relative_filename = pair.first;
const char* data = pair.second.data(); const char* data = pair.second.data();
int size = pair.second.size(); int size = pair.second.size();
if (!TryCreateParentDirectory(prefix, relative_filename)) { if (!TryCreateParentDirectory(prefix, relative_filename)) {
return false; return false;
} }
std::string filename = prefix + relative_filename; std::string filename = absl::StrCat(prefix, relative_filename);
// Create the output file. // Create the output file.
int file_descriptor; int file_descriptor;
@ -621,7 +622,7 @@ bool CommandLineInterface::GeneratorContextImpl::WriteAllToDisk(
} }
bool CommandLineInterface::GeneratorContextImpl::WriteAllToZip( bool CommandLineInterface::GeneratorContextImpl::WriteAllToZip(
const std::string& filename) { absl::string_view filename) {
if (had_error_) { if (had_error_) {
return false; return false;
} }
@ -629,8 +630,8 @@ bool CommandLineInterface::GeneratorContextImpl::WriteAllToZip(
// Create the output file. // Create the output file.
int file_descriptor; int file_descriptor;
do { do {
file_descriptor = file_descriptor = open(std::string(filename).c_str(),
open(filename.c_str(), O_WRONLY | O_CREAT | O_TRUNC | O_BINARY, 0666); O_WRONLY | O_CREAT | O_TRUNC | O_BINARY, 0666);
} while (file_descriptor < 0 && errno == EINTR); } while (file_descriptor < 0 && errno == EINTR);
if (file_descriptor < 0) { if (file_descriptor < 0) {
@ -706,7 +707,7 @@ CommandLineInterface::GeneratorContextImpl::OpenForInsertWithGeneratedCodeInfo(
// ------------------------------------------------------------------- // -------------------------------------------------------------------
CommandLineInterface::MemoryOutputStream::MemoryOutputStream( CommandLineInterface::MemoryOutputStream::MemoryOutputStream(
GeneratorContextImpl* directory, const std::string& filename, GeneratorContextImpl* directory, absl::string_view filename,
bool append_mode) bool append_mode)
: directory_(directory), : directory_(directory),
filename_(filename), filename_(filename),
@ -714,16 +715,16 @@ CommandLineInterface::MemoryOutputStream::MemoryOutputStream(
inner_(new io::StringOutputStream(&data_)) {} inner_(new io::StringOutputStream(&data_)) {}
CommandLineInterface::MemoryOutputStream::MemoryOutputStream( CommandLineInterface::MemoryOutputStream::MemoryOutputStream(
GeneratorContextImpl* directory, const std::string& filename, GeneratorContextImpl* directory, absl::string_view filename,
const std::string& insertion_point) absl::string_view insertion_point)
: directory_(directory), : directory_(directory),
filename_(filename), filename_(filename),
insertion_point_(insertion_point), insertion_point_(insertion_point),
inner_(new io::StringOutputStream(&data_)) {} inner_(new io::StringOutputStream(&data_)) {}
CommandLineInterface::MemoryOutputStream::MemoryOutputStream( CommandLineInterface::MemoryOutputStream::MemoryOutputStream(
GeneratorContextImpl* directory, const std::string& filename, GeneratorContextImpl* directory, absl::string_view filename,
const std::string& insertion_point, const google::protobuf::GeneratedCodeInfo& info) absl::string_view insertion_point, const google::protobuf::GeneratedCodeInfo& info)
: directory_(directory), : directory_(directory),
filename_(filename), filename_(filename),
insertion_point_(insertion_point), insertion_point_(insertion_point),
@ -731,7 +732,7 @@ CommandLineInterface::MemoryOutputStream::MemoryOutputStream(
info_to_insert_(info) {} info_to_insert_(info) {}
void CommandLineInterface::MemoryOutputStream::InsertShiftedInfo( void CommandLineInterface::MemoryOutputStream::InsertShiftedInfo(
const std::string& insertion_content, size_t insertion_offset, absl::string_view insertion_content, size_t insertion_offset,
size_t indent_length, google::protobuf::GeneratedCodeInfo& target_info) { size_t indent_length, google::protobuf::GeneratedCodeInfo& target_info) {
// Keep track of how much extra data was added for indents before the // Keep track of how much extra data was added for indents before the
// current annotation being inserted. `pos` and `source_annotation.begin()` // current annotation being inserted. `pos` and `source_annotation.begin()`
@ -767,7 +768,7 @@ void CommandLineInterface::MemoryOutputStream::InsertShiftedInfo(
} }
void CommandLineInterface::MemoryOutputStream::UpdateMetadata( void CommandLineInterface::MemoryOutputStream::UpdateMetadata(
const std::string& insertion_content, size_t insertion_offset, absl::string_view insertion_content, size_t insertion_offset,
size_t insertion_length, size_t indent_length) { size_t insertion_length, size_t indent_length) {
auto it = directory_->files_.find(absl::StrCat(filename_, ".pb.meta")); auto it = directory_->files_.find(absl::StrCat(filename_, ".pb.meta"));
if (it == directory_->files_.end() && info_to_insert_.annotation().empty()) { if (it == directory_->files_.end() && info_to_insert_.annotation().empty()) {
@ -963,30 +964,31 @@ CommandLineInterface::CommandLineInterface()
CommandLineInterface::~CommandLineInterface() = default; CommandLineInterface::~CommandLineInterface() = default;
void CommandLineInterface::RegisterGenerator(const std::string& flag_name, void CommandLineInterface::RegisterGenerator(absl::string_view flag_name,
CodeGenerator* generator, CodeGenerator* generator,
const std::string& help_text) { absl::string_view help_text) {
GeneratorInfo info; GeneratorInfo info;
info.flag_name = flag_name; info.flag_name = std::string(flag_name);
info.generator = generator; info.generator = generator;
info.help_text = help_text; info.help_text = std::string(help_text);
generators_by_flag_name_[flag_name] = info; generators_by_flag_name_[flag_name] = info;
} }
void CommandLineInterface::RegisterGenerator( void CommandLineInterface::RegisterGenerator(absl::string_view flag_name,
const std::string& flag_name, const std::string& option_flag_name, absl::string_view option_flag_name,
CodeGenerator* generator, const std::string& help_text) { CodeGenerator* generator,
absl::string_view help_text) {
GeneratorInfo info; GeneratorInfo info;
info.flag_name = flag_name; info.flag_name = std::string(flag_name);
info.option_flag_name = option_flag_name; info.option_flag_name = std::string(option_flag_name);
info.generator = generator; info.generator = generator;
info.help_text = help_text; info.help_text = std::string(help_text);
generators_by_flag_name_[flag_name] = info; generators_by_flag_name_[flag_name] = info;
generators_by_option_name_[option_flag_name] = info; generators_by_option_name_[option_flag_name] = info;
} }
void CommandLineInterface::AllowPlugins(const std::string& exe_name_prefix) { void CommandLineInterface::AllowPlugins(absl::string_view exe_name_prefix) {
plugin_prefix_ = exe_name_prefix; plugin_prefix_ = std::string(exe_name_prefix);
} }
namespace { namespace {
@ -1029,7 +1031,7 @@ bool HasReservedFieldNumber(const FieldDescriptor* field) {
namespace { namespace {
std::unique_ptr<SimpleDescriptorDatabase> std::unique_ptr<SimpleDescriptorDatabase>
PopulateSingleSimpleDescriptorDatabase(const std::string& descriptor_set_name); PopulateSingleSimpleDescriptorDatabase(absl::string_view descriptor_set_name);
// Indicates whether the field is compatible with the given target type. // Indicates whether the field is compatible with the given target type.
bool IsFieldCompatible(const FieldDescriptor& field, bool IsFieldCompatible(const FieldDescriptor& field,
@ -1204,7 +1206,7 @@ int CommandLineInterface::Run(int argc, const char* const argv[]) {
// Any --descriptor_set_in FileDescriptorSet objects will be used as a // Any --descriptor_set_in FileDescriptorSet objects will be used as a
// fallback to input_files on command line, so create that db first. // fallback to input_files on command line, so create that db first.
if (!descriptor_set_in_names_.empty()) { if (!descriptor_set_in_names_.empty()) {
for (const std::string& name : descriptor_set_in_names_) { for (absl::string_view name : descriptor_set_in_names_) {
std::unique_ptr<SimpleDescriptorDatabase> database_for_descriptor_set = std::unique_ptr<SimpleDescriptorDatabase> database_for_descriptor_set =
PopulateSingleSimpleDescriptorDatabase(name); PopulateSingleSimpleDescriptorDatabase(name);
if (!database_for_descriptor_set) { if (!database_for_descriptor_set) {
@ -1356,7 +1358,7 @@ int CommandLineInterface::Run(int argc, const char* const argv[]) {
} }
for (const auto& pair : output_directories) { for (const auto& pair : output_directories) {
const std::string& location = pair.first; absl::string_view location = pair.first;
GeneratorContextImpl* directory = pair.second.get(); GeneratorContextImpl* directory = pair.second.get();
if (absl::EndsWith(location, "/")) { if (absl::EndsWith(location, "/")) {
if (!directory->WriteAllToDisk(location)) { if (!directory->WriteAllToDisk(location)) {
@ -1458,10 +1460,10 @@ bool CommandLineInterface::InitializeDiskSourceTree(
namespace { namespace {
std::unique_ptr<SimpleDescriptorDatabase> std::unique_ptr<SimpleDescriptorDatabase>
PopulateSingleSimpleDescriptorDatabase(const std::string& descriptor_set_name) { PopulateSingleSimpleDescriptorDatabase(absl::string_view descriptor_set_name) {
int fd; int fd;
do { do {
fd = open(descriptor_set_name.c_str(), O_RDONLY | O_BINARY); fd = open(std::string(descriptor_set_name).c_str(), O_RDONLY | O_BINARY);
} while (fd < 0 && errno == EINTR); } while (fd < 0 && errno == EINTR);
if (fd < 0) { if (fd < 0) {
std::cerr << descriptor_set_name << ": " << strerror(ENOENT) << std::endl; std::cerr << descriptor_set_name << ": " << strerror(ENOENT) << std::endl;
@ -2026,8 +2028,8 @@ bool CommandLineInterface::ParseArgument(const char* arg, std::string* name,
} }
CommandLineInterface::ParseArgumentStatus CommandLineInterface::ParseArgumentStatus
CommandLineInterface::InterpretArgument(const std::string& name, CommandLineInterface::InterpretArgument(absl::string_view name,
const std::string& value) { absl::string_view value) {
if (name.empty()) { if (name.empty()) {
// Not a flag. Just a filename. // Not a flag. Just a filename.
if (value.empty()) { if (value.empty()) {
@ -2046,8 +2048,8 @@ CommandLineInterface::InterpretArgument(const std::string& name,
// On Windows, the shell (typically cmd.exe) does not expand wildcards in // On Windows, the shell (typically cmd.exe) does not expand wildcards in
// file names (e.g. foo\*.proto), so we do it ourselves. // file names (e.g. foo\*.proto), so we do it ourselves.
switch (google::protobuf::io::win32::ExpandWildcards( switch (google::protobuf::io::win32::ExpandWildcards(
value, [this](const std::string& path) { std::string(value), [this](absl::string_view path) {
this->input_files_.push_back(path); this->input_files_.push_back(std::string(path));
})) { })) {
case google::protobuf::io::win32::ExpandWildcardsResult::kSuccess: case google::protobuf::io::win32::ExpandWildcardsResult::kSuccess:
break; break;
@ -2065,7 +2067,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
#else // not _WIN32 #else // not _WIN32
// On other platforms than Windows (e.g. Linux, Mac OS) the shell (typically // On other platforms than Windows (e.g. Linux, Mac OS) the shell (typically
// Bash) expands wildcards. // Bash) expands wildcards.
input_files_.push_back(value); input_files_.push_back(std::string(value));
#endif // _WIN32 #endif // _WIN32
} else if (name == "-I" || name == "--proto_path") { } else if (name == "-I" || name == "--proto_path") {
@ -2133,7 +2135,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
direct_dependencies_.insert(direct.begin(), direct.end()); direct_dependencies_.insert(direct.begin(), direct.end());
} else if (name == "--direct_dependencies_violation_msg") { } else if (name == "--direct_dependencies_violation_msg") {
direct_dependencies_violation_msg_ = value; direct_dependencies_violation_msg_ = std::string(value);
} else if (name == "--descriptor_set_in") { } else if (name == "--descriptor_set_in") {
if (!descriptor_set_in_names_.empty()) { if (!descriptor_set_in_names_.empty()) {
@ -2174,7 +2176,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
<< std::endl; << std::endl;
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
} }
descriptor_set_out_name_ = value; descriptor_set_out_name_ = std::string(value);
} else if (name == "--dependency_out") { } else if (name == "--dependency_out") {
if (!dependency_out_name_.empty()) { if (!dependency_out_name_.empty()) {
@ -2190,7 +2192,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
<< std::endl; << std::endl;
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
} }
dependency_out_name_ = value; dependency_out_name_ = std::string(value);
} else if (name == "--include_imports") { } else if (name == "--include_imports") {
if (imports_in_descriptor_set_) { if (imports_in_descriptor_set_) {
@ -2261,7 +2263,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
} }
codec_type_ = value; codec_type_ = std::string(value);
} else if (name == "--deterministic_output") { } else if (name == "--deterministic_output") {
deterministic_output_ = true; deterministic_output_ = true;
@ -2288,14 +2290,14 @@ CommandLineInterface::InterpretArgument(const std::string& name,
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
} }
std::string plugin_name; absl::string_view plugin_name;
std::string path; absl::string_view path;
std::string::size_type equals_pos = value.find_first_of('='); auto equals_pos = value.find_first_of('=');
if (equals_pos == std::string::npos) { if (equals_pos == value.npos) {
// Use the basename of the file. // Use the basename of the file.
std::string::size_type slash_pos = value.find_last_of('/'); auto slash_pos = value.find_last_of('/');
if (slash_pos == std::string::npos) { if (slash_pos == value.npos) {
plugin_name = value; plugin_name = value;
} else { } else {
plugin_name = value.substr(slash_pos + 1); plugin_name = value.substr(slash_pos + 1);
@ -2306,7 +2308,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
path = value.substr(equals_pos + 1); path = value.substr(equals_pos + 1);
} }
plugins_[plugin_name] = path; plugins_[plugin_name] = std::string(path);
} else if (name == "--print_free_field_numbers") { } else if (name == "--print_free_field_numbers") {
if (mode_ != MODE_COMPILE) { if (mode_ != MODE_COMPILE) {
@ -2362,7 +2364,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
<< std::endl; << std::endl;
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
} }
edition_defaults_out_name_ = value; edition_defaults_out_name_ = std::string(value);
} else if (name == "--edition_defaults_minimum") { } else if (name == "--edition_defaults_minimum") {
if (edition_defaults_minimum_ != EDITION_UNKNOWN) { if (edition_defaults_minimum_ != EDITION_UNKNOWN) {
std::cerr << name << " may only be passed once." << std::endl; std::cerr << name << " may only be passed once." << std::endl;
@ -2396,14 +2398,14 @@ CommandLineInterface::InterpretArgument(const std::string& name,
if (!parameters->empty()) { if (!parameters->empty()) {
parameters->append(","); parameters->append(",");
} }
parameters->append(value); parameters->append(std::string(value));
} else if (absl::StartsWith(name, "--") && absl::EndsWith(name, "_opt")) { } else if (absl::StartsWith(name, "--") && absl::EndsWith(name, "_opt")) {
std::string* parameters = std::string* parameters =
&plugin_parameters_[PluginName(plugin_prefix_, name)]; &plugin_parameters_[PluginName(plugin_prefix_, name)];
if (!parameters->empty()) { if (!parameters->empty()) {
parameters->append(","); parameters->append(",");
} }
parameters->append(value); parameters->append(std::string(value));
} else { } else {
std::cerr << "Unknown flag: " << name << std::endl; std::cerr << "Unknown flag: " << name << std::endl;
return PARSE_ARGUMENT_FAIL; return PARSE_ARGUMENT_FAIL;
@ -2418,7 +2420,7 @@ CommandLineInterface::InterpretArgument(const std::string& name,
} }
OutputDirective directive; OutputDirective directive;
directive.name = name; directive.name = std::string(name);
if (generator_info == nullptr) { if (generator_info == nullptr) {
directive.generator = nullptr; directive.generator = nullptr;
} else { } else {
@ -2428,12 +2430,12 @@ CommandLineInterface::InterpretArgument(const std::string& name,
// Split value at ':' to separate the generator parameter from the // Split value at ':' to separate the generator parameter from the
// filename. However, avoid doing this if the colon is part of a valid // filename. However, avoid doing this if the colon is part of a valid
// Windows-style absolute path. // Windows-style absolute path.
std::string::size_type colon_pos = value.find_first_of(':'); auto colon_pos = value.find_first_of(':');
if (colon_pos == std::string::npos || IsWindowsAbsolutePath(value)) { if (colon_pos == value.npos || IsWindowsAbsolutePath(value)) {
directive.output_location = value; directive.output_location = std::string(value);
} else { } else {
directive.parameter = value.substr(0, colon_pos); directive.parameter = std::string(value.substr(0, colon_pos));
directive.output_location = value.substr(colon_pos + 1); directive.output_location = std::string(value.substr(colon_pos + 1));
} }
output_directives_.push_back(directive); output_directives_.push_back(directive);
@ -2554,7 +2556,7 @@ Parse PROTO_FILES and generate output based on the options given:
} }
bool CommandLineInterface::EnforceProto3OptionalSupport( bool CommandLineInterface::EnforceProto3OptionalSupport(
const std::string& codegen_name, uint64_t supported_features, absl::string_view codegen_name, uint64_t supported_features,
const std::vector<const FileDescriptor*>& parsed_files) const { const std::vector<const FileDescriptor*>& parsed_files) const {
bool supports_proto3_optional = bool supports_proto3_optional =
supported_features & CodeGenerator::FEATURE_PROTO3_OPTIONAL; supported_features & CodeGenerator::FEATURE_PROTO3_OPTIONAL;
@ -2578,7 +2580,7 @@ bool CommandLineInterface::EnforceProto3OptionalSupport(
} }
bool CommandLineInterface::EnforceEditionsSupport( bool CommandLineInterface::EnforceEditionsSupport(
const std::string& codegen_name, uint64_t supported_features, absl::string_view codegen_name, uint64_t supported_features,
Edition minimum_edition, Edition maximum_edition, Edition minimum_edition, Edition maximum_edition,
const std::vector<const FileDescriptor*>& parsed_files) const { const std::vector<const FileDescriptor*>& parsed_files) const {
if (experimental_editions_) { if (experimental_editions_) {
@ -2694,12 +2696,13 @@ bool CommandLineInterface::GenerateDependencyManifestFile(
std::vector<std::string> output_filenames; std::vector<std::string> output_filenames;
for (const auto& pair : output_directories) { for (const auto& pair : output_directories) {
const std::string& location = pair.first; absl::string_view location = pair.first;
GeneratorContextImpl* directory = pair.second.get(); GeneratorContextImpl* directory = pair.second.get();
std::vector<std::string> relative_output_filenames; std::vector<std::string> relative_output_filenames;
directory->GetOutputFilenames(&relative_output_filenames); directory->GetOutputFilenames(&relative_output_filenames);
for (size_t i = 0; i < relative_output_filenames.size(); ++i) { for (size_t i = 0; i < relative_output_filenames.size(); ++i) {
std::string output_filename = location + relative_output_filenames[i]; std::string output_filename =
absl::StrCat(location, relative_output_filenames[i]);
if (output_filename.compare(0, 2, "./") == 0) { if (output_filename.compare(0, 2, "./") == 0) {
output_filename = output_filename.substr(2); output_filename = output_filename.substr(2);
} }
@ -2744,7 +2747,7 @@ bool CommandLineInterface::GenerateDependencyManifestFile(
for (int i = 0; i < file_set.file_size(); ++i) { for (int i = 0; i < file_set.file_size(); ++i) {
const FileDescriptorProto& file = file_set.file(i); const FileDescriptorProto& file = file_set.file(i);
const std::string& virtual_file = file.name(); absl::string_view virtual_file = file.name();
std::string disk_file; std::string disk_file;
if (source_tree && if (source_tree &&
source_tree->VirtualFileToDiskFile(virtual_file, &disk_file)) { source_tree->VirtualFileToDiskFile(virtual_file, &disk_file)) {
@ -2763,11 +2766,11 @@ bool CommandLineInterface::GenerateDependencyManifestFile(
bool CommandLineInterface::GeneratePluginOutput( bool CommandLineInterface::GeneratePluginOutput(
const std::vector<const FileDescriptor*>& parsed_files, const std::vector<const FileDescriptor*>& parsed_files,
const std::string& plugin_name, const std::string& parameter, absl::string_view plugin_name, absl::string_view parameter,
GeneratorContext* generator_context, std::string* error) { GeneratorContext* generator_context, std::string* error) {
CodeGeneratorRequest request; CodeGeneratorRequest request;
CodeGeneratorResponse response; CodeGeneratorResponse response;
std::string processed_parameter = parameter; std::string processed_parameter = std::string(parameter);
bool bootstrap = GetBootstrapParam(processed_parameter); bool bootstrap = GetBootstrapParam(processed_parameter);
@ -3095,14 +3098,14 @@ bool CommandLineInterface::WriteEditionDefaults(const DescriptorPool& pool) {
} }
const CommandLineInterface::GeneratorInfo* const CommandLineInterface::GeneratorInfo*
CommandLineInterface::FindGeneratorByFlag(const std::string& name) const { CommandLineInterface::FindGeneratorByFlag(absl::string_view name) const {
auto it = generators_by_flag_name_.find(name); auto it = generators_by_flag_name_.find(name);
if (it == generators_by_flag_name_.end()) return nullptr; if (it == generators_by_flag_name_.end()) return nullptr;
return &it->second; return &it->second;
} }
const CommandLineInterface::GeneratorInfo* const CommandLineInterface::GeneratorInfo*
CommandLineInterface::FindGeneratorByOption(const std::string& option) const { CommandLineInterface::FindGeneratorByOption(absl::string_view option) const {
auto it = generators_by_option_name_.find(option); auto it = generators_by_option_name_.find(option);
if (it == generators_by_option_name_.end()) return nullptr; if (it == generators_by_option_name_.end()) return nullptr;
return &it->second; return &it->second;
@ -3168,10 +3171,10 @@ void GatherOccupiedFieldRanges(
// Utility function for PrintFreeFieldNumbers. // Utility function for PrintFreeFieldNumbers.
// Actually prints the formatted free field numbers for given message name and // Actually prints the formatted free field numbers for given message name and
// occupied ranges. // occupied ranges.
void FormatFreeFieldNumbers(const std::string& name, void FormatFreeFieldNumbers(absl::string_view name,
const absl::btree_set<FieldRange>& ranges) { const absl::btree_set<FieldRange>& ranges) {
std::string output; std::string output;
absl::StrAppendFormat(&output, "%-35s free:", name.c_str()); absl::StrAppendFormat(&output, "%-35s free:", name);
int next_free_number = 1; int next_free_number = 1;
for (const auto& range : ranges) { for (const auto& range : ranges) {
// This happens when groups re-use parent field numbers, in which // This happens when groups re-use parent field numbers, in which
@ -3213,3 +3216,5 @@ void CommandLineInterface::PrintFreeFieldNumbers(const Descriptor* descriptor) {
} // namespace compiler } // namespace compiler
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#include "google/protobuf/port_undef.inc"

@ -120,8 +120,8 @@ class PROTOC_EXPORT CommandLineInterface {
// protoc --foo_out=enable_bar:outdir // protoc --foo_out=enable_bar:outdir
// The text before the colon is passed to CodeGenerator::Generate() as the // The text before the colon is passed to CodeGenerator::Generate() as the
// "parameter". // "parameter".
void RegisterGenerator(const std::string& flag_name, CodeGenerator* generator, void RegisterGenerator(absl::string_view flag_name, CodeGenerator* generator,
const std::string& help_text); absl::string_view help_text);
// Register a code generator for a language. // Register a code generator for a language.
// Besides flag_name you can specify another option_flag_name that could be // Besides flag_name you can specify another option_flag_name that could be
@ -131,10 +131,9 @@ class PROTOC_EXPORT CommandLineInterface {
// Then you could invoke the compiler with a command like: // Then you could invoke the compiler with a command like:
// protoc --foo_out=enable_bar:outdir --foo_opt=enable_baz // protoc --foo_out=enable_bar:outdir --foo_opt=enable_baz
// This will pass "enable_bar,enable_baz" as the parameter to the generator. // This will pass "enable_bar,enable_baz" as the parameter to the generator.
void RegisterGenerator(const std::string& flag_name, void RegisterGenerator(absl::string_view flag_name,
const std::string& option_flag_name, absl::string_view option_flag_name,
CodeGenerator* generator, CodeGenerator* generator, absl::string_view help_text);
const std::string& help_text);
// Enables "plugins". In this mode, if a command-line flag ends with "_out" // Enables "plugins". In this mode, if a command-line flag ends with "_out"
// but does not match any registered generator, the compiler will attempt to // but does not match any registered generator, the compiler will attempt to
@ -172,7 +171,7 @@ class PROTOC_EXPORT CommandLineInterface {
// protoc --plug_out=enable_bar:outdir --plug_opt=enable_baz // protoc --plug_out=enable_bar:outdir --plug_opt=enable_baz
// This will pass "enable_bar,enable_baz" as the parameter to the plugin. // This will pass "enable_bar,enable_baz" as the parameter to the plugin.
// //
void AllowPlugins(const std::string& exe_name_prefix); void AllowPlugins(absl::string_view exe_name_prefix);
// Run the Protocol Compiler with the given command-line parameters. // Run the Protocol Compiler with the given command-line parameters.
// Returns the error code which should be returned by main(). // Returns the error code which should be returned by main().
@ -190,7 +189,9 @@ class PROTOC_EXPORT CommandLineInterface {
// Provides some text which will be printed when the --version flag is // Provides some text which will be printed when the --version flag is
// used. The version of libprotoc will also be printed on the next line // used. The version of libprotoc will also be printed on the next line
// after this text. // after this text.
void SetVersionInfo(const std::string& text) { version_info_ = text; } void SetVersionInfo(absl::string_view text) {
version_info_ = std::string(text);
}
// Configure protoc to act as if we're in opensource. // Configure protoc to act as if we're in opensource.
@ -224,11 +225,11 @@ class PROTOC_EXPORT CommandLineInterface {
// Fails if these files use proto3 optional and the code generator doesn't // Fails if these files use proto3 optional and the code generator doesn't
// support it. This is a permanent check. // support it. This is a permanent check.
bool EnforceProto3OptionalSupport( bool EnforceProto3OptionalSupport(
const std::string& codegen_name, uint64_t supported_features, absl::string_view codegen_name, uint64_t supported_features,
const std::vector<const FileDescriptor*>& parsed_files) const; const std::vector<const FileDescriptor*>& parsed_files) const;
bool EnforceEditionsSupport( bool EnforceEditionsSupport(
const std::string& codegen_name, uint64_t supported_features, absl::string_view codegen_name, uint64_t supported_features,
Edition minimum_edition, Edition maximum_edition, Edition minimum_edition, Edition maximum_edition,
const std::vector<const FileDescriptor*>& parsed_files) const; const std::vector<const FileDescriptor*>& parsed_files) const;
@ -262,8 +263,8 @@ class PROTOC_EXPORT CommandLineInterface {
bool ParseArgument(const char* arg, std::string* name, std::string* value); bool ParseArgument(const char* arg, std::string* name, std::string* value);
// Interprets arguments parsed with ParseArgument. // Interprets arguments parsed with ParseArgument.
ParseArgumentStatus InterpretArgument(const std::string& name, ParseArgumentStatus InterpretArgument(absl::string_view name,
const std::string& value); absl::string_view value);
// Print the --help text to stderr. // Print the --help text to stderr.
void PrintHelpText(); void PrintHelpText();
@ -289,7 +290,7 @@ class PROTOC_EXPORT CommandLineInterface {
GeneratorContext* generator_context); GeneratorContext* generator_context);
bool GeneratePluginOutput( bool GeneratePluginOutput(
const std::vector<const FileDescriptor*>& parsed_files, const std::vector<const FileDescriptor*>& parsed_files,
const std::string& plugin_name, const std::string& parameter, absl::string_view plugin_name, absl::string_view parameter,
GeneratorContext* generator_context, std::string* error); GeneratorContext* generator_context, std::string* error);
// Implements --encode and --decode. // Implements --encode and --decode.
@ -356,8 +357,8 @@ class PROTOC_EXPORT CommandLineInterface {
std::string help_text; std::string help_text;
}; };
const GeneratorInfo* FindGeneratorByFlag(const std::string& name) const; const GeneratorInfo* FindGeneratorByFlag(absl::string_view name) const;
const GeneratorInfo* FindGeneratorByOption(const std::string& option) const; const GeneratorInfo* FindGeneratorByOption(absl::string_view option) const;
absl::btree_map<std::string, GeneratorInfo> generators_by_flag_name_; absl::btree_map<std::string, GeneratorInfo> generators_by_flag_name_;
absl::flat_hash_map<std::string, GeneratorInfo> generators_by_option_name_; absl::flat_hash_map<std::string, GeneratorInfo> generators_by_option_name_;

@ -140,9 +140,10 @@ void CommandLineInterfaceTester::ExpectWarningSubstring(
#if defined(_WIN32) && !defined(__CYGWIN__) #if defined(_WIN32) && !defined(__CYGWIN__)
bool CommandLineInterfaceTester::HasAlternateErrorSubstring( bool CommandLineInterfaceTester::HasAlternateErrorSubstring(
const std::string& expected_substring) { absl::string_view expected_substring) {
EXPECT_NE(0, return_code_); EXPECT_NE(0, return_code_);
return error_text_.find(expected_substring) != std::string::npos; return absl::string_view(error_text_).find(expected_substring) !=
absl::string_view::npos;
} }
#endif // _WIN32 && !__CYGWIN__ #endif // _WIN32 && !__CYGWIN__

@ -51,19 +51,19 @@ class CommandLineInterfaceTester : public testing::Test {
// Returns the temporary directory created for testing. // Returns the temporary directory created for testing.
std::string temp_directory() { return temp_directory_; } std::string temp_directory() { return temp_directory_; }
void AllowPlugins(const std::string& prefix) { cli_.AllowPlugins(prefix); } void AllowPlugins(absl::string_view prefix) { cli_.AllowPlugins(prefix); }
void RegisterGenerator(const std::string& flag_name, void RegisterGenerator(absl::string_view flag_name,
std::unique_ptr<CodeGenerator> generator, std::unique_ptr<CodeGenerator> generator,
const std::string& help_text) { absl::string_view help_text) {
generators_.emplace_back(std::move(generator)); generators_.emplace_back(std::move(generator));
cli_.RegisterGenerator(flag_name, generators_.back().get(), help_text); cli_.RegisterGenerator(flag_name, generators_.back().get(), help_text);
} }
void RegisterGenerator(const std::string& flag_name, void RegisterGenerator(absl::string_view flag_name,
const std::string& option_flag_name, absl::string_view option_flag_name,
std::unique_ptr<CodeGenerator> generator, std::unique_ptr<CodeGenerator> generator,
const std::string& help_text) { absl::string_view help_text) {
generators_.emplace_back(std::move(generator)); generators_.emplace_back(std::move(generator));
cli_.RegisterGenerator(flag_name, option_flag_name, cli_.RegisterGenerator(flag_name, option_flag_name,
generators_.back().get(), help_text); generators_.back().get(), help_text);
@ -119,7 +119,7 @@ class CommandLineInterfaceTester : public testing::Test {
#if defined(_WIN32) && !defined(__CYGWIN__) #if defined(_WIN32) && !defined(__CYGWIN__)
// Returns true if ExpectErrorSubstring(expected_substring) would pass, but // Returns true if ExpectErrorSubstring(expected_substring) would pass, but
// does not fail otherwise. // does not fail otherwise.
bool HasAlternateErrorSubstring(const std::string& expected_substring); bool HasAlternateErrorSubstring(absl::string_view expected_substring);
#endif // _WIN32 && !__CYGWIN__ #endif // _WIN32 && !__CYGWIN__
void ExpectFileContent(absl::string_view filename, absl::string_view content); void ExpectFileContent(absl::string_view filename, absl::string_view content);

@ -13,21 +13,12 @@
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/types.h> #include <sys/types.h>
#include <cstddef>
#include <cstdint>
#include <gmock/gmock.h>
#include "absl/log/absl_check.h"
#include "absl/strings/escaping.h"
#include "absl/strings/str_cat.h"
#include "absl/types/span.h"
#include "google/protobuf/compiler/command_line_interface_tester.h"
#include "google/protobuf/unittest_features.pb.h"
#include "google/protobuf/unittest_invalid_features.pb.h"
#ifndef _MSC_VER #ifndef _MSC_VER
#include <unistd.h> #include <unistd.h>
#endif #endif
#include <cstddef>
#include <cstdint>
#include <memory> #include <memory>
#include <string> #include <string>
#include <utility> #include <utility>
@ -37,13 +28,19 @@
#include "google/protobuf/testing/file.h" #include "google/protobuf/testing/file.h"
#include "google/protobuf/any.pb.h" #include "google/protobuf/any.pb.h"
#include "google/protobuf/descriptor.pb.h" #include "google/protobuf/descriptor.pb.h"
#include <gmock/gmock.h>
#include "google/protobuf/testing/googletest.h" #include "google/protobuf/testing/googletest.h"
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include "absl/log/absl_check.h"
#include "absl/strings/escaping.h"
#include "absl/strings/str_cat.h"
#include "absl/strings/str_split.h" #include "absl/strings/str_split.h"
#include "absl/strings/string_view.h" #include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "absl/types/span.h"
#include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/code_generator.h"
#include "google/protobuf/compiler/command_line_interface.h" #include "google/protobuf/compiler/command_line_interface.h"
#include "google/protobuf/compiler/command_line_interface_tester.h"
#include "google/protobuf/compiler/cpp/names.h" #include "google/protobuf/compiler/cpp/names.h"
#include "google/protobuf/compiler/mock_code_generator.h" #include "google/protobuf/compiler/mock_code_generator.h"
#include "google/protobuf/compiler/plugin.pb.h" #include "google/protobuf/compiler/plugin.pb.h"
@ -51,6 +48,9 @@
#include "google/protobuf/test_util2.h" #include "google/protobuf/test_util2.h"
#include "google/protobuf/unittest.pb.h" #include "google/protobuf/unittest.pb.h"
#include "google/protobuf/unittest_custom_options.pb.h" #include "google/protobuf/unittest_custom_options.pb.h"
#include "google/protobuf/unittest_features.pb.h"
#include "google/protobuf/unittest_invalid_features.pb.h"
#ifdef GOOGLE_PROTOBUF_USE_BAZEL_GENERATED_PLUGIN_PATHS #ifdef GOOGLE_PROTOBUF_USE_BAZEL_GENERATED_PLUGIN_PATHS
// This is needed because of https://github.com/bazelbuild/bazel/issues/19124. // This is needed because of https://github.com/bazelbuild/bazel/issues/19124.
@ -158,29 +158,29 @@ class CommandLineInterfaceTest : public CommandLineInterfaceTester {
// generate given these inputs. message_name is the name of the first // generate given these inputs. message_name is the name of the first
// message that appeared in the proto file; this is just to make extra // message that appeared in the proto file; this is just to make extra
// sure that the correct file was parsed. // sure that the correct file was parsed.
void ExpectGenerated(const std::string& generator_name, void ExpectGenerated(absl::string_view generator_name,
const std::string& parameter, absl::string_view parameter,
const std::string& proto_name, absl::string_view proto_name,
const std::string& message_name); absl::string_view message_name);
void ExpectGenerated(const std::string& generator_name, void ExpectGenerated(absl::string_view generator_name,
const std::string& parameter, absl::string_view parameter,
const std::string& proto_name, absl::string_view proto_name,
const std::string& message_name, absl::string_view message_name,
const std::string& output_directory); absl::string_view output_directory);
void ExpectGeneratedWithMultipleInputs(const std::string& generator_name, void ExpectGeneratedWithMultipleInputs(absl::string_view generator_name,
const std::string& all_proto_names, absl::string_view all_proto_names,
const std::string& proto_name, absl::string_view proto_name,
const std::string& message_name); absl::string_view message_name);
void ExpectGeneratedWithInsertions(const std::string& generator_name, void ExpectGeneratedWithInsertions(absl::string_view generator_name,
const std::string& parameter, absl::string_view parameter,
const std::string& insertions, absl::string_view insertions,
const std::string& proto_name, absl::string_view proto_name,
const std::string& message_name); absl::string_view message_name);
void CheckGeneratedAnnotations(const std::string& name, void CheckGeneratedAnnotations(absl::string_view name,
const std::string& file); absl::string_view file);
#if defined(_WIN32) #if defined(_WIN32)
void ExpectNullCodeGeneratorCalled(const std::string& parameter); void ExpectNullCodeGeneratorCalled(absl::string_view parameter);
#endif // _WIN32 #endif // _WIN32
@ -195,8 +195,8 @@ class CommandLineInterfaceTest : public CommandLineInterfaceTester {
// The default code generators support all features. Use this to create a // The default code generators support all features. Use this to create a
// code generator that omits the given feature(s). // code generator that omits the given feature(s).
void CreateGeneratorWithMissingFeatures(const std::string& name, void CreateGeneratorWithMissingFeatures(absl::string_view name,
const std::string& description, absl::string_view description,
uint64_t features) { uint64_t features) {
auto generator = std::make_unique<MockCodeGenerator>(name); auto generator = std::make_unique<MockCodeGenerator>(name);
generator->SuppressFeatures(features); generator->SuppressFeatures(features);
@ -286,48 +286,49 @@ void CommandLineInterfaceTest::RunWithArgs(std::vector<std::string> args) {
// ------------------------------------------------------------------- // -------------------------------------------------------------------
void CommandLineInterfaceTest::ExpectGenerated( void CommandLineInterfaceTest::ExpectGenerated(absl::string_view generator_name,
const std::string& generator_name, const std::string& parameter, absl::string_view parameter,
const std::string& proto_name, const std::string& message_name) { absl::string_view proto_name,
absl::string_view message_name) {
MockCodeGenerator::ExpectGenerated(generator_name, parameter, "", proto_name, MockCodeGenerator::ExpectGenerated(generator_name, parameter, "", proto_name,
message_name, proto_name, message_name, proto_name,
temp_directory()); temp_directory());
} }
void CommandLineInterfaceTest::ExpectGenerated( void CommandLineInterfaceTest::ExpectGenerated(
const std::string& generator_name, const std::string& parameter, absl::string_view generator_name, absl::string_view parameter,
const std::string& proto_name, const std::string& message_name, absl::string_view proto_name, absl::string_view message_name,
const std::string& output_directory) { absl::string_view output_directory) {
MockCodeGenerator::ExpectGenerated( MockCodeGenerator::ExpectGenerated(
generator_name, parameter, "", proto_name, message_name, proto_name, generator_name, parameter, "", proto_name, message_name, proto_name,
absl::StrCat(temp_directory(), "/", output_directory)); absl::StrCat(temp_directory(), "/", output_directory));
} }
void CommandLineInterfaceTest::ExpectGeneratedWithMultipleInputs( void CommandLineInterfaceTest::ExpectGeneratedWithMultipleInputs(
const std::string& generator_name, const std::string& all_proto_names, absl::string_view generator_name, absl::string_view all_proto_names,
const std::string& proto_name, const std::string& message_name) { absl::string_view proto_name, absl::string_view message_name) {
MockCodeGenerator::ExpectGenerated(generator_name, "", "", proto_name, MockCodeGenerator::ExpectGenerated(generator_name, "", "", proto_name,
message_name, all_proto_names, message_name, all_proto_names,
temp_directory()); temp_directory());
} }
void CommandLineInterfaceTest::ExpectGeneratedWithInsertions( void CommandLineInterfaceTest::ExpectGeneratedWithInsertions(
const std::string& generator_name, const std::string& parameter, absl::string_view generator_name, absl::string_view parameter,
const std::string& insertions, const std::string& proto_name, absl::string_view insertions, absl::string_view proto_name,
const std::string& message_name) { absl::string_view message_name) {
MockCodeGenerator::ExpectGenerated(generator_name, parameter, insertions, MockCodeGenerator::ExpectGenerated(generator_name, parameter, insertions,
proto_name, message_name, proto_name, proto_name, message_name, proto_name,
temp_directory()); temp_directory());
} }
void CommandLineInterfaceTest::CheckGeneratedAnnotations( void CommandLineInterfaceTest::CheckGeneratedAnnotations(
const std::string& name, const std::string& file) { absl::string_view name, absl::string_view file) {
MockCodeGenerator::CheckGeneratedAnnotations(name, file, temp_directory()); MockCodeGenerator::CheckGeneratedAnnotations(name, file, temp_directory());
} }
#if defined(_WIN32) #if defined(_WIN32)
void CommandLineInterfaceTest::ExpectNullCodeGeneratorCalled( void CommandLineInterfaceTest::ExpectNullCodeGeneratorCalled(
const std::string& parameter) { absl::string_view parameter) {
EXPECT_TRUE(null_generator_->called_); EXPECT_TRUE(null_generator_->called_);
EXPECT_EQ(parameter, null_generator_->parameter_); EXPECT_EQ(parameter, null_generator_->parameter_);
} }
@ -4024,14 +4025,14 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
close(duped_stdin_); close(duped_stdin_);
} }
void RedirectStdinFromText(const std::string& input) { void RedirectStdinFromText(absl::string_view input) {
std::string filename = absl::StrCat(TestTempDir(), "/test_stdin"); std::string filename = absl::StrCat(TestTempDir(), "/test_stdin");
ABSL_CHECK_OK(File::SetContents(filename, input, true)); ABSL_CHECK_OK(File::SetContents(filename, input, true));
ABSL_CHECK(RedirectStdinFromFile(filename)); ABSL_CHECK(RedirectStdinFromFile(filename));
} }
bool RedirectStdinFromFile(const std::string& filename) { bool RedirectStdinFromFile(absl::string_view filename) {
int fd = open(filename.c_str(), O_RDONLY); int fd = open(std::string(filename).c_str(), O_RDONLY);
if (fd < 0) return false; if (fd < 0) return false;
dup2(fd, STDIN_FILENO); dup2(fd, STDIN_FILENO);
close(fd); close(fd);
@ -4039,7 +4040,7 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
} }
// Remove '\r' characters from text. // Remove '\r' characters from text.
std::string StripCR(const std::string& text) { std::string StripCR(absl::string_view text) {
std::string result; std::string result;
for (size_t i = 0; i < text.size(); ++i) { for (size_t i = 0; i < text.size(); ++i) {
@ -4054,7 +4055,7 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
enum Type { TEXT, BINARY }; enum Type { TEXT, BINARY };
enum ReturnCode { SUCCESS, ERROR }; enum ReturnCode { SUCCESS, ERROR };
bool Run(const std::string& command, bool specify_proto_files = true) { bool Run(absl::string_view command, bool specify_proto_files = true) {
std::vector<std::string> args; std::vector<std::string> args;
args.push_back("protoc"); args.push_back("protoc");
for (absl::string_view split_piece : for (absl::string_view split_piece :
@ -4094,7 +4095,7 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
return result == 0; return result == 0;
} }
void ExpectStdoutMatchesBinaryFile(const std::string& filename) { void ExpectStdoutMatchesBinaryFile(absl::string_view filename) {
std::string expected_output; std::string expected_output;
ABSL_CHECK_OK( ABSL_CHECK_OK(
File::GetContents(filename, &expected_output, true)); File::GetContents(filename, &expected_output, true));
@ -4104,7 +4105,7 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
EXPECT_TRUE(captured_stdout_ == expected_output); EXPECT_TRUE(captured_stdout_ == expected_output);
} }
void ExpectStdoutMatchesTextFile(const std::string& filename) { void ExpectStdoutMatchesTextFile(absl::string_view filename) {
std::string expected_output; std::string expected_output;
ABSL_CHECK_OK( ABSL_CHECK_OK(
File::GetContents(filename, &expected_output, true)); File::GetContents(filename, &expected_output, true));
@ -4112,15 +4113,15 @@ class EncodeDecodeTest : public testing::TestWithParam<EncodeDecodeTestMode> {
ExpectStdoutMatchesText(expected_output); ExpectStdoutMatchesText(expected_output);
} }
void ExpectStdoutMatchesText(const std::string& expected_text) { void ExpectStdoutMatchesText(absl::string_view expected_text) {
EXPECT_EQ(StripCR(expected_text), StripCR(captured_stdout_)); EXPECT_EQ(StripCR(expected_text), StripCR(captured_stdout_));
} }
void ExpectStderrMatchesText(const std::string& expected_text) { void ExpectStderrMatchesText(absl::string_view expected_text) {
EXPECT_EQ(StripCR(expected_text), StripCR(captured_stderr_)); EXPECT_EQ(StripCR(expected_text), StripCR(captured_stderr_));
} }
void ExpectStderrContainsText(const std::string& expected_text) { void ExpectStderrContainsText(absl::string_view expected_text) {
EXPECT_NE(StripCR(captured_stderr_).find(StripCR(expected_text)), EXPECT_NE(StripCR(captured_stderr_).find(StripCR(expected_text)),
std::string::npos); std::string::npos);
} }

@ -283,6 +283,7 @@ cc_test(
"//src/google/protobuf/testing", "//src/google/protobuf/testing",
"//src/google/protobuf/testing:file", "//src/google/protobuf/testing:file",
"@com_google_absl//absl/log:absl_check", "@com_google_absl//absl/log:absl_check",
"@com_google_absl//absl/strings:string_view",
"@com_google_googletest//:gtest", "@com_google_googletest//:gtest",
"@com_google_googletest//:gtest_main", "@com_google_googletest//:gtest_main",
], ],

@ -43,9 +43,9 @@ namespace cpp {
namespace { namespace {
std::string FindWithDefault( std::string FindWithDefault(
const absl::flat_hash_map<absl::string_view, std::string>& m, const absl::flat_hash_map<absl::string_view, std::string>& m,
const std::string& k, const std::string& v) { absl::string_view k, absl::string_view v) {
auto it = m.find(k); auto it = m.find(k);
if (it == m.end()) return v; if (it == m.end()) return std::string(v);
return it->second; return it->second;
} }
@ -66,8 +66,8 @@ class MockErrorCollector : public MultiFileErrorCollector {
class MockGeneratorContext : public GeneratorContext { class MockGeneratorContext : public GeneratorContext {
public: public:
void ExpectFileMatches(const std::string& virtual_filename, void ExpectFileMatches(absl::string_view virtual_filename,
const std::string& physical_filename) { absl::string_view physical_filename) {
auto it = files_.find(virtual_filename); auto it = files_.find(virtual_filename);
ASSERT_TRUE(it != files_.end()) ASSERT_TRUE(it != files_.end())
<< "Generator failed to generate file: " << virtual_filename; << "Generator failed to generate file: " << virtual_filename;

@ -183,7 +183,7 @@ RunMap FindRuns(const std::vector<const FieldDescriptor*>& fields,
return runs; return runs;
} }
void EmitNonDefaultCheck(io::Printer* p, const std::string& prefix, void EmitNonDefaultCheck(io::Printer* p, absl::string_view prefix,
const FieldDescriptor* field) { const FieldDescriptor* field) {
ABSL_CHECK(!HasHasbit(field)); ABSL_CHECK(!HasHasbit(field));
ABSL_CHECK(!field->is_repeated()); ABSL_CHECK(!field->is_repeated());
@ -222,7 +222,7 @@ bool ShouldEmitNonDefaultCheck(const FieldDescriptor* field) {
// considered non-default (will be sent over the wire), for message types // considered non-default (will be sent over the wire), for message types
// without true field presence. Should only be called if // without true field presence. Should only be called if
// !HasHasbit(field). // !HasHasbit(field).
bool MayEmitIfNonDefaultCheck(io::Printer* p, const std::string& prefix, bool MayEmitIfNonDefaultCheck(io::Printer* p, absl::string_view prefix,
const FieldDescriptor* field) { const FieldDescriptor* field) {
ABSL_CHECK(!HasHasbit(field)); ABSL_CHECK(!HasHasbit(field));
if (!ShouldEmitNonDefaultCheck(field)) return false; if (!ShouldEmitNonDefaultCheck(field)) return false;
@ -410,7 +410,7 @@ static int popcnt(uint32_t n) {
bool MaybeEmitHaswordsCheck(ChunkIterator it, ChunkIterator end, bool MaybeEmitHaswordsCheck(ChunkIterator it, ChunkIterator end,
const Options& options, const Options& options,
const std::vector<int>& has_bit_indices, const std::vector<int>& has_bit_indices,
int cached_has_word_index, const std::string& from, int cached_has_word_index, absl::string_view from,
io::Printer* p) { io::Printer* p) {
if (!it->has_hasbit || !IsProfileDriven(options) || if (!it->has_hasbit || !IsProfileDriven(options) ||
std::distance(it, end) < 2 || !it->is_rarely_present) { std::distance(it, end) < 2 || !it->is_rarely_present) {

@ -15,6 +15,7 @@
#include "google/protobuf/descriptor.pb.h" #include "google/protobuf/descriptor.pb.h"
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include "absl/log/absl_check.h" #include "absl/log/absl_check.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/annotation_test_util.h" #include "google/protobuf/compiler/annotation_test_util.h"
#include "google/protobuf/compiler/cpp/helpers.h" #include "google/protobuf/compiler/cpp/helpers.h"
@ -33,7 +34,7 @@ class CppMetadataTest : public ::testing::Test {
// code from the previously added file with name `filename`. Returns true on // code from the previously added file with name `filename`. Returns true on
// success. If pb_h is non-null, expects a .pb.h and a .pb.h.meta (copied to // success. If pb_h is non-null, expects a .pb.h and a .pb.h.meta (copied to
// pb_h and pb_h_info respecfively); similarly for proto_h and proto_h_info. // pb_h and pb_h_info respecfively); similarly for proto_h and proto_h_info.
bool CaptureMetadata(const std::string& filename, FileDescriptorProto* file, bool CaptureMetadata(absl::string_view filename, FileDescriptorProto* file,
std::string* pb_h, GeneratedCodeInfo* pb_h_info, std::string* pb_h, GeneratedCodeInfo* pb_h_info,
std::string* proto_h, GeneratedCodeInfo* proto_h_info, std::string* proto_h, GeneratedCodeInfo* proto_h_info,
std::string* pb_cc) { std::string* pb_cc) {
@ -46,7 +47,8 @@ class CppMetadataTest : public ::testing::Test {
"annotation_guard_name=guard_name:", "annotation_guard_name=guard_name:",
::testing::TempDir()); ::testing::TempDir());
const bool result = atu::RunProtoCompiler(filename, cpp_out, &cli, file); const bool result =
atu::RunProtoCompiler(std::string(filename), cpp_out, &cli, file);
if (!result) { if (!result) {
return result; return result;

@ -5,6 +5,7 @@
#include <vector> #include <vector>
#include "absl/log/die_if_null.h" #include "absl/log/die_if_null.h"
#include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "google/protobuf/io/printer.h" #include "google/protobuf/io/printer.h"
@ -18,7 +19,7 @@ NamespacePrinter::NamespacePrinter(
: p_(ABSL_DIE_IF_NULL(p)), : p_(ABSL_DIE_IF_NULL(p)),
namespace_components_(std::move(namespace_components)) { namespace_components_(std::move(namespace_components)) {
// Open the namespace. // Open the namespace.
for (const std::string& ns : namespace_components_) { for (absl::string_view ns : namespace_components_) {
p_->Print(absl::Substitute("namespace $0 {\n", ns)); p_->Print(absl::Substitute("namespace $0 {\n", ns));
} }
p_->Print("\n"); p_->Print("\n");
@ -26,7 +27,7 @@ NamespacePrinter::NamespacePrinter(
NamespacePrinter::~NamespacePrinter() { NamespacePrinter::~NamespacePrinter() {
// Close the namespace. // Close the namespace.
for (const std::string& ns : namespace_components_) { for (absl::string_view ns : namespace_components_) {
p_->Print(absl::Substitute("} // namespace $0\n", ns)); p_->Print(absl::Substitute("} // namespace $0\n", ns));
} }
} }

@ -22,6 +22,7 @@
#include "absl/strings/escaping.h" #include "absl/strings/escaping.h"
#include "absl/strings/str_replace.h" #include "absl/strings/str_replace.h"
#include "absl/strings/str_split.h" #include "absl/strings/str_split.h"
#include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "google/protobuf/compiler/csharp/csharp_generator.h" #include "google/protobuf/compiler/csharp/csharp_generator.h"
#include "google/protobuf/compiler/importer.h" #include "google/protobuf/compiler/importer.h"
@ -52,8 +53,8 @@ class MockErrorCollector : public MultiFileErrorCollector {
class MockGeneratorContext : public GeneratorContext { class MockGeneratorContext : public GeneratorContext {
public: public:
void ExpectFileMatches(const std::string& virtual_filename, void ExpectFileMatches(absl::string_view virtual_filename,
const std::string& physical_filename) { absl::string_view physical_filename) {
auto it = files_.find(virtual_filename); auto it = files_.find(virtual_filename);
ASSERT_TRUE(it != files_.end()) ASSERT_TRUE(it != files_.end())
<< "Generator failed to generate file: " << virtual_filename; << "Generator failed to generate file: " << virtual_filename;

@ -59,7 +59,6 @@ class FieldGeneratorBase : public SourceGeneratorBase {
void AddDeprecatedFlag(io::Printer* printer); void AddDeprecatedFlag(io::Printer* printer);
void AddNullCheck(io::Printer* printer); void AddNullCheck(io::Printer* printer);
void AddNullCheck(io::Printer* printer, const std::string& name);
void AddPublicMemberAttributes(io::Printer* printer); void AddPublicMemberAttributes(io::Printer* printer);
void SetCommonOneofFieldVariables( void SetCommonOneofFieldVariables(

@ -32,6 +32,7 @@
#include "absl/strings/str_split.h" #include "absl/strings/str_split.h"
#include "absl/strings/string_view.h" #include "absl/strings/string_view.h"
#include "google/protobuf/compiler/parser.h" #include "google/protobuf/compiler/parser.h"
#include "google/protobuf/descriptor.h"
#include "google/protobuf/io/io_win32.h" #include "google/protobuf/io/io_win32.h"
#include "google/protobuf/io/tokenizer.h" #include "google/protobuf/io/tokenizer.h"
#include "google/protobuf/io/zero_copy_stream_impl.h" #include "google/protobuf/io/zero_copy_stream_impl.h"
@ -72,7 +73,7 @@ MultiFileErrorCollector::~MultiFileErrorCollector() {}
class SourceTreeDescriptorDatabase::SingleFileErrorCollector class SourceTreeDescriptorDatabase::SingleFileErrorCollector
: public io::ErrorCollector { : public io::ErrorCollector {
public: public:
SingleFileErrorCollector(const std::string& filename, SingleFileErrorCollector(absl::string_view filename,
MultiFileErrorCollector* multi_file_error_collector) MultiFileErrorCollector* multi_file_error_collector)
: filename_(filename), : filename_(filename),
multi_file_error_collector_(multi_file_error_collector), multi_file_error_collector_(multi_file_error_collector),
@ -212,11 +213,11 @@ Importer::Importer(SourceTree* source_tree,
Importer::~Importer() {} Importer::~Importer() {}
const FileDescriptor* Importer::Import(const std::string& filename) { const FileDescriptor* Importer::Import(absl::string_view filename) {
return pool_.FindFileByName(filename); return pool_.FindFileByName(filename);
} }
void Importer::AddUnusedImportTrackFile(const std::string& file_name, void Importer::AddUnusedImportTrackFile(absl::string_view file_name,
bool is_error) { bool is_error) {
pool_.AddUnusedImportTrackFile(file_name, is_error); pool_.AddUnusedImportTrackFile(file_name, is_error);
} }

@ -153,13 +153,13 @@ class PROTOBUF_EXPORT Importer {
// you want to see errors for the same files repeatedly, you can use a // you want to see errors for the same files repeatedly, you can use a
// separate Importer object to import each one (but use the same // separate Importer object to import each one (but use the same
// DescriptorPool so that they can be cross-linked). // DescriptorPool so that they can be cross-linked).
const FileDescriptor* Import(const std::string& filename); const FileDescriptor* Import(absl::string_view filename);
// The DescriptorPool in which all imported FileDescriptors and their // The DescriptorPool in which all imported FileDescriptors and their
// contents are stored. // contents are stored.
inline const DescriptorPool* pool() const { return &pool_; } inline const DescriptorPool* pool() const { return &pool_; }
void AddUnusedImportTrackFile(const std::string& file_name, void AddUnusedImportTrackFile(absl::string_view file_name,
bool is_error = false); bool is_error = false);
void ClearUnusedImportTrackFiles(); void ClearUnusedImportTrackFiles();

@ -22,6 +22,7 @@
#include "absl/log/absl_check.h" #include "absl/log/absl_check.h"
#include "absl/status/status.h" #include "absl/status/status.h"
#include "absl/strings/str_cat.h" #include "absl/strings/str_cat.h"
#include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "google/protobuf/descriptor.h" #include "google/protobuf/descriptor.h"
#include "google/protobuf/io/zero_copy_stream_impl.h" #include "google/protobuf/io/zero_copy_stream_impl.h"
@ -32,7 +33,7 @@ namespace compiler {
namespace { namespace {
bool FileExists(const std::string& path) { bool FileExists(absl::string_view path) {
return File::Exists(path); return File::Exists(path);
} }
@ -93,7 +94,7 @@ class ImporterTest : public testing::Test {
protected: protected:
ImporterTest() : importer_(&source_tree_, &error_collector_) {} ImporterTest() : importer_(&source_tree_, &error_collector_) {}
void AddFile(const std::string& filename, const char* text) { void AddFile(absl::string_view filename, const char* text) {
source_tree_.AddFile(filename, text); source_tree_.AddFile(filename, text);
} }
@ -258,15 +259,15 @@ class DiskSourceTreeTest : public testing::Test {
} }
} }
void AddFile(const std::string& filename, const char* contents) { void AddFile(absl::string_view filename, const char* contents) {
ABSL_CHECK_OK(File::SetContents(filename, contents, true)); ABSL_CHECK_OK(File::SetContents(filename, contents, true));
} }
void AddSubdir(const std::string& dirname) { void AddSubdir(absl::string_view dirname) {
ABSL_CHECK_OK(File::CreateDir(dirname, 0777)); ABSL_CHECK_OK(File::CreateDir(dirname, 0777));
} }
void ExpectFileContents(const std::string& filename, void ExpectFileContents(absl::string_view filename,
const char* expected_contents) { const char* expected_contents) {
std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename)); std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename));
@ -283,8 +284,8 @@ class DiskSourceTreeTest : public testing::Test {
EXPECT_EQ(expected_contents, file_contents); EXPECT_EQ(expected_contents, file_contents);
} }
void ExpectCannotOpenFile(const std::string& filename, void ExpectCannotOpenFile(absl::string_view filename,
const std::string& error_message) { absl::string_view error_message) {
std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename)); std::unique_ptr<io::ZeroCopyInputStream> input(source_tree_.Open(filename));
EXPECT_TRUE(input == nullptr); EXPECT_TRUE(input == nullptr);
EXPECT_EQ(error_message, source_tree_.GetLastErrorMessage()); EXPECT_EQ(error_message, source_tree_.GetLastErrorMessage());

@ -16,6 +16,7 @@
#include "google/protobuf/compiler/java/field_common.h" #include "google/protobuf/compiler/java/field_common.h"
#include "google/protobuf/compiler/java/helpers.h" #include "google/protobuf/compiler/java/helpers.h"
#include "google/protobuf/compiler/java/name_resolver.h" #include "google/protobuf/compiler/java/name_resolver.h"
#include "google/protobuf/compiler/java/names.h"
#include "google/protobuf/descriptor.h" #include "google/protobuf/descriptor.h"
namespace google { namespace google {
@ -120,10 +121,10 @@ void Context::InitializeFieldGeneratorInfoForFields(
std::vector<std::string> conflict_reason(fields.size()); std::vector<std::string> conflict_reason(fields.size());
for (int i = 0; i < fields.size(); ++i) { for (int i = 0; i < fields.size(); ++i) {
const FieldDescriptor* field = fields[i]; const FieldDescriptor* field = fields[i];
const std::string& name = CapitalizedFieldName(field); const std::string name = CapitalizedFieldName(field);
for (int j = i + 1; j < fields.size(); ++j) { for (int j = i + 1; j < fields.size(); ++j) {
const FieldDescriptor* other = fields[j]; const FieldDescriptor* other = fields[j];
const std::string& other_name = CapitalizedFieldName(other); const std::string other_name = CapitalizedFieldName(other);
if (name == other_name) { if (name == other_name) {
is_conflict[i] = is_conflict[j] = true; is_conflict[i] = is_conflict[j] = true;
conflict_reason[i] = conflict_reason[j] = conflict_reason[i] = conflict_reason[j] =

@ -12,6 +12,7 @@
#include <vector> #include <vector>
#include "absl/container/flat_hash_map.h" #include "absl/container/flat_hash_map.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/java/helpers.h" #include "google/protobuf/compiler/java/helpers.h"
#include "google/protobuf/compiler/java/options.h" #include "google/protobuf/compiler/java/options.h"
#include "google/protobuf/port.h" #include "google/protobuf/port.h"
@ -87,7 +88,7 @@ class Context {
template <typename Descriptor> template <typename Descriptor>
void MaybePrintGeneratedAnnotation(Context* context, io::Printer* printer, void MaybePrintGeneratedAnnotation(Context* context, io::Printer* printer,
Descriptor* descriptor, bool immutable, Descriptor* descriptor, bool immutable,
const std::string& suffix = "") { absl::string_view suffix = "") {
if (IsOwnFile(descriptor, immutable)) { if (IsOwnFile(descriptor, immutable)) {
PrintGeneratedAnnotation(printer, '$', PrintGeneratedAnnotation(printer, '$',
context->options().annotate_code context->options().annotate_code
@ -97,7 +98,6 @@ void MaybePrintGeneratedAnnotation(Context* context, io::Printer* printer,
} }
} }
} // namespace java } // namespace java
} // namespace compiler } // namespace compiler
} // namespace protobuf } // namespace protobuf

@ -29,7 +29,7 @@ namespace protobuf {
namespace compiler { namespace compiler {
namespace java { namespace java {
std::string EscapeJavadoc(const std::string& input) { std::string EscapeJavadoc(absl::string_view input) {
std::string result; std::string result;
result.reserve(input.size() * 2); result.reserve(input.size() * 2);
@ -87,7 +87,7 @@ std::string EscapeJavadoc(const std::string& input) {
return result; return result;
} }
static std::string EscapeKdoc(const std::string& input) { static std::string EscapeKdoc(absl::string_view input) {
std::string result; std::string result;
result.reserve(input.size() * 2); result.reserve(input.size() * 2);
@ -189,8 +189,8 @@ static void WriteDocCommentBody(io::Printer* printer,
} }
} }
static std::string FirstLineOf(const std::string& value) { static std::string FirstLineOf(absl::string_view value) {
std::string result = value; std::string result = std::string(value);
std::string::size_type pos = result.find_first_of('\n'); std::string::size_type pos = result.find_first_of('\n');
if (pos != std::string::npos) { if (pos != std::string::npos) {

@ -12,6 +12,7 @@
#ifndef GOOGLE_PROTOBUF_COMPILER_JAVA_DOC_COMMENT_H__ #ifndef GOOGLE_PROTOBUF_COMPILER_JAVA_DOC_COMMENT_H__
#define GOOGLE_PROTOBUF_COMPILER_JAVA_DOC_COMMENT_H__ #define GOOGLE_PROTOBUF_COMPILER_JAVA_DOC_COMMENT_H__
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/java/options.h" #include "google/protobuf/compiler/java/options.h"
#include "google/protobuf/descriptor.h" #include "google/protobuf/descriptor.h"
@ -71,7 +72,7 @@ void WriteMethodDocComment(io::Printer* printer, const MethodDescriptor* method,
// Exposed for testing only. // Exposed for testing only.
// Also called by proto1-Java code generator. // Also called by proto1-Java code generator.
PROTOC_EXPORT std::string EscapeJavadoc(const std::string& input); PROTOC_EXPORT std::string EscapeJavadoc(absl::string_view input);
} // namespace java } // namespace java
} // namespace compiler } // namespace compiler

@ -108,7 +108,7 @@ bool CollectExtensions(const Message& message, FieldDescriptorSet* extensions) {
void CollectExtensions(const FileDescriptorProto& file_proto, void CollectExtensions(const FileDescriptorProto& file_proto,
const DescriptorPool& alternate_pool, const DescriptorPool& alternate_pool,
FieldDescriptorSet* extensions, FieldDescriptorSet* extensions,
const std::string& file_data) { absl::string_view file_data) {
if (!CollectExtensions(file_proto, extensions)) { if (!CollectExtensions(file_proto, extensions)) {
// There are unknown fields in the file_proto, which are probably // There are unknown fields in the file_proto, which are probably
// extensions. We need to parse the data into a dynamic message based on the // extensions. We need to parse the data into a dynamic message based on the
@ -513,10 +513,10 @@ void FileGenerator::GenerateDescriptorInitializationCodeForImmutable(
template <typename GeneratorClass, typename DescriptorClass> template <typename GeneratorClass, typename DescriptorClass>
static void GenerateSibling( static void GenerateSibling(
const std::string& package_dir, const std::string& java_package, absl::string_view package_dir, absl::string_view java_package,
const DescriptorClass* descriptor, GeneratorContext* context, const DescriptorClass* descriptor, GeneratorContext* context,
std::vector<std::string>* file_list, bool annotate_code, std::vector<std::string>* file_list, bool annotate_code,
std::vector<std::string>* annotation_list, const std::string& name_suffix, std::vector<std::string>* annotation_list, absl::string_view name_suffix,
GeneratorClass* generator, bool opensource_runtime, GeneratorClass* generator, bool opensource_runtime,
void (GeneratorClass::*pfn)(io::Printer* printer)) { void (GeneratorClass::*pfn)(io::Printer* printer)) {
std::string filename = std::string filename =
@ -561,7 +561,7 @@ static void GenerateSibling(
} }
void FileGenerator::GenerateSiblings( void FileGenerator::GenerateSiblings(
const std::string& package_dir, GeneratorContext* context, absl::string_view package_dir, GeneratorContext* context,
std::vector<std::string>* file_list, std::vector<std::string>* file_list,
std::vector<std::string>* annotation_list) { std::vector<std::string>* annotation_list) {
if (MultipleJavaFiles(file_, immutable_api_)) { if (MultipleJavaFiles(file_, immutable_api_)) {
@ -625,14 +625,15 @@ void FileGenerator::GenerateKotlin(io::Printer* printer) {
} }
void FileGenerator::GenerateKotlinSiblings( void FileGenerator::GenerateKotlinSiblings(
const std::string& package_dir, GeneratorContext* context, absl::string_view package_dir, GeneratorContext* context,
std::vector<std::string>* file_list, std::vector<std::string>* file_list,
std::vector<std::string>* annotation_list) { std::vector<std::string>* annotation_list) {
for (int i = 0; i < file_->message_type_count(); i++) { for (int i = 0; i < file_->message_type_count(); i++) {
const Descriptor* descriptor = file_->message_type(i); const Descriptor* descriptor = file_->message_type(i);
MessageGenerator* generator = message_generators_[i].get(); MessageGenerator* generator = message_generators_[i].get();
auto open_file = [context](const std::string& filename) { auto open_file = [context](absl::string_view filename) {
return std::unique_ptr<io::ZeroCopyOutputStream>(context->Open(filename)); return std::unique_ptr<io::ZeroCopyOutputStream>(
context->Open(std::string(filename)));
}; };
std::string filename = std::string filename =
absl::StrCat(package_dir, descriptor->name(), "Kt.kt"); absl::StrCat(package_dir, descriptor->name(), "Kt.kt");

@ -16,6 +16,7 @@
#include <string> #include <string>
#include <vector> #include <vector>
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/java/options.h" #include "google/protobuf/compiler/java/options.h"
#include "google/protobuf/port.h" #include "google/protobuf/port.h"
@ -60,7 +61,7 @@ class FileGenerator {
std::string GetKotlinClassname(); std::string GetKotlinClassname();
void GenerateKotlin(io::Printer* printer); void GenerateKotlin(io::Printer* printer);
void GenerateKotlinSiblings(const std::string& package_dir, void GenerateKotlinSiblings(absl::string_view package_dir,
GeneratorContext* generator_context, GeneratorContext* generator_context,
std::vector<std::string>* file_list, std::vector<std::string>* file_list,
std::vector<std::string>* annotation_list); std::vector<std::string>* annotation_list);
@ -68,7 +69,7 @@ class FileGenerator {
// If we aren't putting everything into one file, this will write all the // If we aren't putting everything into one file, this will write all the
// files other than the outer file (i.e. one for each message, enum, and // files other than the outer file (i.e. one for each message, enum, and
// service type). // service type).
void GenerateSiblings(const std::string& package_dir, void GenerateSiblings(absl::string_view package_dir,
GeneratorContext* generator_context, GeneratorContext* generator_context,
std::vector<std::string>* file_list, std::vector<std::string>* file_list,
std::vector<std::string>* annotation_list); std::vector<std::string>* annotation_list);

@ -15,6 +15,7 @@
#include "absl/log/absl_check.h" #include "absl/log/absl_check.h"
#include "absl/strings/str_cat.h" #include "absl/strings/str_cat.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/java/context.h" #include "google/protobuf/compiler/java/context.h"
#include "google/protobuf/compiler/java/doc_comment.h" #include "google/protobuf/compiler/java/doc_comment.h"
#include "google/protobuf/compiler/java/field_common.h" #include "google/protobuf/compiler/java/field_common.h"

@ -181,7 +181,7 @@ bool IsForbiddenKotlin(absl::string_view field_name) {
return kKotlinForbiddenNames.contains(field_name); return kKotlinForbiddenNames.contains(field_name);
} }
std::string EscapeKotlinKeywords(std::string name) { std::string EscapeKotlinKeywords(absl::string_view name) {
std::vector<std::string> escaped_packages; std::vector<std::string> escaped_packages;
std::vector<std::string> packages = absl::StrSplit(name, "."); // NOLINT std::vector<std::string> packages = absl::StrSplit(name, "."); // NOLINT
for (absl::string_view package : packages) { for (absl::string_view package : packages) {
@ -211,7 +211,7 @@ std::string FileClassName(const FileDescriptor* file, bool immutable) {
return ClassNameResolver().GetFileClassName(file, immutable); return ClassNameResolver().GetFileClassName(file, immutable);
} }
std::string JavaPackageToDir(std::string package_name) { std::string JavaPackageToDir(absl::string_view package_name) {
std::string package_dir = absl::StrReplaceAll(package_name, {{".", "/"}}); std::string package_dir = absl::StrReplaceAll(package_name, {{".", "/"}});
if (!package_dir.empty()) absl::StrAppend(&package_dir, "/"); if (!package_dir.empty()) absl::StrAppend(&package_dir, "/");
return package_dir; return package_dir;

@ -85,10 +85,10 @@ std::string FileJavaPackage(const FileDescriptor* file, bool immutable,
Options options = {}); Options options = {});
// Returns output directory for the given package name. // Returns output directory for the given package name.
std::string JavaPackageToDir(std::string package_name); std::string JavaPackageToDir(absl::string_view package_name);
// Returns the name with Kotlin keywords enclosed in backticks // Returns the name with Kotlin keywords enclosed in backticks
std::string EscapeKotlinKeywords(std::string name); std::string EscapeKotlinKeywords(absl::string_view name);
// Comma-separate list of option-specified interfaces implemented by the // Comma-separate list of option-specified interfaces implemented by the
// Message, to follow the "implements" declaration of the Message definition. // Message, to follow the "implements" declaration of the Message definition.

@ -13,6 +13,7 @@
#include "absl/strings/escaping.h" #include "absl/strings/escaping.h"
#include "absl/strings/str_cat.h" #include "absl/strings/str_cat.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/code_generator.h"
#include "google/protobuf/compiler/java/helpers.h" #include "google/protobuf/compiler/java/helpers.h"
#include "google/protobuf/compiler/java/name_resolver.h" #include "google/protobuf/compiler/java/name_resolver.h"
@ -189,7 +190,7 @@ void SharedCodeGenerator::GenerateDescriptors(io::Printer* printer) {
" new com.google.protobuf.Descriptors.FileDescriptor[] {\n"); " new com.google.protobuf.Descriptors.FileDescriptor[] {\n");
for (int i = 0; i < dependencies.size(); i++) { for (int i = 0; i < dependencies.size(); i++) {
const std::string& dependency = dependencies[i].second; absl::string_view dependency = dependencies[i].second;
printer->Print(" $dependency$.getDescriptor(),\n", "dependency", printer->Print(" $dependency$.getDescriptor(),\n", "dependency",
dependency); dependency);
} }

@ -50,10 +50,10 @@ void SetCommonFieldVariables(
UnCamelCaseFieldName(camel_case_name, descriptor)); UnCamelCaseFieldName(camel_case_name, descriptor));
const bool needs_custom_name = (raw_field_name != un_camel_case_name); const bool needs_custom_name = (raw_field_name != un_camel_case_name);
const std::string& classname = ClassName(descriptor->containing_type()); const std::string classname = ClassName(descriptor->containing_type());
(*variables)["classname"] = classname; (*variables)["classname"] = classname;
(*variables)["name"] = camel_case_name; (*variables)["name"] = camel_case_name;
const std::string& capitalized_name = FieldNameCapitalized(descriptor); const std::string capitalized_name = FieldNameCapitalized(descriptor);
(*variables)["capitalized_name"] = capitalized_name; (*variables)["capitalized_name"] = capitalized_name;
(*variables)["raw_field_name"] = raw_field_name; (*variables)["raw_field_name"] = raw_field_name;
(*variables)["field_number_name"] = (*variables)["field_number_name"] =
@ -137,8 +137,7 @@ bool HasNonZeroDefaultValue(const FieldDescriptor* field) {
case FieldDescriptor::CPPTYPE_BOOL: case FieldDescriptor::CPPTYPE_BOOL:
return field->default_value_bool(); return field->default_value_bool();
case FieldDescriptor::CPPTYPE_STRING: { case FieldDescriptor::CPPTYPE_STRING: {
const std::string& default_string = field->default_value_string(); return !field->default_value_string().empty();
return !default_string.empty();
} }
case FieldDescriptor::CPPTYPE_ENUM: case FieldDescriptor::CPPTYPE_ENUM:
// The default value for an enum field is the first enum value, so there // The default value for an enum field is the first enum value, so there

@ -75,8 +75,8 @@ class FieldGenerator {
} }
bool needs_textformat_name_support() const { bool needs_textformat_name_support() const {
const std::string& field_flags = variable("fieldflags"); return absl::StrContains(variable("fieldflags"),
return absl::StrContains(field_flags, "GPBFieldTextFormatNameCustom"); "GPBFieldTextFormatNameCustom");
} }
std::string generated_objc_name() const { return variable("name"); } std::string generated_objc_name() const { return variable("name"); }
std::string raw_field_name() const { return variable("raw_field_name"); } std::string raw_field_name() const { return variable("raw_field_name"); }

@ -97,9 +97,9 @@ bool ProtoFrameworkCollector::ConsumeLine(absl::string_view line,
} // namespace } // namespace
ImportWriter::ImportWriter( ImportWriter::ImportWriter(
const std::string& generate_for_named_framework, absl::string_view generate_for_named_framework,
const std::string& named_framework_to_proto_path_mappings_path, absl::string_view named_framework_to_proto_path_mappings_path,
const std::string& runtime_import_prefix, bool for_bundled_proto) absl::string_view runtime_import_prefix, bool for_bundled_proto)
: generate_for_named_framework_(generate_for_named_framework), : generate_for_named_framework_(generate_for_named_framework),
named_framework_to_proto_path_mappings_path_( named_framework_to_proto_path_mappings_path_(
named_framework_to_proto_path_mappings_path), named_framework_to_proto_path_mappings_path),
@ -108,7 +108,7 @@ ImportWriter::ImportWriter(
need_to_parse_mapping_file_(true) {} need_to_parse_mapping_file_(true) {}
void ImportWriter::AddFile(const FileDescriptor* file, void ImportWriter::AddFile(const FileDescriptor* file,
const std::string& header_extension) { absl::string_view header_extension) {
if (IsProtobufLibraryBundledProtoFile(file)) { if (IsProtobufLibraryBundledProtoFile(file)) {
// The imports of the WKTs are only needed within the library itself, // The imports of the WKTs are only needed within the library itself,
// in other cases, they get skipped because the generated code already // in other cases, they get skipped because the generated code already
@ -135,11 +135,11 @@ void ImportWriter::AddFile(const FileDescriptor* file,
return; return;
} }
other_imports_.push_back(FilePath(file) + header_extension); other_imports_.push_back(absl::StrCat(FilePath(file), header_extension));
} }
void ImportWriter::AddRuntimeImport(const std::string& header_name) { void ImportWriter::AddRuntimeImport(absl::string_view header_name) {
protobuf_imports_.push_back(header_name); protobuf_imports_.push_back(std::string(header_name));
} }
std::string ImportWriter::ModuleForFile(const FileDescriptor* file) { std::string ImportWriter::ModuleForFile(const FileDescriptor* file) {

@ -12,6 +12,7 @@
#include <vector> #include <vector>
#include "absl/container/flat_hash_map.h" #include "absl/container/flat_hash_map.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/descriptor.h" #include "google/protobuf/descriptor.h"
namespace google { namespace google {
@ -23,14 +24,13 @@ namespace objectivec {
// import statements. // import statements.
class ImportWriter { class ImportWriter {
public: public:
ImportWriter(const std::string& generate_for_named_framework, ImportWriter(absl::string_view generate_for_named_framework,
const std::string& named_framework_to_proto_path_mappings_path, absl::string_view named_framework_to_proto_path_mappings_path,
const std::string& runtime_import_prefix, absl::string_view runtime_import_prefix, bool for_bundled_proto);
bool for_bundled_proto);
~ImportWriter() = default; ~ImportWriter() = default;
void AddFile(const FileDescriptor* file, const std::string& header_extension); void AddFile(const FileDescriptor* file, absl::string_view header_extension);
void AddRuntimeImport(const std::string& header_name); void AddRuntimeImport(absl::string_view header_name);
// This can return an empty string if there is no module for the file. It also // This can return an empty string if there is no module for the file. It also
// does not handle bundled proto files. // does not handle bundled proto files.
std::string ModuleForFile(const FileDescriptor* file); std::string ModuleForFile(const FileDescriptor* file);

@ -192,7 +192,7 @@ const FieldDescriptor** SortFieldsByStorageSize(const Descriptor* descriptor) {
} // namespace } // namespace
MessageGenerator::MessageGenerator(const std::string& file_description_name, MessageGenerator::MessageGenerator(absl::string_view file_description_name,
const Descriptor* descriptor, const Descriptor* descriptor,
const GenerationOptions& generation_options) const GenerationOptions& generation_options)
: file_description_name_(file_description_name), : file_description_name_(file_description_name),

@ -15,6 +15,7 @@
#include "absl/container/btree_set.h" #include "absl/container/btree_set.h"
#include "absl/container/flat_hash_set.h" #include "absl/container/flat_hash_set.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/compiler/objectivec/field.h" #include "google/protobuf/compiler/objectivec/field.h"
#include "google/protobuf/compiler/objectivec/oneof.h" #include "google/protobuf/compiler/objectivec/oneof.h"
#include "google/protobuf/compiler/objectivec/options.h" #include "google/protobuf/compiler/objectivec/options.h"
@ -30,7 +31,7 @@ class ExtensionGenerator;
class MessageGenerator { class MessageGenerator {
public: public:
MessageGenerator(const std::string& file_description_name, MessageGenerator(absl::string_view file_description_name,
const Descriptor* descriptor, const Descriptor* descriptor,
const GenerationOptions& generation_options); const GenerationOptions& generation_options);
~MessageGenerator() = default; ~MessageGenerator() = default;

@ -16,6 +16,8 @@
#include "absl/strings/ascii.h" #include "absl/strings/ascii.h"
#include "absl/strings/escaping.h" #include "absl/strings/escaping.h"
#include "absl/strings/match.h" #include "absl/strings/match.h"
#include "absl/strings/str_cat.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/io/coded_stream.h" #include "google/protobuf/io/coded_stream.h"
#include "google/protobuf/io/zero_copy_stream_impl.h" #include "google/protobuf/io/zero_copy_stream_impl.h"
@ -136,19 +138,18 @@ bool DecodeDataBuilder::AddCharacter(char desired, char input) {
// If decode data can't be generated, a directive for the raw string // If decode data can't be generated, a directive for the raw string
// is used instead. // is used instead.
std::string DirectDecodeString(const std::string& str) { std::string DirectDecodeString(absl::string_view str) {
std::string result; return absl::StrCat(absl::string_view("\0", 1), // Marker for full string.
result += (char)'\0'; // Marker for full string. str, //
result += str; absl::string_view("\0", 1) // End of string.
result += (char)'\0'; // End of string. );
return result;
} }
} // namespace } // namespace
void TextFormatDecodeData::AddString(int32_t key, void TextFormatDecodeData::AddString(int32_t key,
const std::string& input_for_decode, absl::string_view input_for_decode,
const std::string& desired_output) { absl::string_view desired_output) {
for (std::vector<DataEntry>::const_iterator i = entries_.begin(); for (std::vector<DataEntry>::const_iterator i = entries_.begin();
i != entries_.end(); ++i) { i != entries_.end(); ++i) {
ABSL_CHECK(i->first != key) ABSL_CHECK(i->first != key)
@ -157,8 +158,8 @@ void TextFormatDecodeData::AddString(int32_t key,
<< "\", desired: \"" << desired_output << "\"."; << "\", desired: \"" << desired_output << "\".";
} }
const std::string& data = TextFormatDecodeData::DecodeDataForString( std::string data = TextFormatDecodeData::DecodeDataForString(input_for_decode,
input_for_decode, desired_output); desired_output);
entries_.push_back(DataEntry(key, data)); entries_.push_back(DataEntry(key, data));
} }
@ -183,7 +184,7 @@ std::string TextFormatDecodeData::Data() const {
// static // static
std::string TextFormatDecodeData::DecodeDataForString( std::string TextFormatDecodeData::DecodeDataForString(
const std::string& input_for_decode, const std::string& desired_output) { absl::string_view input_for_decode, absl::string_view desired_output) {
ABSL_CHECK(!input_for_decode.empty() && !desired_output.empty()) ABSL_CHECK(!input_for_decode.empty() && !desired_output.empty())
<< "error: got empty string for making TextFormat data, input: \"" << "error: got empty string for making TextFormat data, input: \""
<< input_for_decode << "\", desired: \"" << desired_output << "\"."; << input_for_decode << "\", desired: \"" << desired_output << "\".";

@ -14,6 +14,8 @@
#include <utility> #include <utility>
#include <vector> #include <vector>
#include "absl/strings/string_view.h"
// Must be included last // Must be included last
#include "google/protobuf/port_def.inc" #include "google/protobuf/port_def.inc"
@ -35,13 +37,13 @@ class PROTOC_EXPORT TextFormatDecodeData {
TextFormatDecodeData(const TextFormatDecodeData&) = delete; TextFormatDecodeData(const TextFormatDecodeData&) = delete;
TextFormatDecodeData& operator=(const TextFormatDecodeData&) = delete; TextFormatDecodeData& operator=(const TextFormatDecodeData&) = delete;
void AddString(int32_t key, const std::string& input_for_decode, void AddString(int32_t key, absl::string_view input_for_decode,
const std::string& desired_output); absl::string_view desired_output);
size_t num_entries() const { return entries_.size(); } size_t num_entries() const { return entries_.size(); }
std::string Data() const; std::string Data() const;
static std::string DecodeDataForString(const std::string& input_for_decode, static std::string DecodeDataForString(absl::string_view input_for_decode,
const std::string& desired_output); absl::string_view desired_output);
private: private:
typedef std::pair<int32_t, std::string> DataEntry; typedef std::pair<int32_t, std::string> DataEntry;

@ -485,7 +485,7 @@ void Parser::LocationRecorder::RecordLegacyLocation(
} }
void Parser::LocationRecorder::RecordLegacyImportLocation( void Parser::LocationRecorder::RecordLegacyImportLocation(
const Message* descriptor, const std::string& name) { const Message* descriptor, absl::string_view name) {
if (parser_->source_location_table_ != nullptr) { if (parser_->source_location_table_ != nullptr) {
parser_->source_location_table_->AddImport( parser_->source_location_table_->AddImport(
descriptor, name, location_->span(0), location_->span(1)); descriptor, name, location_->span(0), location_->span(1));
@ -2641,9 +2641,9 @@ void SourceLocationTable::Add(
} }
void SourceLocationTable::AddImport(const Message* descriptor, void SourceLocationTable::AddImport(const Message* descriptor,
const std::string& name, int line, absl::string_view name, int line,
int column) { int column) {
import_location_map_[std::make_pair(descriptor, name)] = import_location_map_[std::make_pair(descriptor, std::string(name))] =
std::make_pair(line, column); std::make_pair(line, column);
} }

@ -286,7 +286,7 @@ class PROTOBUF_EXPORT Parser final {
const Message* descriptor, const Message* descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location); DescriptorPool::ErrorCollector::ErrorLocation location);
void RecordLegacyImportLocation(const Message* descriptor, void RecordLegacyImportLocation(const Message* descriptor,
const std::string& name); absl::string_view name);
// Returns the number of path components in the recorder's current location. // Returns the number of path components in the recorder's current location.
int CurrentPathSize() const; int CurrentPathSize() const;
@ -605,7 +605,7 @@ class PROTOBUF_EXPORT SourceLocationTable {
void Add(const Message* descriptor, void Add(const Message* descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, int line, DescriptorPool::ErrorCollector::ErrorLocation location, int line,
int column); int column);
void AddImport(const Message* descriptor, const std::string& name, int line, void AddImport(const Message* descriptor, absl::string_view name, int line,
int column); int column);
// Clears the contents of the table. // Clears the contents of the table.

@ -28,6 +28,7 @@
#include "absl/memory/memory.h" #include "absl/memory/memory.h"
#include "absl/strings/str_cat.h" #include "absl/strings/str_cat.h"
#include "absl/strings/str_join.h" #include "absl/strings/str_join.h"
#include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "google/protobuf/compiler/retention.h" #include "google/protobuf/compiler/retention.h"
#include "google/protobuf/test_util2.h" #include "google/protobuf/test_util2.h"
@ -3177,19 +3178,19 @@ class SourceInfoTest : public ParserTest {
} }
bool HasSpan(char start_marker, char end_marker, bool HasSpan(char start_marker, char end_marker,
const Message& descriptor_proto, const std::string& field_name) { const Message& descriptor_proto, absl::string_view field_name) {
return HasSpan(start_marker, end_marker, descriptor_proto, field_name, -1); return HasSpan(start_marker, end_marker, descriptor_proto, field_name, -1);
} }
bool HasSpan(char start_marker, char end_marker, bool HasSpan(char start_marker, char end_marker,
const Message& descriptor_proto, const std::string& field_name, const Message& descriptor_proto, absl::string_view field_name,
int index) { int index) {
return HasSpan(start_marker, end_marker, descriptor_proto, field_name, return HasSpan(start_marker, end_marker, descriptor_proto, field_name,
index, nullptr, nullptr, nullptr); index, nullptr, nullptr, nullptr);
} }
bool HasSpan(char start_marker, char end_marker, bool HasSpan(char start_marker, char end_marker,
const Message& descriptor_proto, const std::string& field_name, const Message& descriptor_proto, absl::string_view field_name,
int index, const char* expected_leading_comments, int index, const char* expected_leading_comments,
const char* expected_trailing_comments, const char* expected_trailing_comments,
const char* expected_leading_detached_comments) { const char* expected_leading_detached_comments) {
@ -3212,7 +3213,7 @@ class SourceInfoTest : public ParserTest {
nullptr, nullptr, nullptr); nullptr, nullptr, nullptr);
} }
bool HasSpan(const Message& descriptor_proto, const std::string& field_name) { bool HasSpan(const Message& descriptor_proto, absl::string_view field_name) {
return HasSpan('\0', '\0', descriptor_proto, field_name, -1); return HasSpan('\0', '\0', descriptor_proto, field_name, -1);
} }

@ -49,7 +49,7 @@ bool IsReservedName(absl::string_view name) {
return false; return false;
} }
std::string ReservedNamePrefix(const std::string& classname, std::string ReservedNamePrefix(absl::string_view classname,
const FileDescriptor* file) { const FileDescriptor* file) {
if (IsReservedName(classname)) { if (IsReservedName(classname)) {
if (file->package() == "google.protobuf") { if (file->package() == "google.protobuf") {
@ -65,11 +65,11 @@ std::string ReservedNamePrefix(const std::string& classname,
namespace { namespace {
template <typename DescriptorType> template <typename DescriptorType>
std::string ClassNamePrefixImpl(const std::string& classname, std::string ClassNamePrefixImpl(absl::string_view classname,
const DescriptorType* desc) { const DescriptorType* desc) {
const std::string& prefix = (desc->file()->options()).php_class_prefix(); absl::string_view prefix = desc->file()->options().php_class_prefix();
if (!prefix.empty()) { if (!prefix.empty()) {
return prefix; return std::string(prefix);
} }
return ReservedNamePrefix(classname, desc->file()); return ReservedNamePrefix(classname, desc->file());
@ -94,11 +94,11 @@ std::string GeneratedClassNameImpl(const ServiceDescriptor* desc) {
} // namespace } // namespace
std::string ClassNamePrefix(const std::string& classname, std::string ClassNamePrefix(absl::string_view classname,
const Descriptor* desc) { const Descriptor* desc) {
return ClassNamePrefixImpl(classname, desc); return ClassNamePrefixImpl(classname, desc);
} }
std::string ClassNamePrefix(const std::string& classname, std::string ClassNamePrefix(absl::string_view classname,
const EnumDescriptor* desc) { const EnumDescriptor* desc) {
return ClassNamePrefixImpl(classname, desc); return ClassNamePrefixImpl(classname, desc);
} }

@ -8,11 +8,10 @@
#ifndef GOOGLE_PROTOBUF_COMPILER_PHP_NAMES_H__ #ifndef GOOGLE_PROTOBUF_COMPILER_PHP_NAMES_H__
#define GOOGLE_PROTOBUF_COMPILER_PHP_NAMES_H__ #define GOOGLE_PROTOBUF_COMPILER_PHP_NAMES_H__
#include "google/protobuf/descriptor.h"
#include <string> #include <string>
#include "absl/strings/string_view.h" #include "absl/strings/string_view.h"
#include "google/protobuf/descriptor.h"
#include "google/protobuf/port_def.inc" #include "google/protobuf/port_def.inc"
namespace google { namespace google {
@ -24,13 +23,13 @@ namespace php {
PROTOC_EXPORT bool IsReservedName(absl::string_view name); PROTOC_EXPORT bool IsReservedName(absl::string_view name);
// A prefix to stick in front of reserved names to avoid clashes. // A prefix to stick in front of reserved names to avoid clashes.
PROTOC_EXPORT std::string ReservedNamePrefix(const std::string& classname, PROTOC_EXPORT std::string ReservedNamePrefix(absl::string_view classname,
const FileDescriptor* file); const FileDescriptor* file);
// A prefix to stick in front of all class names. // A prefix to stick in front of all class names.
PROTOC_EXPORT std::string ClassNamePrefix(const std::string& classname, PROTOC_EXPORT std::string ClassNamePrefix(absl::string_view classname,
const Descriptor* desc); const Descriptor* desc);
PROTOC_EXPORT std::string ClassNamePrefix(const std::string& classname, PROTOC_EXPORT std::string ClassNamePrefix(absl::string_view classname,
const EnumDescriptor* desc); const EnumDescriptor* desc);
// To skip reserved keywords in php, some generated classname are prefixed. // To skip reserved keywords in php, some generated classname are prefixed.

@ -494,7 +494,7 @@ void Outdent(io::Printer* printer) {
printer->Outdent(); printer->Outdent();
} }
std::string BinaryToPhpString(const std::string& src) { std::string BinaryToPhpString(absl::string_view src) {
std::string dest; std::string dest;
size_t i; size_t i;
unsigned char symbol[16] = { unsigned char symbol[16] = {

@ -71,7 +71,7 @@ struct ImportModules {
}; };
// Checks whether a descriptor name matches a well-known type. // Checks whether a descriptor name matches a well-known type.
bool IsWellKnownType(const std::string& name) { bool IsWellKnownType(absl::string_view name) {
// LINT.IfChange(wktbases) // LINT.IfChange(wktbases)
return (name == "google.protobuf.Any" || return (name == "google.protobuf.Any" ||
name == "google.protobuf.Duration" || name == "google.protobuf.Duration" ||
@ -132,7 +132,7 @@ void CheckImportModules(const Descriptor* descriptor,
void PyiGenerator::PrintImportForDescriptor( void PyiGenerator::PrintImportForDescriptor(
const FileDescriptor& desc, absl::flat_hash_set<std::string>* seen_aliases, const FileDescriptor& desc, absl::flat_hash_set<std::string>* seen_aliases,
bool* has_importlib) const { bool* has_importlib) const {
const std::string& filename = desc.name(); absl::string_view filename = desc.name();
std::string module_name_owned = StrippedModuleName(filename); std::string module_name_owned = StrippedModuleName(filename);
absl::string_view module_name(module_name_owned); absl::string_view module_name(module_name_owned);
size_t last_dot_pos = module_name.rfind('.'); size_t last_dot_pos = module_name.rfind('.');
@ -282,9 +282,9 @@ printer_->Print("\n");
// Annotate wrapper for debugging purposes // Annotate wrapper for debugging purposes
// Print a message after Annotate to see what is annotated. // Print a message after Annotate to see what is annotated.
template <typename DescriptorT> template <typename DescriptorT>
void PyiGenerator::Annotate(const std::string& label, void PyiGenerator::Annotate(absl::string_view label,
const DescriptorT* descriptor) const { const DescriptorT* descriptor) const {
printer_->Annotate(label.c_str(), descriptor); printer_->Annotate(label, descriptor);
} }
void PyiGenerator::PrintEnum(const EnumDescriptor& enum_descriptor) const { void PyiGenerator::PrintEnum(const EnumDescriptor& enum_descriptor) const {

@ -17,6 +17,7 @@
#include "absl/container/flat_hash_map.h" #include "absl/container/flat_hash_map.h"
#include "absl/container/flat_hash_set.h" #include "absl/container/flat_hash_set.h"
#include "absl/strings/string_view.h"
#include "absl/synchronization/mutex.h" #include "absl/synchronization/mutex.h"
#include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/code_generator.h"
@ -66,7 +67,7 @@ class PROTOC_EXPORT PyiGenerator : public google::protobuf::compiler::CodeGenera
absl::flat_hash_set<std::string>* seen_aliases, absl::flat_hash_set<std::string>* seen_aliases,
bool* has_importlib) const; bool* has_importlib) const;
template <typename DescriptorT> template <typename DescriptorT>
void Annotate(const std::string& label, const DescriptorT* descriptor) const; void Annotate(absl::string_view label, const DescriptorT* descriptor) const;
void PrintImports() const; void PrintImports() const;
void PrintTopLevelEnums() const; void PrintTopLevelEnums() const;
void PrintEnum(const EnumDescriptor& enum_descriptor) const; void PrintEnum(const EnumDescriptor& enum_descriptor) const;

@ -24,6 +24,7 @@
#include "absl/log/absl_log.h" #include "absl/log/absl_log.h"
#include "absl/strings/escaping.h" #include "absl/strings/escaping.h"
#include "absl/strings/str_cat.h" #include "absl/strings/str_cat.h"
#include "absl/strings/string_view.h"
#include "absl/strings/substitute.h" #include "absl/strings/substitute.h"
#include "google/protobuf/io/io_win32.h" #include "google/protobuf/io/io_win32.h"
#include "google/protobuf/message.h" #include "google/protobuf/message.h"
@ -56,7 +57,7 @@ Subprocess::~Subprocess() {
} }
} }
void Subprocess::Start(const std::string& program, SearchMode search_mode) { void Subprocess::Start(absl::string_view program, SearchMode search_mode) {
// Create the pipes. // Create the pipes.
HANDLE stdin_pipe_read; HANDLE stdin_pipe_read;
HANDLE stdin_pipe_write; HANDLE stdin_pipe_write;
@ -97,7 +98,8 @@ void Subprocess::Start(const std::string& program, SearchMode search_mode) {
// get wide string version of program as the path may contain non-ascii characters // get wide string version of program as the path may contain non-ascii characters
std::wstring wprogram; std::wstring wprogram;
if (!io::win32::strings::utf8_to_wcs(program.c_str(), &wprogram)) { if (!io::win32::strings::utf8_to_wcs(std::string(program).c_str(),
&wprogram)) {
ABSL_LOG(FATAL) << "utf8_to_wcs: " << Win32ErrorMessage(GetLastError()); ABSL_LOG(FATAL) << "utf8_to_wcs: " << Win32ErrorMessage(GetLastError());
} }
@ -298,7 +300,7 @@ char* portable_strdup(const char* s) {
} }
} // namespace } // namespace
void Subprocess::Start(const std::string& program, SearchMode search_mode) { void Subprocess::Start(absl::string_view program, SearchMode search_mode) {
// Note that we assume that there are no other threads, thus we don't have to // Note that we assume that there are no other threads, thus we don't have to
// do crazy stuff like using socket pairs or avoiding libc locks. // do crazy stuff like using socket pairs or avoiding libc locks.
@ -309,7 +311,7 @@ void Subprocess::Start(const std::string& program, SearchMode search_mode) {
ABSL_CHECK(pipe(stdin_pipe) != -1); ABSL_CHECK(pipe(stdin_pipe) != -1);
ABSL_CHECK(pipe(stdout_pipe) != -1); ABSL_CHECK(pipe(stdout_pipe) != -1);
char* argv[2] = {portable_strdup(program.c_str()), nullptr}; char* argv[2] = {portable_strdup(std::string(program).c_str()), nullptr};
child_pid_ = fork(); child_pid_ = fork();
if (child_pid_ == -1) { if (child_pid_ == -1) {

@ -21,6 +21,7 @@
#endif // !_WIN32 #endif // !_WIN32
#include <string> #include <string>
#include "absl/strings/string_view.h"
#include "google/protobuf/port.h" #include "google/protobuf/port.h"
// Must be included last. // Must be included last.
@ -46,7 +47,7 @@ class PROTOC_EXPORT Subprocess {
// Start the subprocess. Currently we don't provide a way to specify // Start the subprocess. Currently we don't provide a way to specify
// arguments as protoc plugins don't have any. // arguments as protoc plugins don't have any.
void Start(const std::string& program, SearchMode search_mode); void Start(absl::string_view program, SearchMode search_mode);
// Serialize the input message and pipe it to the subprocess's stdin, then // Serialize the input message and pipe it to the subprocess's stdin, then
// close the pipe. Meanwhile, read from the subprocess's stdout and parse // close the pipe. Meanwhile, read from the subprocess's stdout and parse

@ -14,6 +14,7 @@
#include <cstdint> #include <cstdint>
#include "absl/strings/string_view.h"
#include "google/protobuf/io/coded_stream.h" #include "google/protobuf/io/coded_stream.h"
namespace google { namespace google {
@ -69,7 +70,7 @@ static const uint32_t kCRC32Table[256] = {
0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94, 0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d}; 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d};
static uint32_t ComputeCRC32(const std::string& buf) { static uint32_t ComputeCRC32(absl::string_view buf) {
uint32_t x = ~0U; uint32_t x = ~0U;
for (int i = 0; i < buf.size(); ++i) { for (int i = 0; i < buf.size(); ++i) {
unsigned char c = buf[i]; unsigned char c = buf[i];
@ -89,11 +90,10 @@ ZipWriter::ZipWriter(io::ZeroCopyOutputStream* raw_output)
: raw_output_(raw_output) {} : raw_output_(raw_output) {}
ZipWriter::~ZipWriter() {} ZipWriter::~ZipWriter() {}
bool ZipWriter::Write(const std::string& filename, bool ZipWriter::Write(absl::string_view filename, absl::string_view contents) {
const std::string& contents) {
FileInfo info; FileInfo info;
info.name = filename; info.name = std::string(filename);
uint16_t filename_size = filename.size(); uint16_t filename_size = filename.size();
info.offset = raw_output_->ByteCount(); info.offset = raw_output_->ByteCount();
info.size = contents.size(); info.size = contents.size();
@ -127,7 +127,7 @@ bool ZipWriter::WriteDirectory() {
// write central directory // write central directory
io::CodedOutputStream output(raw_output_); io::CodedOutputStream output(raw_output_);
for (int i = 0; i < num_entries; ++i) { for (int i = 0; i < num_entries; ++i) {
const std::string& filename = files_[i].name; absl::string_view filename = files_[i].name;
uint16_t filename_size = filename.size(); uint16_t filename_size = filename.size();
uint32_t crc32 = files_[i].crc32; uint32_t crc32 = files_[i].crc32;
uint32_t size = files_[i].size; uint32_t size = files_[i].size;

@ -13,6 +13,7 @@
#include <vector> #include <vector>
#include "google/protobuf/stubs/common.h" #include "google/protobuf/stubs/common.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/io/zero_copy_stream.h" #include "google/protobuf/io/zero_copy_stream.h"
namespace google { namespace google {
@ -24,7 +25,7 @@ class ZipWriter {
ZipWriter(io::ZeroCopyOutputStream* raw_output); ZipWriter(io::ZeroCopyOutputStream* raw_output);
~ZipWriter(); ~ZipWriter();
bool Write(const std::string& filename, const std::string& contents); bool Write(absl::string_view filename, absl::string_view contents);
bool WriteDirectory(); bool WriteDirectory();
private: private:

@ -94,7 +94,7 @@ using ::google::protobuf::internal::DownCast;
const int kPackageLimit = 100; const int kPackageLimit = 100;
std::string ToCamelCase(const std::string& input, bool lower_first) { std::string ToCamelCase(absl::string_view input, bool lower_first) {
bool capitalize_next = !lower_first; bool capitalize_next = !lower_first;
std::string result; std::string result;
result.reserve(input.size()); result.reserve(input.size());
@ -118,7 +118,7 @@ std::string ToCamelCase(const std::string& input, bool lower_first) {
return result; return result;
} }
std::string ToJsonName(const std::string& input) { std::string ToJsonName(absl::string_view input) {
bool capitalize_next = false; bool capitalize_next = false;
std::string result; std::string result;
result.reserve(input.size()); result.reserve(input.size());
@ -425,7 +425,7 @@ class FlatAllocatorImpl {
// It will dedup the strings when possible. // It will dedup the strings when possible.
// The resulting array contains `name` at index 0, `full_name` at index 1 // The resulting array contains `name` at index 0, `full_name` at index 1
// and the other 3 indices are specified in the result. // and the other 3 indices are specified in the result.
void PlanFieldNames(const std::string& name, void PlanFieldNames(absl::string_view name,
const std::string* opt_json_name) { const std::string* opt_json_name) {
ABSL_CHECK(!has_allocated()); ABSL_CHECK(!has_allocated());
@ -443,7 +443,7 @@ class FlatAllocatorImpl {
} }
} }
std::string lowercase_name = name; std::string lowercase_name = std::string(name);
absl::AsciiStrToLower(&lowercase_name); absl::AsciiStrToLower(&lowercase_name);
std::string camelcase_name = ToCamelCase(name, /* lower_first = */ true); std::string camelcase_name = ToCamelCase(name, /* lower_first = */ true);
@ -465,13 +465,13 @@ class FlatAllocatorImpl {
int camelcase_index; int camelcase_index;
int json_index; int json_index;
}; };
FieldNamesResult AllocateFieldNames(const std::string& name, FieldNamesResult AllocateFieldNames(absl::string_view name,
const std::string& scope, absl::string_view scope,
const std::string* opt_json_name) { const std::string* opt_json_name) {
ABSL_CHECK(has_allocated()); ABSL_CHECK(has_allocated());
std::string full_name = std::string full_name =
scope.empty() ? name : absl::StrCat(scope, ".", name); scope.empty() ? std::string(name) : absl::StrCat(scope, ".", name);
// Fast path for snake_case names, which follow the style guide. // Fast path for snake_case names, which follow the style guide.
if (opt_json_name == nullptr) { if (opt_json_name == nullptr) {
@ -490,7 +490,7 @@ class FlatAllocatorImpl {
} }
std::vector<std::string> names; std::vector<std::string> names;
names.push_back(name); names.push_back(std::string(name));
names.push_back(std::move(full_name)); names.push_back(std::move(full_name));
const auto push_name = [&](std::string new_name) { const auto push_name = [&](std::string new_name) {
@ -507,7 +507,7 @@ class FlatAllocatorImpl {
FieldNamesResult result{nullptr, 0, 0, 0}; FieldNamesResult result{nullptr, 0, 0, 0};
std::string lowercase_name = name; std::string lowercase_name = std::string(name);
absl::AsciiStrToLower(&lowercase_name); absl::AsciiStrToLower(&lowercase_name);
result.lowercase_index = push_name(std::move(lowercase_name)); result.lowercase_index = push_name(std::move(lowercase_name));
result.camelcase_index = result.camelcase_index =
@ -547,8 +547,8 @@ class FlatAllocatorImpl {
static bool IsLowerOrDigit(char c) { return IsLower(c) || IsDigit(c); } static bool IsLowerOrDigit(char c) { return IsLower(c) || IsDigit(c); }
enum class FieldNameCase { kAllLower, kSnakeCase, kOther }; enum class FieldNameCase { kAllLower, kSnakeCase, kOther };
FieldNameCase GetFieldNameCase(const std::string& name) { FieldNameCase GetFieldNameCase(absl::string_view name) {
if (!IsLower(name[0])) return FieldNameCase::kOther; if (!name.empty() && !IsLower(name[0])) return FieldNameCase::kOther;
FieldNameCase best = FieldNameCase::kAllLower; FieldNameCase best = FieldNameCase::kAllLower;
for (char c : name) { for (char c : name) {
if (IsLowerOrDigit(c)) { if (IsLowerOrDigit(c)) {
@ -835,7 +835,7 @@ const int FieldDescriptor::kLastReservedNumber;
namespace { namespace {
std::string EnumValueToPascalCase(const std::string& input) { std::string EnumValueToPascalCase(absl::string_view input) {
bool next_upper = true; bool next_upper = true;
std::string result; std::string result;
result.reserve(input.size()); result.reserve(input.size());
@ -1082,7 +1082,7 @@ absl::flat_hash_set<std::string>* NewAllowedProto3Extendee() {
// Only extensions to descriptor options are allowed. We use name comparison // Only extensions to descriptor options are allowed. We use name comparison
// instead of comparing the descriptor directly because the extensions may be // instead of comparing the descriptor directly because the extensions may be
// defined in a different pool. // defined in a different pool.
bool AllowedExtendeeInProto3(const std::string& name) { bool AllowedExtendeeInProto3(absl::string_view name) {
static auto allowed_proto3_extendees = static auto allowed_proto3_extendees =
internal::OnShutdownDelete(NewAllowedProto3Extendee()); internal::OnShutdownDelete(NewAllowedProto3Extendee());
return allowed_proto3_extendees->find(name) != return allowed_proto3_extendees->find(name) !=
@ -3257,7 +3257,7 @@ bool FormatLineOptions(int depth, const Message& options,
std::string prefix(depth * 2, ' '); std::string prefix(depth * 2, ' ');
std::vector<std::string> all_options; std::vector<std::string> all_options;
if (RetrieveOptions(depth, options, pool, &all_options)) { if (RetrieveOptions(depth, options, pool, &all_options)) {
for (const std::string& option : all_options) { for (absl::string_view option : all_options) {
absl::SubstituteAndAppend(output, "$0option $1;\n", prefix, option); absl::SubstituteAndAppend(output, "$0option $1;\n", prefix, option);
} }
} }
@ -3275,7 +3275,7 @@ static std::string GetLegacySyntaxName(Edition edition) {
class SourceLocationCommentPrinter { class SourceLocationCommentPrinter {
public: public:
template <typename DescType> template <typename DescType>
SourceLocationCommentPrinter(const DescType* desc, const std::string& prefix, SourceLocationCommentPrinter(const DescType* desc, absl::string_view prefix,
const DebugStringOptions& options) const DebugStringOptions& options)
: options_(options), prefix_(prefix) { : options_(options), prefix_(prefix) {
// Perform the SourceLocation lookup only if we're including user comments, // Perform the SourceLocation lookup only if we're including user comments,
@ -3285,7 +3285,7 @@ class SourceLocationCommentPrinter {
} }
SourceLocationCommentPrinter(const FileDescriptor* file, SourceLocationCommentPrinter(const FileDescriptor* file,
const std::vector<int>& path, const std::vector<int>& path,
const std::string& prefix, absl::string_view prefix,
const DebugStringOptions& options) const DebugStringOptions& options)
: options_(options), prefix_(prefix) { : options_(options), prefix_(prefix) {
// Perform the SourceLocation lookup only if we're including user comments, // Perform the SourceLocation lookup only if we're including user comments,
@ -3296,7 +3296,7 @@ class SourceLocationCommentPrinter {
void AddPreComment(std::string* output) { void AddPreComment(std::string* output) {
if (have_source_loc_) { if (have_source_loc_) {
// Detached leading comments. // Detached leading comments.
for (const std::string& leading_detached_comment : for (absl::string_view leading_detached_comment :
source_loc_.leading_detached_comments) { source_loc_.leading_detached_comments) {
absl::StrAppend(output, FormatComment(leading_detached_comment), "\n"); absl::StrAppend(output, FormatComment(leading_detached_comment), "\n");
} }
@ -3314,9 +3314,9 @@ class SourceLocationCommentPrinter {
// Format comment such that each line becomes a full-line C++-style comment in // Format comment such that each line becomes a full-line C++-style comment in
// the DebugString() output. // the DebugString() output.
std::string FormatComment(const std::string& comment_text) { std::string FormatComment(absl::string_view comment_text) {
std::string stripped_comment = comment_text; absl::string_view stripped_comment =
absl::StripAsciiWhitespace(&stripped_comment); absl::StripAsciiWhitespace(comment_text);
std::string output; std::string output;
for (absl::string_view line : absl::StrSplit(stripped_comment, '\n')) { for (absl::string_view line : absl::StrSplit(stripped_comment, '\n')) {
absl::SubstituteAndAppend(&output, "$0// $1\n", prefix_, line); absl::SubstituteAndAppend(&output, "$0// $1\n", prefix_, line);
@ -4219,10 +4219,10 @@ class DescriptorBuilder {
// The `const char*` overload should only be used for string literal messages // The `const char*` overload should only be used for string literal messages
// where this is a frustrating amount of overhead and there is no harm in // where this is a frustrating amount of overhead and there is no harm in
// directly using the literal. // directly using the literal.
void AddError(const std::string& element_name, const Message& descriptor, void AddError(absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
absl::FunctionRef<std::string()> make_error); absl::FunctionRef<std::string()> make_error);
void AddError(const std::string& element_name, const Message& descriptor, void AddError(absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
const char* error); const char* error);
void AddRecursiveImportError(const FileDescriptorProto& proto, int from_here); void AddRecursiveImportError(const FileDescriptorProto& proto, int from_here);
@ -4232,14 +4232,14 @@ class DescriptorBuilder {
// Adds an error indicating that undefined_symbol was not defined. Must // Adds an error indicating that undefined_symbol was not defined. Must
// only be called after LookupSymbol() fails. // only be called after LookupSymbol() fails.
void AddNotDefinedError( void AddNotDefinedError(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
const std::string& undefined_symbol); absl::string_view undefined_symbol);
void AddWarning(const std::string& element_name, const Message& descriptor, void AddWarning(absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
absl::FunctionRef<std::string()> make_error); absl::FunctionRef<std::string()> make_error);
void AddWarning(const std::string& element_name, const Message& descriptor, void AddWarning(absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
const char* error); const char* error);
@ -4256,16 +4256,16 @@ class DescriptorBuilder {
// - Search the pool's underlay if not found in tables_. // - Search the pool's underlay if not found in tables_.
// - Insure that the resulting Symbol is from one of the file's declared // - Insure that the resulting Symbol is from one of the file's declared
// dependencies. // dependencies.
Symbol FindSymbol(const std::string& name, bool build_it = true); Symbol FindSymbol(absl::string_view name, bool build_it = true);
// Like FindSymbol() but does not require that the symbol is in one of the // Like FindSymbol() but does not require that the symbol is in one of the
// file's declared dependencies. // file's declared dependencies.
Symbol FindSymbolNotEnforcingDeps(const std::string& name, Symbol FindSymbolNotEnforcingDeps(absl::string_view name,
bool build_it = true); bool build_it = true);
// This implements the body of FindSymbolNotEnforcingDeps(). // This implements the body of FindSymbolNotEnforcingDeps().
Symbol FindSymbolNotEnforcingDepsHelper(const DescriptorPool* pool, Symbol FindSymbolNotEnforcingDepsHelper(const DescriptorPool* pool,
const std::string& name, absl::string_view name,
bool build_it = true); bool build_it = true);
// Like FindSymbol(), but looks up the name relative to some other symbol // Like FindSymbol(), but looks up the name relative to some other symbol
@ -4283,7 +4283,7 @@ class DescriptorBuilder {
// if it believes that's all it could refer to. The caller should always // if it believes that's all it could refer to. The caller should always
// check that it receives the type of symbol it was expecting. // check that it receives the type of symbol it was expecting.
enum ResolveMode { LOOKUP_ALL, LOOKUP_TYPES }; enum ResolveMode { LOOKUP_ALL, LOOKUP_TYPES };
Symbol LookupSymbol(const std::string& name, const std::string& relative_to, Symbol LookupSymbol(absl::string_view name, absl::string_view relative_to,
DescriptorPool::PlaceholderType placeholder_type = DescriptorPool::PlaceholderType placeholder_type =
DescriptorPool::PLACEHOLDER_MESSAGE, DescriptorPool::PLACEHOLDER_MESSAGE,
ResolveMode resolve_mode = LOOKUP_ALL, ResolveMode resolve_mode = LOOKUP_ALL,
@ -4291,28 +4291,28 @@ class DescriptorBuilder {
// Like LookupSymbol() but will not return a placeholder even if // Like LookupSymbol() but will not return a placeholder even if
// AllowUnknownDependencies() has been used. // AllowUnknownDependencies() has been used.
Symbol LookupSymbolNoPlaceholder(const std::string& name, Symbol LookupSymbolNoPlaceholder(absl::string_view name,
const std::string& relative_to, absl::string_view relative_to,
ResolveMode resolve_mode = LOOKUP_ALL, ResolveMode resolve_mode = LOOKUP_ALL,
bool build_it = true); bool build_it = true);
// Calls tables_->AddSymbol() and records an error if it fails. Returns // Calls tables_->AddSymbol() and records an error if it fails. Returns
// true if successful or false if failed, though most callers can ignore // true if successful or false if failed, though most callers can ignore
// the return value since an error has already been recorded. // the return value since an error has already been recorded.
bool AddSymbol(const std::string& full_name, const void* parent, bool AddSymbol(absl::string_view full_name, const void* parent,
const std::string& name, const Message& proto, Symbol symbol); absl::string_view name, const Message& proto, Symbol symbol);
// Like AddSymbol(), but succeeds if the symbol is already defined as long // Like AddSymbol(), but succeeds if the symbol is already defined as long
// as the existing definition is also a package (because it's OK to define // as the existing definition is also a package (because it's OK to define
// the same package in two different files). Also adds all parents of the // the same package in two different files). Also adds all parents of the
// package to the symbol table (e.g. AddPackage("foo.bar", ...) will add // package to the symbol table (e.g. AddPackage("foo.bar", ...) will add
// "foo.bar" and "foo" to the table). // "foo.bar" and "foo" to the table).
void AddPackage(const std::string& name, const Message& proto, void AddPackage(absl::string_view name, const Message& proto,
FileDescriptor* file); FileDescriptor* file, bool top_level);
// Checks that the symbol name contains only alphanumeric characters and // Checks that the symbol name contains only alphanumeric characters and
// underscores. Records an error otherwise. // underscores. Records an error otherwise.
void ValidateSymbolName(const std::string& name, const std::string& full_name, void ValidateSymbolName(absl::string_view name, absl::string_view full_name,
const Message& proto); const Message& proto);
// Allocates a copy of orig_options in tables_ and stores it in the // Allocates a copy of orig_options in tables_ and stores it in the
@ -4365,8 +4365,8 @@ class DescriptorBuilder {
// Allocates an array of two strings, the first one is a copy of // Allocates an array of two strings, the first one is a copy of
// `proto_name`, and the second one is the full name. Full proto name is // `proto_name`, and the second one is the full name. Full proto name is
// "scope.proto_name" if scope is non-empty and "proto_name" otherwise. // "scope.proto_name" if scope is non-empty and "proto_name" otherwise.
const std::string* AllocateNameStrings(const std::string& scope, const std::string* AllocateNameStrings(absl::string_view scope,
const std::string& proto_name, absl::string_view proto_name,
internal::FlatAllocator& alloc); internal::FlatAllocator& alloc);
// These methods all have the same signature for the sake of the BUILD_ARRAY // These methods all have the same signature for the sake of the BUILD_ARRAY
@ -4411,7 +4411,7 @@ class DescriptorBuilder {
void CheckFieldJsonNameUniqueness(const DescriptorProto& proto, void CheckFieldJsonNameUniqueness(const DescriptorProto& proto,
const Descriptor* result); const Descriptor* result);
void CheckFieldJsonNameUniqueness(const std::string& message_name, void CheckFieldJsonNameUniqueness(absl::string_view message_name,
const DescriptorProto& message, const DescriptorProto& message,
const Descriptor* descriptor, const Descriptor* descriptor,
bool use_custom_names); bool use_custom_names);
@ -4510,7 +4510,7 @@ class DescriptorBuilder {
std::vector<const FieldDescriptor*>::const_iterator std::vector<const FieldDescriptor*>::const_iterator
intermediate_fields_end, intermediate_fields_end,
const FieldDescriptor* innermost_field, const FieldDescriptor* innermost_field,
const std::string& debug_msg_name, absl::string_view debug_msg_name,
const UnknownFieldSet& unknown_fields); const UnknownFieldSet& unknown_fields);
// Validates the value for the option field of the currently interpreted // Validates the value for the option field of the currently interpreted
@ -4641,7 +4641,7 @@ class DescriptorBuilder {
void ValidateExtensionRangeOptions(const DescriptorProto& proto, void ValidateExtensionRangeOptions(const DescriptorProto& proto,
const Descriptor& message); const Descriptor& message);
void ValidateExtensionDeclaration( void ValidateExtensionDeclaration(
const std::string& full_name, absl::string_view full_name,
const RepeatedPtrField<ExtensionRangeOptions_Declaration>& declarations, const RepeatedPtrField<ExtensionRangeOptions_Declaration>& declarations,
const DescriptorProto_ExtensionRange& proto, const DescriptorProto_ExtensionRange& proto,
absl::flat_hash_set<absl::string_view>& full_name_set); absl::flat_hash_set<absl::string_view>& full_name_set);
@ -4780,7 +4780,7 @@ DescriptorBuilder::DescriptorBuilder(
DescriptorBuilder::~DescriptorBuilder() = default; DescriptorBuilder::~DescriptorBuilder() = default;
PROTOBUF_NOINLINE void DescriptorBuilder::AddError( PROTOBUF_NOINLINE void DescriptorBuilder::AddError(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
absl::FunctionRef<std::string()> make_error) { absl::FunctionRef<std::string()> make_error) {
std::string error = make_error(); std::string error = make_error();
@ -4798,15 +4798,15 @@ PROTOBUF_NOINLINE void DescriptorBuilder::AddError(
} }
PROTOBUF_NOINLINE void DescriptorBuilder::AddError( PROTOBUF_NOINLINE void DescriptorBuilder::AddError(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, const char* error) { DescriptorPool::ErrorCollector::ErrorLocation location, const char* error) {
AddError(element_name, descriptor, location, [error] { return error; }); AddError(element_name, descriptor, location, [error] { return error; });
} }
PROTOBUF_NOINLINE void DescriptorBuilder::AddNotDefinedError( PROTOBUF_NOINLINE void DescriptorBuilder::AddNotDefinedError(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
const std::string& undefined_symbol) { absl::string_view undefined_symbol) {
if (possible_undeclared_dependency_ == nullptr && if (possible_undeclared_dependency_ == nullptr &&
undefine_resolved_name_.empty()) { undefine_resolved_name_.empty()) {
AddError(element_name, descriptor, location, [&] { AddError(element_name, descriptor, location, [&] {
@ -4840,7 +4840,7 @@ PROTOBUF_NOINLINE void DescriptorBuilder::AddNotDefinedError(
} }
PROTOBUF_NOINLINE void DescriptorBuilder::AddWarning( PROTOBUF_NOINLINE void DescriptorBuilder::AddWarning(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, DescriptorPool::ErrorCollector::ErrorLocation location,
absl::FunctionRef<std::string()> make_error) { absl::FunctionRef<std::string()> make_error) {
std::string error = make_error(); std::string error = make_error();
@ -4853,7 +4853,7 @@ PROTOBUF_NOINLINE void DescriptorBuilder::AddWarning(
} }
PROTOBUF_NOINLINE void DescriptorBuilder::AddWarning( PROTOBUF_NOINLINE void DescriptorBuilder::AddWarning(
const std::string& element_name, const Message& descriptor, absl::string_view element_name, const Message& descriptor,
DescriptorPool::ErrorCollector::ErrorLocation location, const char* error) { DescriptorPool::ErrorCollector::ErrorLocation location, const char* error) {
AddWarning(element_name, descriptor, location, AddWarning(element_name, descriptor, location,
[error]() -> std::string { return error; }); [error]() -> std::string { return error; });
@ -4874,7 +4874,7 @@ void DescriptorBuilder::RecordPublicDependencies(const FileDescriptor* file) {
} }
Symbol DescriptorBuilder::FindSymbolNotEnforcingDepsHelper( Symbol DescriptorBuilder::FindSymbolNotEnforcingDepsHelper(
const DescriptorPool* pool, const std::string& name, bool build_it) { const DescriptorPool* pool, absl::string_view name, bool build_it) {
// If we are looking at an underlay, we must lock its mutex_, since we are // If we are looking at an underlay, we must lock its mutex_, since we are
// accessing the underlay's tables_ directly. // accessing the underlay's tables_ directly.
absl::MutexLockMaybe lock((pool == pool_) ? nullptr : pool->mutex_); absl::MutexLockMaybe lock((pool == pool_) ? nullptr : pool->mutex_);
@ -4902,7 +4902,7 @@ Symbol DescriptorBuilder::FindSymbolNotEnforcingDepsHelper(
return result; return result;
} }
Symbol DescriptorBuilder::FindSymbolNotEnforcingDeps(const std::string& name, Symbol DescriptorBuilder::FindSymbolNotEnforcingDeps(absl::string_view name,
bool build_it) { bool build_it) {
Symbol result = FindSymbolNotEnforcingDepsHelper(pool_, name, build_it); Symbol result = FindSymbolNotEnforcingDepsHelper(pool_, name, build_it);
// Only find symbols which were defined in this file or one of its // Only find symbols which were defined in this file or one of its
@ -4914,7 +4914,7 @@ Symbol DescriptorBuilder::FindSymbolNotEnforcingDeps(const std::string& name,
return result; return result;
} }
Symbol DescriptorBuilder::FindSymbol(const std::string& name, bool build_it) { Symbol DescriptorBuilder::FindSymbol(absl::string_view name, bool build_it) {
Symbol result = FindSymbolNotEnforcingDeps(name, build_it); Symbol result = FindSymbolNotEnforcingDeps(name, build_it);
if (result.IsNull()) return result; if (result.IsNull()) return result;
@ -4947,12 +4947,12 @@ Symbol DescriptorBuilder::FindSymbol(const std::string& name, bool build_it) {
} }
possible_undeclared_dependency_ = file; possible_undeclared_dependency_ = file;
possible_undeclared_dependency_name_ = name; possible_undeclared_dependency_name_ = std::string(name);
return Symbol(); return Symbol();
} }
Symbol DescriptorBuilder::LookupSymbolNoPlaceholder( Symbol DescriptorBuilder::LookupSymbolNoPlaceholder(
const std::string& name, const std::string& relative_to, absl::string_view name, absl::string_view relative_to,
ResolveMode resolve_mode, bool build_it) { ResolveMode resolve_mode, bool build_it) {
possible_undeclared_dependency_ = nullptr; possible_undeclared_dependency_ = nullptr;
undefine_resolved_name_.clear(); undefine_resolved_name_.clear();
@ -4973,9 +4973,9 @@ Symbol DescriptorBuilder::LookupSymbolNoPlaceholder(
// } // }
// So, we look for just "Foo" first, then look for "Bar.baz" within it if // So, we look for just "Foo" first, then look for "Bar.baz" within it if
// found. // found.
std::string::size_type name_dot_pos = name.find_first_of('.'); auto name_dot_pos = name.find_first_of('.');
std::string first_part_of_name; absl::string_view first_part_of_name;
if (name_dot_pos == std::string::npos) { if (name_dot_pos == name.npos) {
first_part_of_name = name; first_part_of_name = name;
} else { } else {
first_part_of_name = name.substr(0, name_dot_pos); first_part_of_name = name.substr(0, name_dot_pos);
@ -4995,14 +4995,14 @@ Symbol DescriptorBuilder::LookupSymbolNoPlaceholder(
// Append ".first_part_of_name" and try to find. // Append ".first_part_of_name" and try to find.
std::string::size_type old_size = scope_to_try.size(); std::string::size_type old_size = scope_to_try.size();
scope_to_try.append(1, '.'); scope_to_try.append(1, '.');
scope_to_try.append(first_part_of_name); scope_to_try.append(std::string(first_part_of_name));
Symbol result = FindSymbol(scope_to_try, build_it); Symbol result = FindSymbol(scope_to_try, build_it);
if (!result.IsNull()) { if (!result.IsNull()) {
if (first_part_of_name.size() < name.size()) { if (first_part_of_name.size() < name.size()) {
// name is a compound symbol, of which we only found the first part. // name is a compound symbol, of which we only found the first part.
// Now try to look up the rest of it. // Now try to look up the rest of it.
if (result.IsAggregate()) { if (result.IsAggregate()) {
scope_to_try.append(name, first_part_of_name.size(), scope_to_try.append(std::string(name), first_part_of_name.size(),
name.size() - first_part_of_name.size()); name.size() - first_part_of_name.size());
result = FindSymbol(scope_to_try, build_it); result = FindSymbol(scope_to_try, build_it);
if (result.IsNull()) { if (result.IsNull()) {
@ -5027,7 +5027,7 @@ Symbol DescriptorBuilder::LookupSymbolNoPlaceholder(
} }
Symbol DescriptorBuilder::LookupSymbol( Symbol DescriptorBuilder::LookupSymbol(
const std::string& name, const std::string& relative_to, absl::string_view name, absl::string_view relative_to,
DescriptorPool::PlaceholderType placeholder_type, ResolveMode resolve_mode, DescriptorPool::PlaceholderType placeholder_type, ResolveMode resolve_mode,
bool build_it) { bool build_it) {
Symbol result = Symbol result =
@ -5224,8 +5224,8 @@ FileDescriptor* DescriptorPool::NewPlaceholderFileWithMutexHeld(
return placeholder; return placeholder;
} }
bool DescriptorBuilder::AddSymbol(const std::string& full_name, bool DescriptorBuilder::AddSymbol(absl::string_view full_name,
const void* parent, const std::string& name, const void* parent, absl::string_view name,
const Message& proto, Symbol symbol) { const Message& proto, Symbol symbol) {
// If the caller passed nullptr for the parent, the symbol is at file scope. // If the caller passed nullptr for the parent, the symbol is at file scope.
// Use its file as the parent instead. // Use its file as the parent instead.
@ -5253,8 +5253,8 @@ bool DescriptorBuilder::AddSymbol(const std::string& full_name,
} else { } else {
const FileDescriptor* other_file = tables_->FindSymbol(full_name).GetFile(); const FileDescriptor* other_file = tables_->FindSymbol(full_name).GetFile();
if (other_file == file_) { if (other_file == file_) {
std::string::size_type dot_pos = full_name.find_last_of('.'); auto dot_pos = full_name.find_last_of('.');
if (dot_pos == std::string::npos) { if (dot_pos == full_name.npos) {
AddError(full_name, proto, DescriptorPool::ErrorCollector::NAME, [&] { AddError(full_name, proto, DescriptorPool::ErrorCollector::NAME, [&] {
return absl::StrCat("\"", full_name, "\" is already defined."); return absl::StrCat("\"", full_name, "\" is already defined.");
}); });
@ -5277,8 +5277,8 @@ bool DescriptorBuilder::AddSymbol(const std::string& full_name,
} }
} }
void DescriptorBuilder::AddPackage(const std::string& name, void DescriptorBuilder::AddPackage(absl::string_view name, const Message& proto,
const Message& proto, FileDescriptor* file) { FileDescriptor* file, bool top_level) {
if (absl::StrContains(name, '\0')) { if (absl::StrContains(name, '\0')) {
AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] { AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] {
return absl::StrCat("\"", name, "\" contains null character."); return absl::StrCat("\"", name, "\" contains null character.");
@ -5289,7 +5289,8 @@ void DescriptorBuilder::AddPackage(const std::string& name,
Symbol existing_symbol = tables_->FindSymbol(name); Symbol existing_symbol = tables_->FindSymbol(name);
// It's OK to redefine a package. // It's OK to redefine a package.
if (existing_symbol.IsNull()) { if (existing_symbol.IsNull()) {
if (name.data() == file->package().data()) { if (top_level) {
ABSL_DCHECK_EQ(name, file->package());
// It is the toplevel package name, so insert the descriptor directly. // It is the toplevel package name, so insert the descriptor directly.
tables_->AddSymbol(file->package(), Symbol(file)); tables_->AddSymbol(file->package(), Symbol(file));
} else { } else {
@ -5301,13 +5302,13 @@ void DescriptorBuilder::AddPackage(const std::string& name,
tables_->AddSymbol(name, Symbol(package)); tables_->AddSymbol(name, Symbol(package));
} }
// Also add parent package, if any. // Also add parent package, if any.
std::string::size_type dot_pos = name.find_last_of('.'); auto dot_pos = name.find_last_of('.');
if (dot_pos == std::string::npos) { if (dot_pos == name.npos) {
// No parents. // No parents.
ValidateSymbolName(name, name, proto); ValidateSymbolName(name, name, proto);
} else { } else {
// Has parent. // Has parent.
AddPackage(name.substr(0, dot_pos), proto, file); AddPackage(name.substr(0, dot_pos), proto, file, false);
ValidateSymbolName(name.substr(dot_pos + 1), name, proto); ValidateSymbolName(name.substr(dot_pos + 1), name, proto);
} }
} else if (!existing_symbol.IsPackage()) { } else if (!existing_symbol.IsPackage()) {
@ -5323,8 +5324,8 @@ void DescriptorBuilder::AddPackage(const std::string& name,
} }
} }
void DescriptorBuilder::ValidateSymbolName(const std::string& name, void DescriptorBuilder::ValidateSymbolName(absl::string_view name,
const std::string& full_name, absl::string_view full_name,
const Message& proto) { const Message& proto) {
if (name.empty()) { if (name.empty()) {
AddError(full_name, proto, DescriptorPool::ErrorCollector::NAME, AddError(full_name, proto, DescriptorPool::ErrorCollector::NAME,
@ -5949,7 +5950,7 @@ FileDescriptor* DescriptorBuilder::BuildFileImpl(
"Exceeds Maximum Package Depth"); "Exceeds Maximum Package Depth");
return nullptr; return nullptr;
} }
AddPackage(result->package(), proto, result); AddPackage(result->package(), proto, result, true);
} }
// Make sure all dependencies are loaded. // Make sure all dependencies are loaded.
@ -6225,7 +6226,7 @@ FileDescriptor* DescriptorBuilder::BuildFileImpl(
const std::string* DescriptorBuilder::AllocateNameStrings( const std::string* DescriptorBuilder::AllocateNameStrings(
const std::string& scope, const std::string& proto_name, absl::string_view scope, absl::string_view proto_name,
internal::FlatAllocator& alloc) { internal::FlatAllocator& alloc) {
if (scope.empty()) { if (scope.empty()) {
return alloc.AllocateStrings(proto_name, proto_name); return alloc.AllocateStrings(proto_name, proto_name);
@ -6261,7 +6262,7 @@ void DescriptorBuilder::BuildMessage(const DescriptorProto& proto,
const Descriptor* parent, const Descriptor* parent,
Descriptor* result, Descriptor* result,
internal::FlatAllocator& alloc) { internal::FlatAllocator& alloc) {
const std::string& scope = absl::string_view scope =
(parent == nullptr) ? file_->package() : parent->full_name(); (parent == nullptr) ? file_->package() : parent->full_name();
result->all_names_ = AllocateNameStrings(scope, proto.name(), alloc); result->all_names_ = AllocateNameStrings(scope, proto.name(), alloc);
ValidateSymbolName(proto.name(), result->full_name(), proto); ValidateSymbolName(proto.name(), result->full_name(), proto);
@ -6344,7 +6345,7 @@ void DescriptorBuilder::BuildMessage(const DescriptorProto& proto,
} }
absl::flat_hash_set<absl::string_view> reserved_name_set; absl::flat_hash_set<absl::string_view> reserved_name_set;
for (const std::string& name : proto.reserved_name()) { for (absl::string_view name : proto.reserved_name()) {
if (!reserved_name_set.insert(name).second) { if (!reserved_name_set.insert(name).second) {
AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] { AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] {
return absl::Substitute("Field name \"$0\" is reserved multiple times.", return absl::Substitute("Field name \"$0\" is reserved multiple times.",
@ -6466,7 +6467,7 @@ bool JsonNameLooksLikeExtension(std::string name) {
} // namespace } // namespace
void DescriptorBuilder::CheckFieldJsonNameUniqueness( void DescriptorBuilder::CheckFieldJsonNameUniqueness(
const std::string& message_name, const DescriptorProto& message, absl::string_view message_name, const DescriptorProto& message,
const Descriptor* descriptor, bool use_custom_names) { const Descriptor* descriptor, bool use_custom_names) {
absl::flat_hash_map<std::string, JsonNameDetails> name_to_field; absl::flat_hash_map<std::string, JsonNameDetails> name_to_field;
for (const FieldDescriptorProto& field : message.field()) { for (const FieldDescriptorProto& field : message.field()) {
@ -6530,7 +6531,7 @@ void DescriptorBuilder::BuildFieldOrExtension(const FieldDescriptorProto& proto,
FieldDescriptor* result, FieldDescriptor* result,
bool is_extension, bool is_extension,
internal::FlatAllocator& alloc) { internal::FlatAllocator& alloc) {
const std::string& scope = absl::string_view scope =
(parent == nullptr) ? file_->package() : parent->full_name(); (parent == nullptr) ? file_->package() : parent->full_name();
// We allocate all names in a single array, and dedup them. // We allocate all names in a single array, and dedup them.
@ -6966,7 +6967,7 @@ void DescriptorBuilder::BuildEnum(const EnumDescriptorProto& proto,
const Descriptor* parent, const Descriptor* parent,
EnumDescriptor* result, EnumDescriptor* result,
internal::FlatAllocator& alloc) { internal::FlatAllocator& alloc) {
const std::string& scope = absl::string_view scope =
(parent == nullptr) ? file_->package() : parent->full_name(); (parent == nullptr) ? file_->package() : parent->full_name();
result->all_names_ = AllocateNameStrings(scope, proto.name(), alloc); result->all_names_ = AllocateNameStrings(scope, proto.name(), alloc);
@ -7036,7 +7037,7 @@ void DescriptorBuilder::BuildEnum(const EnumDescriptorProto& proto,
} }
absl::flat_hash_set<absl::string_view> reserved_name_set; absl::flat_hash_set<absl::string_view> reserved_name_set;
for (const std::string& name : proto.reserved_name()) { for (absl::string_view name : proto.reserved_name()) {
if (!reserved_name_set.insert(name).second) { if (!reserved_name_set.insert(name).second) {
AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] { AddError(name, proto, DescriptorPool::ErrorCollector::NAME, [&] {
return absl::Substitute("Enum value \"$0\" is reserved multiple times.", return absl::Substitute("Enum value \"$0\" is reserved multiple times.",
@ -7453,7 +7454,7 @@ void DescriptorBuilder::CrossLinkField(FieldDescriptor* field,
<< proto; << proto;
// Save the symbol names for later for lookup, and allocate the once // Save the symbol names for later for lookup, and allocate the once
// object needed for the accessors. // object needed for the accessors.
const std::string& name = proto.type_name(); absl::string_view name = proto.type_name();
int name_sizes = static_cast<int>(name.size() + 1 + int name_sizes = static_cast<int>(name.size() + 1 +
proto.default_value().size() + 1); proto.default_value().size() + 1);
@ -7463,7 +7464,8 @@ void DescriptorBuilder::CrossLinkField(FieldDescriptor* field,
absl::once_flag{}; absl::once_flag{};
char* names = reinterpret_cast<char*>(field->type_once_ + 1); char* names = reinterpret_cast<char*>(field->type_once_ + 1);
memcpy(names, name.c_str(), name.size() + 1); memcpy(names, name.data(), name.size());
names[name.size()] = 0;
memcpy(names + name.size() + 1, proto.default_value().c_str(), memcpy(names + name.size() + 1, proto.default_value().c_str(),
proto.default_value().size() + 1); proto.default_value().size() + 1);
@ -8284,7 +8286,7 @@ absl::optional<std::string> ValidateSymbolForDeclaration(
void DescriptorBuilder::ValidateExtensionDeclaration( void DescriptorBuilder::ValidateExtensionDeclaration(
const std::string& full_name, absl::string_view full_name,
const RepeatedPtrField<ExtensionRangeOptions_Declaration>& declarations, const RepeatedPtrField<ExtensionRangeOptions_Declaration>& declarations,
const DescriptorProto_ExtensionRange& proto, const DescriptorProto_ExtensionRange& proto,
absl::flat_hash_set<absl::string_view>& full_name_set) { absl::flat_hash_set<absl::string_view>& full_name_set) {
@ -8764,7 +8766,7 @@ bool DescriptorBuilder::OptionInterpreter::InterpretSingleOption(
for (int i = 0; i < uninterpreted_option_->name_size(); ++i) { for (int i = 0; i < uninterpreted_option_->name_size(); ++i) {
builder_->undefine_resolved_name_.clear(); builder_->undefine_resolved_name_.clear();
const std::string& name_part = uninterpreted_option_->name(i).name_part(); absl::string_view name_part = uninterpreted_option_->name(i).name_part();
if (!debug_msg_name.empty()) { if (!debug_msg_name.empty()) {
absl::StrAppend(&debug_msg_name, "."); absl::StrAppend(&debug_msg_name, ".");
} }
@ -9047,7 +9049,7 @@ bool DescriptorBuilder::OptionInterpreter::ExamineIfOptionIsSet(
std::vector<const FieldDescriptor*>::const_iterator std::vector<const FieldDescriptor*>::const_iterator
intermediate_fields_iter, intermediate_fields_iter,
std::vector<const FieldDescriptor*>::const_iterator intermediate_fields_end, std::vector<const FieldDescriptor*>::const_iterator intermediate_fields_end,
const FieldDescriptor* innermost_field, const std::string& debug_msg_name, const FieldDescriptor* innermost_field, absl::string_view debug_msg_name,
const UnknownFieldSet& unknown_fields) { const UnknownFieldSet& unknown_fields) {
// We do linear searches of the UnknownFieldSet and its sub-groups. This // We do linear searches of the UnknownFieldSet and its sub-groups. This
// should be fine since it's unlikely that any one options structure will // should be fine since it's unlikely that any one options structure will
@ -9294,7 +9296,7 @@ bool DescriptorBuilder::OptionInterpreter::SetOptionValue(
}); });
} }
const EnumDescriptor* enum_type = option_field->enum_type(); const EnumDescriptor* enum_type = option_field->enum_type();
const std::string& value_name = uninterpreted_option_->identifier_value(); absl::string_view value_name = uninterpreted_option_->identifier_value();
const EnumValueDescriptor* enum_value = nullptr; const EnumValueDescriptor* enum_value = nullptr;
if (enum_type->file()->pool() != DescriptorPool::generated_pool()) { if (enum_type->file()->pool() != DescriptorPool::generated_pool()) {
@ -9303,7 +9305,7 @@ bool DescriptorBuilder::OptionInterpreter::SetOptionValue(
std::string fully_qualified_name = enum_type->full_name(); std::string fully_qualified_name = enum_type->full_name();
fully_qualified_name.resize(fully_qualified_name.size() - fully_qualified_name.resize(fully_qualified_name.size() -
enum_type->name().size()); enum_type->name().size());
fully_qualified_name += value_name; absl::StrAppend(&fully_qualified_name, value_name);
// Search for the enum value's descriptor in the builder's pool. Note // Search for the enum value's descriptor in the builder's pool. Note
// that we use DescriptorBuilder::FindSymbolNotEnforcingDeps(), not // that we use DescriptorBuilder::FindSymbolNotEnforcingDeps(), not

@ -962,17 +962,17 @@ uint8_t* EpsCopyOutputStream::WriteCord(const absl::Cord& cord, uint8_t* ptr) {
} }
} }
uint8_t* EpsCopyOutputStream::WriteStringMaybeAliasedOutline(uint32_t num, uint8_t* EpsCopyOutputStream::WriteStringMaybeAliasedOutline(
const std::string& s, uint32_t num, absl::string_view s, uint8_t* ptr) {
uint8_t* ptr) {
ptr = EnsureSpace(ptr); ptr = EnsureSpace(ptr);
uint32_t size = s.size(); uint32_t size = s.size();
ptr = WriteLengthDelim(num, size, ptr); ptr = WriteLengthDelim(num, size, ptr);
return WriteRawMaybeAliased(s.data(), size, ptr); return WriteRawMaybeAliased(s.data(), size, ptr);
} }
uint8_t* EpsCopyOutputStream::WriteStringOutline(uint32_t num, const std::string& s, uint8_t* EpsCopyOutputStream::WriteStringOutline(uint32_t num,
uint8_t* ptr) { const std::string& s,
uint8_t* ptr) {
ptr = EnsureSpace(ptr); ptr = EnsureSpace(ptr);
uint32_t size = s.size(); uint32_t size = s.size();
ptr = WriteLengthDelim(num, size, ptr); ptr = WriteLengthDelim(num, size, ptr);
@ -1004,8 +1004,8 @@ uint8_t* CodedOutputStream::WriteCordToArray(const absl::Cord& cord,
} }
uint8_t* CodedOutputStream::WriteStringWithSizeToArray(const std::string& str, uint8_t* CodedOutputStream::WriteStringWithSizeToArray(absl::string_view str,
uint8_t* target) { uint8_t* target) {
ABSL_DCHECK_LE(str.size(), std::numeric_limits<uint32_t>::max()); ABSL_DCHECK_LE(str.size(), std::numeric_limits<uint32_t>::max());
target = WriteVarint32ToArray(str.size(), target); target = WriteVarint32ToArray(str.size(), target);
return WriteStringToArray(str, target); return WriteStringToArray(str, target);

@ -689,7 +689,7 @@ class PROTOBUF_EXPORT EpsCopyOutputStream {
#ifndef NDEBUG #ifndef NDEBUG
PROTOBUF_NOINLINE PROTOBUF_NOINLINE
#endif #endif
uint8_t* WriteStringMaybeAliased(uint32_t num, const std::string& s, uint8_t* WriteStringMaybeAliased(uint32_t num, absl::string_view s,
uint8_t* ptr) { uint8_t* ptr) {
std::ptrdiff_t size = s.size(); std::ptrdiff_t size = s.size();
if (PROTOBUF_PREDICT_FALSE( if (PROTOBUF_PREDICT_FALSE(
@ -701,7 +701,7 @@ class PROTOBUF_EXPORT EpsCopyOutputStream {
std::memcpy(ptr, s.data(), size); std::memcpy(ptr, s.data(), size);
return ptr + size; return ptr + size;
} }
uint8_t* WriteBytesMaybeAliased(uint32_t num, const std::string& s, uint8_t* WriteBytesMaybeAliased(uint32_t num, absl::string_view s,
uint8_t* ptr) { uint8_t* ptr) {
return WriteStringMaybeAliased(num, s, ptr); return WriteStringMaybeAliased(num, s, ptr);
} }
@ -860,7 +860,7 @@ class PROTOBUF_EXPORT EpsCopyOutputStream {
uint8_t* WriteAliasedRaw(const void* data, int size, uint8_t* ptr); uint8_t* WriteAliasedRaw(const void* data, int size, uint8_t* ptr);
uint8_t* WriteStringMaybeAliasedOutline(uint32_t num, const std::string& s, uint8_t* WriteStringMaybeAliasedOutline(uint32_t num, absl::string_view s,
uint8_t* ptr); uint8_t* ptr);
uint8_t* WriteStringOutline(uint32_t num, const std::string& s, uint8_t* ptr); uint8_t* WriteStringOutline(uint32_t num, const std::string& s, uint8_t* ptr);
uint8_t* WriteStringOutline(uint32_t num, absl::string_view s, uint8_t* ptr); uint8_t* WriteStringOutline(uint32_t num, absl::string_view s, uint8_t* ptr);
@ -1113,11 +1113,11 @@ class PROTOBUF_EXPORT CodedOutputStream {
uint8_t* target); uint8_t* target);
// Equivalent to WriteRaw(str.data(), str.size()). // Equivalent to WriteRaw(str.data(), str.size()).
void WriteString(const std::string& str); void WriteString(absl::string_view str);
// Like WriteString() but writing directly to the target array. // Like WriteString() but writing directly to the target array.
static uint8_t* WriteStringToArray(const std::string& str, uint8_t* target); static uint8_t* WriteStringToArray(absl::string_view str, uint8_t* target);
// Write the varint-encoded size of str followed by str. // Write the varint-encoded size of str followed by str.
static uint8_t* WriteStringWithSizeToArray(const std::string& str, static uint8_t* WriteStringWithSizeToArray(absl::string_view str,
uint8_t* target); uint8_t* target);
// Like WriteString() but writes a Cord. // Like WriteString() but writes a Cord.
@ -1778,7 +1778,7 @@ inline size_t CodedOutputStream::VarintSize32SignExtendedPlusOne(
} }
#undef PROTOBUF_CODED_STREAM_H_PREFER_BSR #undef PROTOBUF_CODED_STREAM_H_PREFER_BSR
inline void CodedOutputStream::WriteString(const std::string& str) { inline void CodedOutputStream::WriteString(absl::string_view str) {
WriteRaw(str.data(), static_cast<int>(str.size())); WriteRaw(str.data(), static_cast<int>(str.size()));
} }
@ -1793,7 +1793,7 @@ inline uint8_t* CodedOutputStream::WriteRawToArray(const void* data, int size,
return target + size; return target + size;
} }
inline uint8_t* CodedOutputStream::WriteStringToArray(const std::string& str, inline uint8_t* CodedOutputStream::WriteStringToArray(absl::string_view str,
uint8_t* target) { uint8_t* target) {
return WriteRawToArray(str.data(), static_cast<int>(str.size()), target); return WriteRawToArray(str.data(), static_cast<int>(str.size()), target);
} }

@ -296,7 +296,7 @@ struct Proto2Descriptor {
// Like WithFieldType, but using dynamic lookup by type URL. // Like WithFieldType, but using dynamic lookup by type URL.
template <typename F> template <typename F>
static absl::Status WithDynamicType(const Desc& desc, static absl::Status WithDynamicType(const Desc& desc,
const std::string& type_url, F body) { absl::string_view type_url, F body) {
size_t slash = type_url.rfind('/'); size_t slash = type_url.rfind('/');
if (slash == absl::string_view::npos || slash == 0) { if (slash == absl::string_view::npos || slash == 0) {
return absl::InvalidArgumentError(absl::StrCat( return absl::InvalidArgumentError(absl::StrCat(
@ -498,7 +498,7 @@ struct Proto3Type {
template <typename F> template <typename F>
static absl::Status WithDynamicType(const Desc& desc, static absl::Status WithDynamicType(const Desc& desc,
const std::string& type_url, F body) { absl::string_view type_url, F body) {
auto dyn_desc = desc.pool()->FindMessage(type_url); auto dyn_desc = desc.pool()->FindMessage(type_url);
RETURN_IF_ERROR(dyn_desc.status()); RETURN_IF_ERROR(dyn_desc.status());
return body(**dyn_desc); return body(**dyn_desc);

@ -1297,7 +1297,7 @@ absl::Status JsonStringToMessage(absl::string_view input, Message* message,
} }
absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver, absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* json_input, io::ZeroCopyInputStream* json_input,
io::ZeroCopyOutputStream* binary_output, io::ZeroCopyOutputStream* binary_output,
json_internal::ParseOptions options) { json_internal::ParseOptions options) {
@ -1360,3 +1360,5 @@ absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver,
} // namespace json_internal } // namespace json_internal
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#include "google/protobuf/port_undef.inc"

@ -25,7 +25,7 @@ absl::Status JsonStringToMessage(absl::string_view input, Message* message,
// Internal version of google::protobuf::util::JsonToBinaryStream; see json_util.h for // Internal version of google::protobuf::util::JsonToBinaryStream; see json_util.h for
// details. // details.
absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver, absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* json_input, io::ZeroCopyInputStream* json_input,
io::ZeroCopyOutputStream* binary_output, io::ZeroCopyOutputStream* binary_output,
json_internal::ParseOptions options); json_internal::ParseOptions options);

@ -117,7 +117,7 @@ struct ParseProto2Descriptor : Proto2Descriptor {
// //
// Body should have a signature `absl::Status(const Desc&, Msg&)`. // Body should have a signature `absl::Status(const Desc&, Msg&)`.
template <typename F> template <typename F>
static absl::Status NewDynamic(Field f, const std::string& type_url, Msg& msg, static absl::Status NewDynamic(Field f, absl::string_view type_url, Msg& msg,
F body) { F body) {
RecordAsSeen(f, msg); RecordAsSeen(f, msg);
return WithDynamicType( return WithDynamicType(
@ -263,7 +263,7 @@ struct ParseProto3Type : Proto3Type {
} }
template <typename F> template <typename F>
static absl::Status NewDynamic(Field f, const std::string& type_url, Msg& msg, static absl::Status NewDynamic(Field f, absl::string_view type_url, Msg& msg,
F body) { F body) {
RecordAsSeen(f, msg); RecordAsSeen(f, msg);
return WithDynamicType( return WithDynamicType(

@ -846,7 +846,7 @@ absl::Status MessageToJsonString(const Message& message, std::string* output,
} }
absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver, absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* binary_input, io::ZeroCopyInputStream* binary_input,
io::ZeroCopyOutputStream* json_output, io::ZeroCopyOutputStream* json_output,
json_internal::WriterOptions options) { json_internal::WriterOptions options) {
@ -905,3 +905,5 @@ absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
} // namespace json_internal } // namespace json_internal
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#include "google/protobuf/port_undef.inc"

@ -25,7 +25,7 @@ absl::Status MessageToJsonString(const Message& message, std::string* output,
// Internal version of google::protobuf::util::BinaryToJsonStream; see json_util.h for // Internal version of google::protobuf::util::BinaryToJsonStream; see json_util.h for
// details. // details.
absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver, absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* binary_input, io::ZeroCopyInputStream* binary_input,
io::ZeroCopyOutputStream* json_output, io::ZeroCopyOutputStream* json_output,
json_internal::WriterOptions options); json_internal::WriterOptions options);

@ -25,7 +25,7 @@ namespace protobuf {
namespace json { namespace json {
absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver, absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* binary_input, io::ZeroCopyInputStream* binary_input,
io::ZeroCopyOutputStream* json_output, io::ZeroCopyOutputStream* json_output,
const PrintOptions& options) { const PrintOptions& options) {
@ -45,8 +45,8 @@ absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
} }
absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* resolver, absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
const std::string& binary_input, absl::string_view binary_input,
std::string* json_output, std::string* json_output,
const PrintOptions& options) { const PrintOptions& options) {
io::ArrayInputStream input_stream(binary_input.data(), binary_input.size()); io::ArrayInputStream input_stream(binary_input.data(), binary_input.size());
@ -56,7 +56,7 @@ absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* resolver,
} }
absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver, absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* json_input, io::ZeroCopyInputStream* json_input,
io::ZeroCopyOutputStream* binary_output, io::ZeroCopyOutputStream* binary_output,
const ParseOptions& options) { const ParseOptions& options) {
@ -72,7 +72,7 @@ absl::Status JsonToBinaryStream(google::protobuf::util::TypeResolver* resolver,
} }
absl::Status JsonToBinaryString(google::protobuf::util::TypeResolver* resolver, absl::Status JsonToBinaryString(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
absl::string_view json_input, absl::string_view json_input,
std::string* binary_output, std::string* binary_output,
const ParseOptions& options) { const ParseOptions& options) {
@ -112,3 +112,5 @@ absl::Status JsonStringToMessage(absl::string_view input, Message* message,
} // namespace json } // namespace json
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#include "google/protobuf/port_undef.inc"

@ -94,12 +94,12 @@ inline absl::Status JsonStringToMessage(absl::string_view input,
// Please note that non-OK statuses are not a stable output of this API and // Please note that non-OK statuses are not a stable output of this API and
// subject to change without notice. // subject to change without notice.
PROTOBUF_EXPORT absl::Status BinaryToJsonStream( PROTOBUF_EXPORT absl::Status BinaryToJsonStream(
google::protobuf::util::TypeResolver* resolver, const std::string& type_url, google::protobuf::util::TypeResolver* resolver, absl::string_view type_url,
io::ZeroCopyInputStream* binary_input, io::ZeroCopyInputStream* binary_input,
io::ZeroCopyOutputStream* json_output, const PrintOptions& options); io::ZeroCopyOutputStream* json_output, const PrintOptions& options);
inline absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver, inline absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
io::ZeroCopyInputStream* binary_input, io::ZeroCopyInputStream* binary_input,
io::ZeroCopyOutputStream* json_output) { io::ZeroCopyOutputStream* json_output) {
return BinaryToJsonStream(resolver, type_url, binary_input, json_output, return BinaryToJsonStream(resolver, type_url, binary_input, json_output,
@ -107,13 +107,13 @@ inline absl::Status BinaryToJsonStream(google::protobuf::util::TypeResolver* res
} }
PROTOBUF_EXPORT absl::Status BinaryToJsonString( PROTOBUF_EXPORT absl::Status BinaryToJsonString(
google::protobuf::util::TypeResolver* resolver, const std::string& type_url, google::protobuf::util::TypeResolver* resolver, absl::string_view type_url,
const std::string& binary_input, std::string* json_output, absl::string_view binary_input, std::string* json_output,
const PrintOptions& options); const PrintOptions& options);
inline absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* resolver, inline absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
const std::string& binary_input, absl::string_view binary_input,
std::string* json_output) { std::string* json_output) {
return BinaryToJsonString(resolver, type_url, binary_input, json_output, return BinaryToJsonString(resolver, type_url, binary_input, json_output,
PrintOptions()); PrintOptions());
@ -128,12 +128,12 @@ inline absl::Status BinaryToJsonString(google::protobuf::util::TypeResolver* res
// Please note that non-OK statuses are not a stable output of this API and // Please note that non-OK statuses are not a stable output of this API and
// subject to change without notice. // subject to change without notice.
PROTOBUF_EXPORT absl::Status JsonToBinaryStream( PROTOBUF_EXPORT absl::Status JsonToBinaryStream(
google::protobuf::util::TypeResolver* resolver, const std::string& type_url, google::protobuf::util::TypeResolver* resolver, absl::string_view type_url,
io::ZeroCopyInputStream* json_input, io::ZeroCopyInputStream* json_input,
io::ZeroCopyOutputStream* binary_output, const ParseOptions& options); io::ZeroCopyOutputStream* binary_output, const ParseOptions& options);
inline absl::Status JsonToBinaryStream( inline absl::Status JsonToBinaryStream(
google::protobuf::util::TypeResolver* resolver, const std::string& type_url, google::protobuf::util::TypeResolver* resolver, absl::string_view type_url,
io::ZeroCopyInputStream* json_input, io::ZeroCopyInputStream* json_input,
io::ZeroCopyOutputStream* binary_output) { io::ZeroCopyOutputStream* binary_output) {
return JsonToBinaryStream(resolver, type_url, json_input, binary_output, return JsonToBinaryStream(resolver, type_url, json_input, binary_output,
@ -141,12 +141,12 @@ inline absl::Status JsonToBinaryStream(
} }
PROTOBUF_EXPORT absl::Status JsonToBinaryString( PROTOBUF_EXPORT absl::Status JsonToBinaryString(
google::protobuf::util::TypeResolver* resolver, const std::string& type_url, google::protobuf::util::TypeResolver* resolver, absl::string_view type_url,
absl::string_view json_input, std::string* binary_output, absl::string_view json_input, std::string* binary_output,
const ParseOptions& options); const ParseOptions& options);
inline absl::Status JsonToBinaryString(google::protobuf::util::TypeResolver* resolver, inline absl::Status JsonToBinaryString(google::protobuf::util::TypeResolver* resolver,
const std::string& type_url, absl::string_view type_url,
absl::string_view json_input, absl::string_view json_input,
std::string* binary_output) { std::string* binary_output) {
return JsonToBinaryString(resolver, type_url, json_input, binary_output, return JsonToBinaryString(resolver, type_url, json_input, binary_output,

@ -51,14 +51,14 @@ using google::protobuf::io::win32::mkdir;
using google::protobuf::io::win32::stat; using google::protobuf::io::win32::stat;
#endif #endif
bool File::Exists(const std::string& name) { bool File::Exists(absl::string_view name) {
return access(name.c_str(), F_OK) == 0; return access(std::string(name).c_str(), F_OK) == 0;
} }
absl::Status File::ReadFileToString(const std::string& name, absl::Status File::ReadFileToString(absl::string_view name, std::string* output,
std::string* output, bool text_mode) { bool text_mode) {
char buffer[1024]; char buffer[1024];
FILE* file = fopen(name.c_str(), text_mode ? "rt" : "rb"); FILE* file = fopen(std::string(name).c_str(), text_mode ? "rt" : "rb");
if (file == nullptr) return absl::NotFoundError("Could not open file"); if (file == nullptr) return absl::NotFoundError("Could not open file");
while (true) { while (true) {
@ -73,13 +73,13 @@ absl::Status File::ReadFileToString(const std::string& name,
return absl::OkStatus(); return absl::OkStatus();
} }
void File::ReadFileToStringOrDie(const std::string& name, std::string* output) { void File::ReadFileToStringOrDie(absl::string_view name, std::string* output) {
ABSL_CHECK_OK(ReadFileToString(name, output)) << "Could not read: " << name; ABSL_CHECK_OK(ReadFileToString(name, output)) << "Could not read: " << name;
} }
absl::Status File::WriteStringToFile(absl::string_view contents, absl::Status File::WriteStringToFile(absl::string_view contents,
const std::string& name) { absl::string_view name) {
FILE* file = fopen(name.c_str(), "wb"); FILE* file = fopen(std::string(name).c_str(), "wb");
if (file == nullptr) { if (file == nullptr) {
return absl::InternalError( return absl::InternalError(
absl::StrCat("fopen(", name, ", \"wb\"): ", strerror(errno))); absl::StrCat("fopen(", name, ", \"wb\"): ", strerror(errno)));
@ -98,8 +98,8 @@ absl::Status File::WriteStringToFile(absl::string_view contents,
} }
void File::WriteStringToFileOrDie(absl::string_view contents, void File::WriteStringToFileOrDie(absl::string_view contents,
const std::string& name) { absl::string_view name) {
FILE* file = fopen(name.c_str(), "wb"); FILE* file = fopen(std::string(name).c_str(), "wb");
ABSL_CHECK(file != nullptr) ABSL_CHECK(file != nullptr)
<< "fopen(" << name << ", \"wb\"): " << strerror(errno); << "fopen(" << name << ", \"wb\"): " << strerror(errno);
ABSL_CHECK_EQ(fwrite(contents.data(), 1, contents.size(), file), ABSL_CHECK_EQ(fwrite(contents.data(), 1, contents.size(), file),
@ -109,17 +109,17 @@ void File::WriteStringToFileOrDie(absl::string_view contents,
<< "fclose(" << name << "): " << strerror(errno); << "fclose(" << name << "): " << strerror(errno);
} }
absl::Status File::CreateDir(const std::string& name, int mode) { absl::Status File::CreateDir(absl::string_view name, int mode) {
if (!name.empty()) { if (!name.empty()) {
ABSL_CHECK(name[name.size() - 1] != '.'); ABSL_CHECK(name[name.size() - 1] != '.');
} }
if (mkdir(name.c_str(), mode) != 0) { if (mkdir(std::string(name).c_str(), mode) != 0) {
return absl::InternalError("Failed to create directory"); return absl::InternalError("Failed to create directory");
} }
return absl::OkStatus(); return absl::OkStatus();
} }
absl::Status File::RecursivelyCreateDir(const std::string& path, int mode) { absl::Status File::RecursivelyCreateDir(absl::string_view path, int mode) {
if (CreateDir(path, mode).ok()) return absl::OkStatus(); if (CreateDir(path, mode).ok()) return absl::OkStatus();
if (Exists(path)) return absl::AlreadyExistsError("Path already exists"); if (Exists(path)) return absl::AlreadyExistsError("Path already exists");
@ -134,9 +134,11 @@ absl::Status File::RecursivelyCreateDir(const std::string& path, int mode) {
return CreateDir(path, mode); return CreateDir(path, mode);
} }
void File::DeleteRecursively(const std::string& name, void* dummy1, void File::DeleteRecursively(absl::string_view name_view, void* dummy1,
void* dummy2) { void* dummy2) {
if (name.empty()) return; if (name_view.empty()) return;
std::string name(name_view);
// We don't care too much about error checking here since this is only used // We don't care too much about error checking here since this is only used
// in tests to delete temporary directories that are under /tmp anyway. // in tests to delete temporary directories that are under /tmp anyway.
@ -148,7 +150,7 @@ void File::DeleteRecursively(const std::string& name, void* dummy1,
FindFirstFileA(absl::StrCat(name, "/*").c_str(), &find_data); FindFirstFileA(absl::StrCat(name, "/*").c_str(), &find_data);
if (find_handle == INVALID_HANDLE_VALUE) { if (find_handle == INVALID_HANDLE_VALUE) {
// Just delete it, whatever it is. // Just delete it, whatever it is.
DeleteFileA(name.c_str()); DeleteFileA(std::string(name).c_str());
RemoveDirectoryA(name.c_str()); RemoveDirectoryA(name.c_str());
return; return;
} }
@ -196,8 +198,8 @@ void File::DeleteRecursively(const std::string& name, void* dummy1,
#endif #endif
} }
bool File::ChangeWorkingDirectory(const std::string& new_working_directory) { bool File::ChangeWorkingDirectory(absl::string_view new_working_directory) {
return chdir(new_working_directory.c_str()) == 0; return chdir(std::string(new_working_directory).c_str()) == 0;
} }
} // namespace protobuf } // namespace protobuf

@ -28,55 +28,55 @@ class File {
File& operator=(const File&) = delete; File& operator=(const File&) = delete;
// Check if the file exists. // Check if the file exists.
static bool Exists(const std::string& name); static bool Exists(absl::string_view name);
// Read an entire file to a string. Return true if successful, false // Read an entire file to a string. Return true if successful, false
// otherwise. // otherwise.
static absl::Status ReadFileToString(const std::string& name, static absl::Status ReadFileToString(absl::string_view name,
std::string* output, std::string* output,
bool text_mode = false); bool text_mode = false);
// Same as above, but crash on failure. // Same as above, but crash on failure.
static void ReadFileToStringOrDie(const std::string& name, static void ReadFileToStringOrDie(absl::string_view name,
std::string* output); std::string* output);
// Create a file and write a string to it. // Create a file and write a string to it.
static absl::Status WriteStringToFile(absl::string_view contents, static absl::Status WriteStringToFile(absl::string_view contents,
const std::string& name); absl::string_view name);
// Same as above, but crash on failure. // Same as above, but crash on failure.
static void WriteStringToFileOrDie(absl::string_view contents, static void WriteStringToFileOrDie(absl::string_view contents,
const std::string& name); absl::string_view name);
// Create a directory. // Create a directory.
static absl::Status CreateDir(const std::string& name, int mode); static absl::Status CreateDir(absl::string_view name, int mode);
// Create a directory and all parent directories if necessary. // Create a directory and all parent directories if necessary.
static absl::Status RecursivelyCreateDir(const std::string& path, int mode); static absl::Status RecursivelyCreateDir(absl::string_view path, int mode);
// If "name" is a file, we delete it. If it is a directory, we // If "name" is a file, we delete it. If it is a directory, we
// call DeleteRecursively() for each file or directory (other than // call DeleteRecursively() for each file or directory (other than
// dot and double-dot) within it, and then delete the directory itself. // dot and double-dot) within it, and then delete the directory itself.
// The "dummy" parameters have a meaning in the original version of this // The "dummy" parameters have a meaning in the original version of this
// method but they are not used anywhere in protocol buffers. // method but they are not used anywhere in protocol buffers.
static void DeleteRecursively(const std::string& name, void* dummy1, static void DeleteRecursively(absl::string_view name, void* dummy1,
void* dummy2); void* dummy2);
// Change working directory to given directory. // Change working directory to given directory.
static bool ChangeWorkingDirectory(const std::string& new_working_directory); static bool ChangeWorkingDirectory(absl::string_view new_working_directory);
static absl::Status GetContents(const std::string& name, std::string* output, static absl::Status GetContents(absl::string_view name, std::string* output,
bool /*is_default*/) { bool /*is_default*/) {
return ReadFileToString(name, output); return ReadFileToString(name, output);
} }
static absl::Status GetContentsAsText(const std::string& name, static absl::Status GetContentsAsText(absl::string_view name,
std::string* output, std::string* output,
bool /*is_default*/) { bool /*is_default*/) {
return ReadFileToString(name, output, true); return ReadFileToString(name, output, true);
} }
static absl::Status SetContents(const std::string& name, static absl::Status SetContents(absl::string_view name,
absl::string_view contents, absl::string_view contents,
bool /*is_default*/) { bool /*is_default*/) {
return WriteStringToFile(contents, name); return WriteStringToFile(contents, name);

@ -2234,8 +2234,8 @@ void MessageDifferencer::StreamReporter::PrintUnknownFieldValue(
printer_->PrintRaw(output); printer_->PrintRaw(output);
} }
void MessageDifferencer::StreamReporter::Print(const std::string& str) { void MessageDifferencer::StreamReporter::Print(absl::string_view str) {
printer_->Print(str.c_str()); printer_->Print(str);
} }
void MessageDifferencer::StreamReporter::PrintMapKey( void MessageDifferencer::StreamReporter::PrintMapKey(
@ -2382,3 +2382,5 @@ MessageDifferencer::CreateMultipleFieldsMapKeyComparator(
} // namespace util } // namespace util
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#include "google/protobuf/port_undef.inc"

@ -30,6 +30,7 @@
#include "absl/container/flat_hash_map.h" #include "absl/container/flat_hash_map.h"
#include "absl/container/flat_hash_set.h" #include "absl/container/flat_hash_set.h"
#include "absl/log/absl_check.h" #include "absl/log/absl_check.h"
#include "absl/strings/string_view.h"
#include "google/protobuf/descriptor.h" // FieldDescriptor #include "google/protobuf/descriptor.h" // FieldDescriptor
#include "google/protobuf/message.h" // Message #include "google/protobuf/message.h" // Message
#include "google/protobuf/text_format.h" #include "google/protobuf/text_format.h"
@ -728,7 +729,7 @@ class PROTOBUF_EXPORT MessageDifferencer {
virtual void PrintUnknownFieldValue(const UnknownField* unknown_field); virtual void PrintUnknownFieldValue(const UnknownField* unknown_field);
// Just print a string // Just print a string
void Print(const std::string& str); void Print(absl::string_view str);
private: private:
// helper function for PrintPath that contains logic for printing maps // helper function for PrintPath that contains logic for printing maps

Loading…
Cancel
Save