1 // Protocol Buffers - Google's data interchange format 2 // Copyright 2008 Google Inc. All rights reserved. 3 // https://developers.google.com/protocol-buffers/ 4 // 5 // Redistribution and use in source and binary forms, with or without 6 // modification, are permitted provided that the following conditions are 7 // met: 8 // 9 // * Redistributions of source code must retain the above copyright 10 // notice, this list of conditions and the following disclaimer. 11 // * Redistributions in binary form must reproduce the above 12 // copyright notice, this list of conditions and the following disclaimer 13 // in the documentation and/or other materials provided with the 14 // distribution. 15 // * Neither the name of Google Inc. nor the names of its 16 // contributors may be used to endorse or promote products derived from 17 // this software without specific prior written permission. 18 // 19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 30 31 // Author: jschorr@google.com (Joseph Schorr) 32 // Based on original Protocol Buffers design by 33 // Sanjay Ghemawat, Jeff Dean, and others. 34 35 #include <algorithm> 36 #include <float.h> 37 #include <math.h> 38 #include <stdio.h> 39 #include <stack> 40 #include <limits> 41 #include <vector> 42 43 #include <google/protobuf/text_format.h> 44 45 #include <google/protobuf/descriptor.h> 46 #include <google/protobuf/dynamic_message.h> 47 #include <google/protobuf/repeated_field.h> 48 #include <google/protobuf/wire_format_lite.h> 49 #include <google/protobuf/io/strtod.h> 50 #include <google/protobuf/io/coded_stream.h> 51 #include <google/protobuf/io/zero_copy_stream.h> 52 #include <google/protobuf/io/zero_copy_stream_impl.h> 53 #include <google/protobuf/unknown_field_set.h> 54 #include <google/protobuf/descriptor.pb.h> 55 #include <google/protobuf/io/tokenizer.h> 56 #include <google/protobuf/any.h> 57 #include <google/protobuf/stubs/stringprintf.h> 58 #include <google/protobuf/stubs/strutil.h> 59 #include <google/protobuf/stubs/map_util.h> 60 #include <google/protobuf/stubs/stl_util.h> 61 62 namespace google { 63 namespace protobuf { 64 65 namespace { 66 IsHexNumber(const string & str)67 inline bool IsHexNumber(const string& str) { 68 return (str.length() >= 2 && str[0] == '0' && 69 (str[1] == 'x' || str[1] == 'X')); 70 } 71 IsOctNumber(const string & str)72 inline bool IsOctNumber(const string& str) { 73 return (str.length() >= 2 && str[0] == '0' && 74 (str[1] >= '0' && str[1] < '8')); 75 } 76 GetAnyFieldDescriptors(const Message & message,const FieldDescriptor ** type_url_field,const FieldDescriptor ** value_field)77 inline bool GetAnyFieldDescriptors(const Message& message, 78 const FieldDescriptor** type_url_field, 79 const FieldDescriptor** value_field) { 80 const Descriptor* descriptor = message.GetDescriptor(); 81 *type_url_field = descriptor->FindFieldByNumber(1); 82 *value_field = descriptor->FindFieldByNumber(2); 83 return (*type_url_field != NULL && 84 (*type_url_field)->type() == FieldDescriptor::TYPE_STRING && 85 *value_field != NULL && 86 (*value_field)->type() == FieldDescriptor::TYPE_BYTES); 87 } 88 89 } // namespace 90 DebugString() const91 string Message::DebugString() const { 92 string debug_string; 93 94 TextFormat::Printer printer; 95 printer.SetExpandAny(true); 96 97 printer.PrintToString(*this, &debug_string); 98 99 return debug_string; 100 } 101 ShortDebugString() const102 string Message::ShortDebugString() const { 103 string debug_string; 104 105 TextFormat::Printer printer; 106 printer.SetSingleLineMode(true); 107 printer.SetExpandAny(true); 108 109 printer.PrintToString(*this, &debug_string); 110 // Single line mode currently might have an extra space at the end. 111 if (debug_string.size() > 0 && 112 debug_string[debug_string.size() - 1] == ' ') { 113 debug_string.resize(debug_string.size() - 1); 114 } 115 116 return debug_string; 117 } 118 Utf8DebugString() const119 string Message::Utf8DebugString() const { 120 string debug_string; 121 122 TextFormat::Printer printer; 123 printer.SetUseUtf8StringEscaping(true); 124 printer.SetExpandAny(true); 125 126 printer.PrintToString(*this, &debug_string); 127 128 return debug_string; 129 } 130 PrintDebugString() const131 void Message::PrintDebugString() const { 132 printf("%s", DebugString().c_str()); 133 } 134 135 136 // =========================================================================== 137 // Implementation of the parse information tree class. ParseInfoTree()138 TextFormat::ParseInfoTree::ParseInfoTree() { } 139 ~ParseInfoTree()140 TextFormat::ParseInfoTree::~ParseInfoTree() { 141 // Remove any nested information trees, as they are owned by this tree. 142 for (NestedMap::iterator it = nested_.begin(); it != nested_.end(); ++it) { 143 STLDeleteElements(&(it->second)); 144 } 145 } 146 RecordLocation(const FieldDescriptor * field,TextFormat::ParseLocation location)147 void TextFormat::ParseInfoTree::RecordLocation( 148 const FieldDescriptor* field, 149 TextFormat::ParseLocation location) { 150 locations_[field].push_back(location); 151 } 152 CreateNested(const FieldDescriptor * field)153 TextFormat::ParseInfoTree* TextFormat::ParseInfoTree::CreateNested( 154 const FieldDescriptor* field) { 155 // Owned by us in the map. 156 TextFormat::ParseInfoTree* instance = new TextFormat::ParseInfoTree(); 157 vector<TextFormat::ParseInfoTree*>* trees = &nested_[field]; 158 GOOGLE_CHECK(trees); 159 trees->push_back(instance); 160 return instance; 161 } 162 CheckFieldIndex(const FieldDescriptor * field,int index)163 void CheckFieldIndex(const FieldDescriptor* field, int index) { 164 if (field == NULL) { return; } 165 166 if (field->is_repeated() && index == -1) { 167 GOOGLE_LOG(DFATAL) << "Index must be in range of repeated field values. " 168 << "Field: " << field->name(); 169 } else if (!field->is_repeated() && index != -1) { 170 GOOGLE_LOG(DFATAL) << "Index must be -1 for singular fields." 171 << "Field: " << field->name(); 172 } 173 } 174 GetLocation(const FieldDescriptor * field,int index) const175 TextFormat::ParseLocation TextFormat::ParseInfoTree::GetLocation( 176 const FieldDescriptor* field, int index) const { 177 CheckFieldIndex(field, index); 178 if (index == -1) { index = 0; } 179 180 const vector<TextFormat::ParseLocation>* locations = 181 FindOrNull(locations_, field); 182 if (locations == NULL || index >= locations->size()) { 183 return TextFormat::ParseLocation(); 184 } 185 186 return (*locations)[index]; 187 } 188 GetTreeForNested(const FieldDescriptor * field,int index) const189 TextFormat::ParseInfoTree* TextFormat::ParseInfoTree::GetTreeForNested( 190 const FieldDescriptor* field, int index) const { 191 CheckFieldIndex(field, index); 192 if (index == -1) { index = 0; } 193 194 const vector<TextFormat::ParseInfoTree*>* trees = FindOrNull(nested_, field); 195 if (trees == NULL || index >= trees->size()) { 196 return NULL; 197 } 198 199 return (*trees)[index]; 200 } 201 202 203 // =========================================================================== 204 // Internal class for parsing an ASCII representation of a Protocol Message. 205 // This class makes use of the Protocol Message compiler's tokenizer found 206 // in //google/protobuf/io/tokenizer.h. Note that class's Parse 207 // method is *not* thread-safe and should only be used in a single thread at 208 // a time. 209 210 // Makes code slightly more readable. The meaning of "DO(foo)" is 211 // "Execute foo and fail if it fails.", where failure is indicated by 212 // returning false. Borrowed from parser.cc (Thanks Kenton!). 213 #define DO(STATEMENT) if (STATEMENT) {} else return false 214 215 class TextFormat::Parser::ParserImpl { 216 public: 217 218 // Determines if repeated values for non-repeated fields and 219 // oneofs are permitted, e.g., the string "foo: 1 foo: 2" for a 220 // required/optional field named "foo", or "baz: 1 qux: 2" 221 // where "baz" and "qux" are members of the same oneof. 222 enum SingularOverwritePolicy { 223 ALLOW_SINGULAR_OVERWRITES = 0, // the last value is retained 224 FORBID_SINGULAR_OVERWRITES = 1, // an error is issued 225 }; 226 ParserImpl(const Descriptor * root_message_type,io::ZeroCopyInputStream * input_stream,io::ErrorCollector * error_collector,TextFormat::Finder * finder,ParseInfoTree * parse_info_tree,SingularOverwritePolicy singular_overwrite_policy,bool allow_case_insensitive_field,bool allow_unknown_field,bool allow_unknown_enum,bool allow_field_number,bool allow_relaxed_whitespace)227 ParserImpl(const Descriptor* root_message_type, 228 io::ZeroCopyInputStream* input_stream, 229 io::ErrorCollector* error_collector, 230 TextFormat::Finder* finder, 231 ParseInfoTree* parse_info_tree, 232 SingularOverwritePolicy singular_overwrite_policy, 233 bool allow_case_insensitive_field, 234 bool allow_unknown_field, 235 bool allow_unknown_enum, 236 bool allow_field_number, 237 bool allow_relaxed_whitespace) 238 : error_collector_(error_collector), 239 finder_(finder), 240 parse_info_tree_(parse_info_tree), 241 tokenizer_error_collector_(this), 242 tokenizer_(input_stream, &tokenizer_error_collector_), 243 root_message_type_(root_message_type), 244 singular_overwrite_policy_(singular_overwrite_policy), 245 allow_case_insensitive_field_(allow_case_insensitive_field), 246 allow_unknown_field_(allow_unknown_field), 247 allow_unknown_enum_(allow_unknown_enum), 248 allow_field_number_(allow_field_number), 249 had_errors_(false) { 250 // For backwards-compatibility with proto1, we need to allow the 'f' suffix 251 // for floats. 252 tokenizer_.set_allow_f_after_float(true); 253 254 // '#' starts a comment. 255 tokenizer_.set_comment_style(io::Tokenizer::SH_COMMENT_STYLE); 256 257 if (allow_relaxed_whitespace) { 258 tokenizer_.set_require_space_after_number(false); 259 tokenizer_.set_allow_multiline_strings(true); 260 } 261 262 // Consume the starting token. 263 tokenizer_.Next(); 264 } ~ParserImpl()265 ~ParserImpl() { } 266 267 // Parses the ASCII representation specified in input and saves the 268 // information into the output pointer (a Message). Returns 269 // false if an error occurs (an error will also be logged to 270 // GOOGLE_LOG(ERROR)). Parse(Message * output)271 bool Parse(Message* output) { 272 // Consume fields until we cannot do so anymore. 273 while (true) { 274 if (LookingAtType(io::Tokenizer::TYPE_END)) { 275 return !had_errors_; 276 } 277 278 DO(ConsumeField(output)); 279 } 280 } 281 ParseField(const FieldDescriptor * field,Message * output)282 bool ParseField(const FieldDescriptor* field, Message* output) { 283 bool suc; 284 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) { 285 suc = ConsumeFieldMessage(output, output->GetReflection(), field); 286 } else { 287 suc = ConsumeFieldValue(output, output->GetReflection(), field); 288 } 289 return suc && LookingAtType(io::Tokenizer::TYPE_END); 290 } 291 ReportError(int line,int col,const string & message)292 void ReportError(int line, int col, const string& message) { 293 had_errors_ = true; 294 if (error_collector_ == NULL) { 295 if (line >= 0) { 296 GOOGLE_LOG(ERROR) << "Error parsing text-format " 297 << root_message_type_->full_name() 298 << ": " << (line + 1) << ":" 299 << (col + 1) << ": " << message; 300 } else { 301 GOOGLE_LOG(ERROR) << "Error parsing text-format " 302 << root_message_type_->full_name() 303 << ": " << message; 304 } 305 } else { 306 error_collector_->AddError(line, col, message); 307 } 308 } 309 ReportWarning(int line,int col,const string & message)310 void ReportWarning(int line, int col, const string& message) { 311 if (error_collector_ == NULL) { 312 if (line >= 0) { 313 GOOGLE_LOG(WARNING) << "Warning parsing text-format " 314 << root_message_type_->full_name() 315 << ": " << (line + 1) << ":" 316 << (col + 1) << ": " << message; 317 } else { 318 GOOGLE_LOG(WARNING) << "Warning parsing text-format " 319 << root_message_type_->full_name() 320 << ": " << message; 321 } 322 } else { 323 error_collector_->AddWarning(line, col, message); 324 } 325 } 326 327 private: 328 GOOGLE_DISALLOW_EVIL_CONSTRUCTORS(ParserImpl); 329 330 // Reports an error with the given message with information indicating 331 // the position (as derived from the current token). ReportError(const string & message)332 void ReportError(const string& message) { 333 ReportError(tokenizer_.current().line, tokenizer_.current().column, 334 message); 335 } 336 337 // Reports a warning with the given message with information indicating 338 // the position (as derived from the current token). ReportWarning(const string & message)339 void ReportWarning(const string& message) { 340 ReportWarning(tokenizer_.current().line, tokenizer_.current().column, 341 message); 342 } 343 344 // Consumes the specified message with the given starting delimiter. 345 // This method checks to see that the end delimiter at the conclusion of 346 // the consumption matches the starting delimiter passed in here. ConsumeMessage(Message * message,const string delimiter)347 bool ConsumeMessage(Message* message, const string delimiter) { 348 while (!LookingAt(">") && !LookingAt("}")) { 349 DO(ConsumeField(message)); 350 } 351 352 // Confirm that we have a valid ending delimiter. 353 DO(Consume(delimiter)); 354 return true; 355 } 356 357 // Consume either "<" or "{". ConsumeMessageDelimiter(string * delimiter)358 bool ConsumeMessageDelimiter(string* delimiter) { 359 if (TryConsume("<")) { 360 *delimiter = ">"; 361 } else { 362 DO(Consume("{")); 363 *delimiter = "}"; 364 } 365 return true; 366 } 367 368 369 // Consumes the current field (as returned by the tokenizer) on the 370 // passed in message. ConsumeField(Message * message)371 bool ConsumeField(Message* message) { 372 const Reflection* reflection = message->GetReflection(); 373 const Descriptor* descriptor = message->GetDescriptor(); 374 375 string field_name; 376 377 const FieldDescriptor* field = NULL; 378 int start_line = tokenizer_.current().line; 379 int start_column = tokenizer_.current().column; 380 381 const FieldDescriptor* any_type_url_field; 382 const FieldDescriptor* any_value_field; 383 if (internal::GetAnyFieldDescriptors(*message, &any_type_url_field, 384 &any_value_field) && 385 TryConsume("[")) { 386 string full_type_name, prefix; 387 DO(ConsumeAnyTypeUrl(&full_type_name, &prefix)); 388 DO(Consume("]")); 389 TryConsume(":"); // ':' is optional between message labels and values. 390 string serialized_value; 391 DO(ConsumeAnyValue(full_type_name, 392 message->GetDescriptor()->file()->pool(), 393 &serialized_value)); 394 reflection->SetString( 395 message, any_type_url_field, 396 string(prefix + full_type_name)); 397 reflection->SetString(message, any_value_field, serialized_value); 398 return true; 399 } 400 if (TryConsume("[")) { 401 // Extension. 402 DO(ConsumeFullTypeName(&field_name)); 403 DO(Consume("]")); 404 405 field = (finder_ != NULL 406 ? finder_->FindExtension(message, field_name) 407 : reflection->FindKnownExtensionByName(field_name)); 408 409 if (field == NULL) { 410 if (!allow_unknown_field_) { 411 ReportError("Extension \"" + field_name + "\" is not defined or " 412 "is not an extension of \"" + 413 descriptor->full_name() + "\"."); 414 return false; 415 } else { 416 ReportWarning("Extension \"" + field_name + "\" is not defined or " 417 "is not an extension of \"" + 418 descriptor->full_name() + "\"."); 419 } 420 } 421 } else { 422 DO(ConsumeIdentifier(&field_name)); 423 424 int32 field_number; 425 if (allow_field_number_ && safe_strto32(field_name, &field_number)) { 426 if (descriptor->IsExtensionNumber(field_number)) { 427 field = reflection->FindKnownExtensionByNumber(field_number); 428 } else { 429 field = descriptor->FindFieldByNumber(field_number); 430 } 431 } else { 432 field = descriptor->FindFieldByName(field_name); 433 // Group names are expected to be capitalized as they appear in the 434 // .proto file, which actually matches their type names, not their 435 // field names. 436 if (field == NULL) { 437 string lower_field_name = field_name; 438 LowerString(&lower_field_name); 439 field = descriptor->FindFieldByName(lower_field_name); 440 // If the case-insensitive match worked but the field is NOT a group, 441 if (field != NULL && field->type() != FieldDescriptor::TYPE_GROUP) { 442 field = NULL; 443 } 444 } 445 // Again, special-case group names as described above. 446 if (field != NULL && field->type() == FieldDescriptor::TYPE_GROUP 447 && field->message_type()->name() != field_name) { 448 field = NULL; 449 } 450 451 if (field == NULL && allow_case_insensitive_field_) { 452 string lower_field_name = field_name; 453 LowerString(&lower_field_name); 454 field = descriptor->FindFieldByLowercaseName(lower_field_name); 455 } 456 } 457 458 if (field == NULL) { 459 if (!allow_unknown_field_) { 460 ReportError("Message type \"" + descriptor->full_name() + 461 "\" has no field named \"" + field_name + "\"."); 462 return false; 463 } else { 464 ReportWarning("Message type \"" + descriptor->full_name() + 465 "\" has no field named \"" + field_name + "\"."); 466 } 467 } 468 } 469 470 // Skips unknown field. 471 if (field == NULL) { 472 GOOGLE_CHECK(allow_unknown_field_); 473 // Try to guess the type of this field. 474 // If this field is not a message, there should be a ":" between the 475 // field name and the field value and also the field value should not 476 // start with "{" or "<" which indicates the beginning of a message body. 477 // If there is no ":" or there is a "{" or "<" after ":", this field has 478 // to be a message or the input is ill-formed. 479 if (TryConsume(":") && !LookingAt("{") && !LookingAt("<")) { 480 return SkipFieldValue(); 481 } else { 482 return SkipFieldMessage(); 483 } 484 } 485 486 if (singular_overwrite_policy_ == FORBID_SINGULAR_OVERWRITES) { 487 // Fail if the field is not repeated and it has already been specified. 488 if (!field->is_repeated() && reflection->HasField(*message, field)) { 489 ReportError("Non-repeated field \"" + field_name + 490 "\" is specified multiple times."); 491 return false; 492 } 493 // Fail if the field is a member of a oneof and another member has already 494 // been specified. 495 const OneofDescriptor* oneof = field->containing_oneof(); 496 if (oneof != NULL && reflection->HasOneof(*message, oneof)) { 497 const FieldDescriptor* other_field = 498 reflection->GetOneofFieldDescriptor(*message, oneof); 499 ReportError("Field \"" + field_name + "\" is specified along with " 500 "field \"" + other_field->name() + "\", another member " 501 "of oneof \"" + oneof->name() + "\"."); 502 return false; 503 } 504 } 505 506 // Perform special handling for embedded message types. 507 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) { 508 // ':' is optional here. 509 TryConsume(":"); 510 } else { 511 // ':' is required here. 512 DO(Consume(":")); 513 } 514 515 if (field->is_repeated() && TryConsume("[")) { 516 // Short repeated format, e.g. "foo: [1, 2, 3]" 517 while (true) { 518 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) { 519 // Perform special handling for embedded message types. 520 DO(ConsumeFieldMessage(message, reflection, field)); 521 } else { 522 DO(ConsumeFieldValue(message, reflection, field)); 523 } 524 if (TryConsume("]")) { 525 break; 526 } 527 DO(Consume(",")); 528 } 529 } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) { 530 DO(ConsumeFieldMessage(message, reflection, field)); 531 } else { 532 DO(ConsumeFieldValue(message, reflection, field)); 533 } 534 535 // For historical reasons, fields may optionally be separated by commas or 536 // semicolons. 537 TryConsume(";") || TryConsume(","); 538 539 if (field->options().deprecated()) { 540 ReportWarning("text format contains deprecated field \"" 541 + field_name + "\""); 542 } 543 544 // If a parse info tree exists, add the location for the parsed 545 // field. 546 if (parse_info_tree_ != NULL) { 547 RecordLocation(parse_info_tree_, field, 548 ParseLocation(start_line, start_column)); 549 } 550 551 return true; 552 } 553 554 // Skips the next field including the field's name and value. SkipField()555 bool SkipField() { 556 string field_name; 557 if (TryConsume("[")) { 558 // Extension name. 559 DO(ConsumeFullTypeName(&field_name)); 560 DO(Consume("]")); 561 } else { 562 DO(ConsumeIdentifier(&field_name)); 563 } 564 565 // Try to guess the type of this field. 566 // If this field is not a message, there should be a ":" between the 567 // field name and the field value and also the field value should not 568 // start with "{" or "<" which indicates the beginning of a message body. 569 // If there is no ":" or there is a "{" or "<" after ":", this field has 570 // to be a message or the input is ill-formed. 571 if (TryConsume(":") && !LookingAt("{") && !LookingAt("<")) { 572 DO(SkipFieldValue()); 573 } else { 574 DO(SkipFieldMessage()); 575 } 576 // For historical reasons, fields may optionally be separated by commas or 577 // semicolons. 578 TryConsume(";") || TryConsume(","); 579 return true; 580 } 581 ConsumeFieldMessage(Message * message,const Reflection * reflection,const FieldDescriptor * field)582 bool ConsumeFieldMessage(Message* message, 583 const Reflection* reflection, 584 const FieldDescriptor* field) { 585 586 // If the parse information tree is not NULL, create a nested one 587 // for the nested message. 588 ParseInfoTree* parent = parse_info_tree_; 589 if (parent != NULL) { 590 parse_info_tree_ = CreateNested(parent, field); 591 } 592 593 string delimiter; 594 DO(ConsumeMessageDelimiter(&delimiter)); 595 if (field->is_repeated()) { 596 DO(ConsumeMessage(reflection->AddMessage(message, field), delimiter)); 597 } else { 598 DO(ConsumeMessage(reflection->MutableMessage(message, field), 599 delimiter)); 600 } 601 602 // Reset the parse information tree. 603 parse_info_tree_ = parent; 604 return true; 605 } 606 607 // Skips the whole body of a message including the beginning delimiter and 608 // the ending delimiter. SkipFieldMessage()609 bool SkipFieldMessage() { 610 string delimiter; 611 DO(ConsumeMessageDelimiter(&delimiter)); 612 while (!LookingAt(">") && !LookingAt("}")) { 613 DO(SkipField()); 614 } 615 DO(Consume(delimiter)); 616 return true; 617 } 618 ConsumeFieldValue(Message * message,const Reflection * reflection,const FieldDescriptor * field)619 bool ConsumeFieldValue(Message* message, 620 const Reflection* reflection, 621 const FieldDescriptor* field) { 622 623 // Define an easy to use macro for setting fields. This macro checks 624 // to see if the field is repeated (in which case we need to use the Add 625 // methods or not (in which case we need to use the Set methods). 626 #define SET_FIELD(CPPTYPE, VALUE) \ 627 if (field->is_repeated()) { \ 628 reflection->Add##CPPTYPE(message, field, VALUE); \ 629 } else { \ 630 reflection->Set##CPPTYPE(message, field, VALUE); \ 631 } \ 632 633 switch(field->cpp_type()) { 634 case FieldDescriptor::CPPTYPE_INT32: { 635 int64 value; 636 DO(ConsumeSignedInteger(&value, kint32max)); 637 SET_FIELD(Int32, static_cast<int32>(value)); 638 break; 639 } 640 641 case FieldDescriptor::CPPTYPE_UINT32: { 642 uint64 value; 643 DO(ConsumeUnsignedInteger(&value, kuint32max)); 644 SET_FIELD(UInt32, static_cast<uint32>(value)); 645 break; 646 } 647 648 case FieldDescriptor::CPPTYPE_INT64: { 649 int64 value; 650 DO(ConsumeSignedInteger(&value, kint64max)); 651 SET_FIELD(Int64, value); 652 break; 653 } 654 655 case FieldDescriptor::CPPTYPE_UINT64: { 656 uint64 value; 657 DO(ConsumeUnsignedInteger(&value, kuint64max)); 658 SET_FIELD(UInt64, value); 659 break; 660 } 661 662 case FieldDescriptor::CPPTYPE_FLOAT: { 663 double value; 664 DO(ConsumeDouble(&value)); 665 SET_FIELD(Float, io::SafeDoubleToFloat(value)); 666 break; 667 } 668 669 case FieldDescriptor::CPPTYPE_DOUBLE: { 670 double value; 671 DO(ConsumeDouble(&value)); 672 SET_FIELD(Double, value); 673 break; 674 } 675 676 case FieldDescriptor::CPPTYPE_STRING: { 677 string value; 678 DO(ConsumeString(&value)); 679 SET_FIELD(String, value); 680 break; 681 } 682 683 case FieldDescriptor::CPPTYPE_BOOL: { 684 if (LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 685 uint64 value; 686 DO(ConsumeUnsignedInteger(&value, 1)); 687 SET_FIELD(Bool, value); 688 } else { 689 string value; 690 DO(ConsumeIdentifier(&value)); 691 if (value == "true" || value == "True" || value == "t") { 692 SET_FIELD(Bool, true); 693 } else if (value == "false" || value == "False" || value == "f") { 694 SET_FIELD(Bool, false); 695 } else { 696 ReportError("Invalid value for boolean field \"" + field->name() 697 + "\". Value: \"" + value + "\"."); 698 return false; 699 } 700 } 701 break; 702 } 703 704 case FieldDescriptor::CPPTYPE_ENUM: { 705 string value; 706 const EnumDescriptor* enum_type = field->enum_type(); 707 const EnumValueDescriptor* enum_value = NULL; 708 709 if (LookingAtType(io::Tokenizer::TYPE_IDENTIFIER)) { 710 DO(ConsumeIdentifier(&value)); 711 // Find the enumeration value. 712 enum_value = enum_type->FindValueByName(value); 713 714 } else if (LookingAt("-") || 715 LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 716 int64 int_value; 717 DO(ConsumeSignedInteger(&int_value, kint32max)); 718 value = SimpleItoa(int_value); // for error reporting 719 enum_value = enum_type->FindValueByNumber(int_value); 720 } else { 721 ReportError("Expected integer or identifier."); 722 return false; 723 } 724 725 if (enum_value == NULL) { 726 if (!allow_unknown_enum_) { 727 ReportError("Unknown enumeration value of \"" + value + "\" for " 728 "field \"" + field->name() + "\"."); 729 return false; 730 } else { 731 ReportWarning("Unknown enumeration value of \"" + value + "\" for " 732 "field \"" + field->name() + "\"."); 733 return true; 734 } 735 } 736 737 SET_FIELD(Enum, enum_value); 738 break; 739 } 740 741 case FieldDescriptor::CPPTYPE_MESSAGE: { 742 // We should never get here. Put here instead of a default 743 // so that if new types are added, we get a nice compiler warning. 744 GOOGLE_LOG(FATAL) << "Reached an unintended state: CPPTYPE_MESSAGE"; 745 break; 746 } 747 } 748 #undef SET_FIELD 749 return true; 750 } 751 SkipFieldValue()752 bool SkipFieldValue() { 753 if (LookingAtType(io::Tokenizer::TYPE_STRING)) { 754 while (LookingAtType(io::Tokenizer::TYPE_STRING)) { 755 tokenizer_.Next(); 756 } 757 return true; 758 } 759 // Possible field values other than string: 760 // 12345 => TYPE_INTEGER 761 // -12345 => TYPE_SYMBOL + TYPE_INTEGER 762 // 1.2345 => TYPE_FLOAT 763 // -1.2345 => TYPE_SYMBOL + TYPE_FLOAT 764 // inf => TYPE_IDENTIFIER 765 // -inf => TYPE_SYMBOL + TYPE_IDENTIFIER 766 // TYPE_INTEGER => TYPE_IDENTIFIER 767 // Divides them into two group, one with TYPE_SYMBOL 768 // and the other without: 769 // Group one: 770 // 12345 => TYPE_INTEGER 771 // 1.2345 => TYPE_FLOAT 772 // inf => TYPE_IDENTIFIER 773 // TYPE_INTEGER => TYPE_IDENTIFIER 774 // Group two: 775 // -12345 => TYPE_SYMBOL + TYPE_INTEGER 776 // -1.2345 => TYPE_SYMBOL + TYPE_FLOAT 777 // -inf => TYPE_SYMBOL + TYPE_IDENTIFIER 778 // As we can see, the field value consists of an optional '-' and one of 779 // TYPE_INTEGER, TYPE_FLOAT and TYPE_IDENTIFIER. 780 bool has_minus = TryConsume("-"); 781 if (!LookingAtType(io::Tokenizer::TYPE_INTEGER) && 782 !LookingAtType(io::Tokenizer::TYPE_FLOAT) && 783 !LookingAtType(io::Tokenizer::TYPE_IDENTIFIER)) { 784 return false; 785 } 786 // Combination of '-' and TYPE_IDENTIFIER may result in an invalid field 787 // value while other combinations all generate valid values. 788 // We check if the value of this combination is valid here. 789 // TYPE_IDENTIFIER after a '-' should be one of the float values listed 790 // below: 791 // inf, inff, infinity, nan 792 if (has_minus && LookingAtType(io::Tokenizer::TYPE_IDENTIFIER)) { 793 string text = tokenizer_.current().text; 794 LowerString(&text); 795 if (text != "inf" && 796 text != "infinity" && 797 text != "nan") { 798 ReportError("Invalid float number: " + text); 799 return false; 800 } 801 } 802 tokenizer_.Next(); 803 return true; 804 } 805 806 // Returns true if the current token's text is equal to that specified. LookingAt(const string & text)807 bool LookingAt(const string& text) { 808 return tokenizer_.current().text == text; 809 } 810 811 // Returns true if the current token's type is equal to that specified. LookingAtType(io::Tokenizer::TokenType token_type)812 bool LookingAtType(io::Tokenizer::TokenType token_type) { 813 return tokenizer_.current().type == token_type; 814 } 815 816 // Consumes an identifier and saves its value in the identifier parameter. 817 // Returns false if the token is not of type IDENTFIER. ConsumeIdentifier(string * identifier)818 bool ConsumeIdentifier(string* identifier) { 819 if (LookingAtType(io::Tokenizer::TYPE_IDENTIFIER)) { 820 *identifier = tokenizer_.current().text; 821 tokenizer_.Next(); 822 return true; 823 } 824 825 // If allow_field_numer_ or allow_unknown_field_ is true, we should able 826 // to parse integer identifiers. 827 if ((allow_field_number_ || allow_unknown_field_) 828 && LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 829 *identifier = tokenizer_.current().text; 830 tokenizer_.Next(); 831 return true; 832 } 833 834 ReportError("Expected identifier."); 835 return false; 836 } 837 838 // Consume a string of form "<id1>.<id2>....<idN>". ConsumeFullTypeName(string * name)839 bool ConsumeFullTypeName(string* name) { 840 DO(ConsumeIdentifier(name)); 841 while (TryConsume(".")) { 842 string part; 843 DO(ConsumeIdentifier(&part)); 844 *name += "."; 845 *name += part; 846 } 847 return true; 848 } 849 850 // Consumes a string and saves its value in the text parameter. 851 // Returns false if the token is not of type STRING. ConsumeString(string * text)852 bool ConsumeString(string* text) { 853 if (!LookingAtType(io::Tokenizer::TYPE_STRING)) { 854 ReportError("Expected string."); 855 return false; 856 } 857 858 text->clear(); 859 while (LookingAtType(io::Tokenizer::TYPE_STRING)) { 860 io::Tokenizer::ParseStringAppend(tokenizer_.current().text, text); 861 862 tokenizer_.Next(); 863 } 864 865 return true; 866 } 867 868 // Consumes a uint64 and saves its value in the value parameter. 869 // Returns false if the token is not of type INTEGER. ConsumeUnsignedInteger(uint64 * value,uint64 max_value)870 bool ConsumeUnsignedInteger(uint64* value, uint64 max_value) { 871 if (!LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 872 ReportError("Expected integer."); 873 return false; 874 } 875 876 if (!io::Tokenizer::ParseInteger(tokenizer_.current().text, 877 max_value, value)) { 878 ReportError("Integer out of range."); 879 return false; 880 } 881 882 tokenizer_.Next(); 883 return true; 884 } 885 886 // Consumes an int64 and saves its value in the value parameter. 887 // Note that since the tokenizer does not support negative numbers, 888 // we actually may consume an additional token (for the minus sign) in this 889 // method. Returns false if the token is not an integer 890 // (signed or otherwise). ConsumeSignedInteger(int64 * value,uint64 max_value)891 bool ConsumeSignedInteger(int64* value, uint64 max_value) { 892 bool negative = false; 893 894 if (TryConsume("-")) { 895 negative = true; 896 // Two's complement always allows one more negative integer than 897 // positive. 898 ++max_value; 899 } 900 901 uint64 unsigned_value; 902 903 DO(ConsumeUnsignedInteger(&unsigned_value, max_value)); 904 905 *value = static_cast<int64>(unsigned_value); 906 907 if (negative) { 908 *value = -*value; 909 } 910 911 return true; 912 } 913 914 // Consumes a uint64 and saves its value in the value parameter. 915 // Accepts decimal numbers only, rejects hex or oct numbers. ConsumeUnsignedDecimalInteger(uint64 * value,uint64 max_value)916 bool ConsumeUnsignedDecimalInteger(uint64* value, uint64 max_value) { 917 if (!LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 918 ReportError("Expected integer."); 919 return false; 920 } 921 922 const string& text = tokenizer_.current().text; 923 if (IsHexNumber(text) || IsOctNumber(text)) { 924 ReportError("Expect a decimal number."); 925 return false; 926 } 927 928 if (!io::Tokenizer::ParseInteger(text, max_value, value)) { 929 ReportError("Integer out of range."); 930 return false; 931 } 932 933 tokenizer_.Next(); 934 return true; 935 } 936 937 // Consumes a double and saves its value in the value parameter. 938 // Note that since the tokenizer does not support negative numbers, 939 // we actually may consume an additional token (for the minus sign) in this 940 // method. Returns false if the token is not a double 941 // (signed or otherwise). ConsumeDouble(double * value)942 bool ConsumeDouble(double* value) { 943 bool negative = false; 944 945 if (TryConsume("-")) { 946 negative = true; 947 } 948 949 // A double can actually be an integer, according to the tokenizer. 950 // Therefore, we must check both cases here. 951 if (LookingAtType(io::Tokenizer::TYPE_INTEGER)) { 952 // We have found an integer value for the double. 953 uint64 integer_value; 954 DO(ConsumeUnsignedDecimalInteger(&integer_value, kuint64max)); 955 956 *value = static_cast<double>(integer_value); 957 } else if (LookingAtType(io::Tokenizer::TYPE_FLOAT)) { 958 // We have found a float value for the double. 959 *value = io::Tokenizer::ParseFloat(tokenizer_.current().text); 960 961 // Mark the current token as consumed. 962 tokenizer_.Next(); 963 } else if (LookingAtType(io::Tokenizer::TYPE_IDENTIFIER)) { 964 string text = tokenizer_.current().text; 965 LowerString(&text); 966 if (text == "inf" || 967 text == "infinity") { 968 *value = std::numeric_limits<double>::infinity(); 969 tokenizer_.Next(); 970 } else if (text == "nan") { 971 *value = std::numeric_limits<double>::quiet_NaN(); 972 tokenizer_.Next(); 973 } else { 974 ReportError("Expected double."); 975 return false; 976 } 977 } else { 978 ReportError("Expected double."); 979 return false; 980 } 981 982 if (negative) { 983 *value = -*value; 984 } 985 986 return true; 987 } 988 989 // Consumes Any::type_url value, of form "type.googleapis.com/full.type.Name" 990 // or "type.googleprod.com/full.type.Name" ConsumeAnyTypeUrl(string * full_type_name,string * prefix)991 bool ConsumeAnyTypeUrl(string* full_type_name, string* prefix) { 992 // TODO(saito) Extend Consume() to consume multiple tokens at once, so that 993 // this code can be written as just DO(Consume(kGoogleApisTypePrefix)). 994 string url1, url2, url3; 995 DO(ConsumeIdentifier(&url1)); // type 996 DO(Consume(".")); 997 DO(ConsumeIdentifier(&url2)); // googleapis 998 DO(Consume(".")); 999 DO(ConsumeIdentifier(&url3)); // com 1000 DO(Consume("/")); 1001 DO(ConsumeFullTypeName(full_type_name)); 1002 1003 *prefix = url1 + "." + url2 + "." + url3 + "/"; 1004 if (*prefix != internal::kTypeGoogleApisComPrefix && 1005 *prefix != internal::kTypeGoogleProdComPrefix) { 1006 ReportError("TextFormat::Parser for Any supports only " 1007 "type.googleapis.com and type.googleprod.com, " 1008 "but found \"" + *prefix + "\""); 1009 return false; 1010 } 1011 return true; 1012 } 1013 1014 // A helper function for reconstructing Any::value. Consumes a text of 1015 // full_type_name, then serializes it into serialized_value. "pool" is used to 1016 // look up and create a temporary object with full_type_name. ConsumeAnyValue(const string & full_type_name,const DescriptorPool * pool,string * serialized_value)1017 bool ConsumeAnyValue(const string& full_type_name, const DescriptorPool* pool, 1018 string* serialized_value) { 1019 const Descriptor* value_descriptor = 1020 pool->FindMessageTypeByName(full_type_name); 1021 if (value_descriptor == NULL) { 1022 ReportError("Could not find type \"" + full_type_name + 1023 "\" stored in google.protobuf.Any."); 1024 return false; 1025 } 1026 DynamicMessageFactory factory; 1027 const Message* value_prototype = factory.GetPrototype(value_descriptor); 1028 if (value_prototype == NULL) { 1029 return false; 1030 } 1031 google::protobuf::scoped_ptr<Message> value(value_prototype->New()); 1032 string sub_delimiter; 1033 DO(ConsumeMessageDelimiter(&sub_delimiter)); 1034 DO(ConsumeMessage(value.get(), sub_delimiter)); 1035 1036 value->AppendToString(serialized_value); 1037 return true; 1038 } 1039 1040 // Consumes a token and confirms that it matches that specified in the 1041 // value parameter. Returns false if the token found does not match that 1042 // which was specified. Consume(const string & value)1043 bool Consume(const string& value) { 1044 const string& current_value = tokenizer_.current().text; 1045 1046 if (current_value != value) { 1047 ReportError("Expected \"" + value + "\", found \"" + current_value 1048 + "\"."); 1049 return false; 1050 } 1051 1052 tokenizer_.Next(); 1053 1054 return true; 1055 } 1056 1057 // Attempts to consume the supplied value. Returns false if a the 1058 // token found does not match the value specified. TryConsume(const string & value)1059 bool TryConsume(const string& value) { 1060 if (tokenizer_.current().text == value) { 1061 tokenizer_.Next(); 1062 return true; 1063 } else { 1064 return false; 1065 } 1066 } 1067 1068 // An internal instance of the Tokenizer's error collector, used to 1069 // collect any base-level parse errors and feed them to the ParserImpl. 1070 class ParserErrorCollector : public io::ErrorCollector { 1071 public: ParserErrorCollector(TextFormat::Parser::ParserImpl * parser)1072 explicit ParserErrorCollector(TextFormat::Parser::ParserImpl* parser) : 1073 parser_(parser) { } 1074 ~ParserErrorCollector()1075 virtual ~ParserErrorCollector() { } 1076 AddError(int line,int column,const string & message)1077 virtual void AddError(int line, int column, const string& message) { 1078 parser_->ReportError(line, column, message); 1079 } 1080 AddWarning(int line,int column,const string & message)1081 virtual void AddWarning(int line, int column, const string& message) { 1082 parser_->ReportWarning(line, column, message); 1083 } 1084 1085 private: 1086 GOOGLE_DISALLOW_EVIL_CONSTRUCTORS(ParserErrorCollector); 1087 TextFormat::Parser::ParserImpl* parser_; 1088 }; 1089 1090 io::ErrorCollector* error_collector_; 1091 TextFormat::Finder* finder_; 1092 ParseInfoTree* parse_info_tree_; 1093 ParserErrorCollector tokenizer_error_collector_; 1094 io::Tokenizer tokenizer_; 1095 const Descriptor* root_message_type_; 1096 SingularOverwritePolicy singular_overwrite_policy_; 1097 const bool allow_case_insensitive_field_; 1098 const bool allow_unknown_field_; 1099 const bool allow_unknown_enum_; 1100 const bool allow_field_number_; 1101 bool had_errors_; 1102 }; 1103 1104 #undef DO 1105 1106 // =========================================================================== 1107 // Internal class for writing text to the io::ZeroCopyOutputStream. Adapted 1108 // from the Printer found in //google/protobuf/io/printer.h 1109 class TextFormat::Printer::TextGenerator { 1110 public: TextGenerator(io::ZeroCopyOutputStream * output,int initial_indent_level)1111 explicit TextGenerator(io::ZeroCopyOutputStream* output, 1112 int initial_indent_level) 1113 : output_(output), 1114 buffer_(NULL), 1115 buffer_size_(0), 1116 at_start_of_line_(true), 1117 failed_(false), 1118 indent_(""), 1119 initial_indent_level_(initial_indent_level) { 1120 indent_.resize(initial_indent_level_ * 2, ' '); 1121 } 1122 ~TextGenerator()1123 ~TextGenerator() { 1124 // Only BackUp() if we're sure we've successfully called Next() at least 1125 // once. 1126 if (!failed_ && buffer_size_ > 0) { 1127 output_->BackUp(buffer_size_); 1128 } 1129 } 1130 1131 // Indent text by two spaces. After calling Indent(), two spaces will be 1132 // inserted at the beginning of each line of text. Indent() may be called 1133 // multiple times to produce deeper indents. Indent()1134 void Indent() { 1135 indent_ += " "; 1136 } 1137 1138 // Reduces the current indent level by two spaces, or crashes if the indent 1139 // level is zero. Outdent()1140 void Outdent() { 1141 if (indent_.empty() || 1142 indent_.size() < initial_indent_level_ * 2) { 1143 GOOGLE_LOG(DFATAL) << " Outdent() without matching Indent()."; 1144 return; 1145 } 1146 1147 indent_.resize(indent_.size() - 2); 1148 } 1149 1150 // Print text to the output stream. Print(const string & str)1151 void Print(const string& str) { 1152 Print(str.data(), str.size()); 1153 } 1154 1155 // Print text to the output stream. Print(const char * text)1156 void Print(const char* text) { 1157 Print(text, strlen(text)); 1158 } 1159 1160 // Print text to the output stream. Print(const char * text,size_t size)1161 void Print(const char* text, size_t size) { 1162 size_t pos = 0; // The number of bytes we've written so far. 1163 1164 for (size_t i = 0; i < size; i++) { 1165 if (text[i] == '\n') { 1166 // Saw newline. If there is more text, we may need to insert an indent 1167 // here. So, write what we have so far, including the '\n'. 1168 Write(text + pos, i - pos + 1); 1169 pos = i + 1; 1170 1171 // Setting this true will cause the next Write() to insert an indent 1172 // first. 1173 at_start_of_line_ = true; 1174 } 1175 } 1176 1177 // Write the rest. 1178 Write(text + pos, size - pos); 1179 } 1180 1181 // True if any write to the underlying stream failed. (We don't just 1182 // crash in this case because this is an I/O failure, not a programming 1183 // error.) failed() const1184 bool failed() const { return failed_; } 1185 1186 private: 1187 GOOGLE_DISALLOW_EVIL_CONSTRUCTORS(TextGenerator); 1188 Write(const char * data,size_t size)1189 void Write(const char* data, size_t size) { 1190 if (failed_) return; 1191 if (size == 0) return; 1192 1193 if (at_start_of_line_) { 1194 // Insert an indent. 1195 at_start_of_line_ = false; 1196 Write(indent_.data(), indent_.size()); 1197 if (failed_) return; 1198 } 1199 1200 while (size > buffer_size_) { 1201 // Data exceeds space in the buffer. Copy what we can and request a 1202 // new buffer. 1203 memcpy(buffer_, data, buffer_size_); 1204 data += buffer_size_; 1205 size -= buffer_size_; 1206 void* void_buffer; 1207 failed_ = !output_->Next(&void_buffer, &buffer_size_); 1208 if (failed_) return; 1209 buffer_ = reinterpret_cast<char*>(void_buffer); 1210 } 1211 1212 // Buffer is big enough to receive the data; copy it. 1213 memcpy(buffer_, data, size); 1214 buffer_ += size; 1215 buffer_size_ -= size; 1216 } 1217 1218 io::ZeroCopyOutputStream* const output_; 1219 char* buffer_; 1220 int buffer_size_; 1221 bool at_start_of_line_; 1222 bool failed_; 1223 1224 string indent_; 1225 int initial_indent_level_; 1226 }; 1227 1228 // =========================================================================== 1229 ~Finder()1230 TextFormat::Finder::~Finder() { 1231 } 1232 Parser()1233 TextFormat::Parser::Parser() 1234 : error_collector_(NULL), 1235 finder_(NULL), 1236 parse_info_tree_(NULL), 1237 allow_partial_(false), 1238 allow_case_insensitive_field_(false), 1239 allow_unknown_field_(false), 1240 allow_unknown_enum_(false), 1241 allow_field_number_(false), 1242 allow_relaxed_whitespace_(false), 1243 allow_singular_overwrites_(false) { 1244 } 1245 ~Parser()1246 TextFormat::Parser::~Parser() {} 1247 Parse(io::ZeroCopyInputStream * input,Message * output)1248 bool TextFormat::Parser::Parse(io::ZeroCopyInputStream* input, 1249 Message* output) { 1250 output->Clear(); 1251 1252 ParserImpl::SingularOverwritePolicy overwrites_policy = 1253 allow_singular_overwrites_ 1254 ? ParserImpl::ALLOW_SINGULAR_OVERWRITES 1255 : ParserImpl::FORBID_SINGULAR_OVERWRITES; 1256 1257 ParserImpl parser(output->GetDescriptor(), input, error_collector_, 1258 finder_, parse_info_tree_, 1259 overwrites_policy, 1260 allow_case_insensitive_field_, allow_unknown_field_, 1261 allow_unknown_enum_, allow_field_number_, 1262 allow_relaxed_whitespace_); 1263 return MergeUsingImpl(input, output, &parser); 1264 } 1265 ParseFromString(const string & input,Message * output)1266 bool TextFormat::Parser::ParseFromString(const string& input, 1267 Message* output) { 1268 io::ArrayInputStream input_stream(input.data(), input.size()); 1269 return Parse(&input_stream, output); 1270 } 1271 Merge(io::ZeroCopyInputStream * input,Message * output)1272 bool TextFormat::Parser::Merge(io::ZeroCopyInputStream* input, 1273 Message* output) { 1274 ParserImpl parser(output->GetDescriptor(), input, error_collector_, 1275 finder_, parse_info_tree_, 1276 ParserImpl::ALLOW_SINGULAR_OVERWRITES, 1277 allow_case_insensitive_field_, allow_unknown_field_, 1278 allow_unknown_enum_, allow_field_number_, 1279 allow_relaxed_whitespace_); 1280 return MergeUsingImpl(input, output, &parser); 1281 } 1282 MergeFromString(const string & input,Message * output)1283 bool TextFormat::Parser::MergeFromString(const string& input, 1284 Message* output) { 1285 io::ArrayInputStream input_stream(input.data(), input.size()); 1286 return Merge(&input_stream, output); 1287 } 1288 MergeUsingImpl(io::ZeroCopyInputStream *,Message * output,ParserImpl * parser_impl)1289 bool TextFormat::Parser::MergeUsingImpl(io::ZeroCopyInputStream* /* input */, 1290 Message* output, 1291 ParserImpl* parser_impl) { 1292 if (!parser_impl->Parse(output)) return false; 1293 if (!allow_partial_ && !output->IsInitialized()) { 1294 vector<string> missing_fields; 1295 output->FindInitializationErrors(&missing_fields); 1296 parser_impl->ReportError(-1, 0, "Message missing required fields: " + 1297 Join(missing_fields, ", ")); 1298 return false; 1299 } 1300 return true; 1301 } 1302 ParseFieldValueFromString(const string & input,const FieldDescriptor * field,Message * output)1303 bool TextFormat::Parser::ParseFieldValueFromString( 1304 const string& input, 1305 const FieldDescriptor* field, 1306 Message* output) { 1307 io::ArrayInputStream input_stream(input.data(), input.size()); 1308 ParserImpl parser(output->GetDescriptor(), &input_stream, error_collector_, 1309 finder_, parse_info_tree_, 1310 ParserImpl::ALLOW_SINGULAR_OVERWRITES, 1311 allow_case_insensitive_field_, allow_unknown_field_, 1312 allow_unknown_enum_, allow_field_number_, 1313 allow_relaxed_whitespace_); 1314 return parser.ParseField(field, output); 1315 } 1316 Parse(io::ZeroCopyInputStream * input,Message * output)1317 /* static */ bool TextFormat::Parse(io::ZeroCopyInputStream* input, 1318 Message* output) { 1319 return Parser().Parse(input, output); 1320 } 1321 Merge(io::ZeroCopyInputStream * input,Message * output)1322 /* static */ bool TextFormat::Merge(io::ZeroCopyInputStream* input, 1323 Message* output) { 1324 return Parser().Merge(input, output); 1325 } 1326 ParseFromString(const string & input,Message * output)1327 /* static */ bool TextFormat::ParseFromString(const string& input, 1328 Message* output) { 1329 return Parser().ParseFromString(input, output); 1330 } 1331 MergeFromString(const string & input,Message * output)1332 /* static */ bool TextFormat::MergeFromString(const string& input, 1333 Message* output) { 1334 return Parser().MergeFromString(input, output); 1335 } 1336 1337 // =========================================================================== 1338 1339 // The default implementation for FieldValuePrinter. The base class just 1340 // does simple formatting. That way, deriving classes could decide to fallback 1341 // to that behavior. FieldValuePrinter()1342 TextFormat::FieldValuePrinter::FieldValuePrinter() {} ~FieldValuePrinter()1343 TextFormat::FieldValuePrinter::~FieldValuePrinter() {} PrintBool(bool val) const1344 string TextFormat::FieldValuePrinter::PrintBool(bool val) const { 1345 return val ? "true" : "false"; 1346 } PrintInt32(int32 val) const1347 string TextFormat::FieldValuePrinter::PrintInt32(int32 val) const { 1348 return SimpleItoa(val); 1349 } PrintUInt32(uint32 val) const1350 string TextFormat::FieldValuePrinter::PrintUInt32(uint32 val) const { 1351 return SimpleItoa(val); 1352 } PrintInt64(int64 val) const1353 string TextFormat::FieldValuePrinter::PrintInt64(int64 val) const { 1354 return SimpleItoa(val); 1355 } PrintUInt64(uint64 val) const1356 string TextFormat::FieldValuePrinter::PrintUInt64(uint64 val) const { 1357 return SimpleItoa(val); 1358 } PrintFloat(float val) const1359 string TextFormat::FieldValuePrinter::PrintFloat(float val) const { 1360 return SimpleFtoa(val); 1361 } PrintDouble(double val) const1362 string TextFormat::FieldValuePrinter::PrintDouble(double val) const { 1363 return SimpleDtoa(val); 1364 } PrintString(const string & val) const1365 string TextFormat::FieldValuePrinter::PrintString(const string& val) const { 1366 string printed("\""); 1367 CEscapeAndAppend(val, &printed); 1368 printed.push_back('\"'); 1369 return printed; 1370 } PrintBytes(const string & val) const1371 string TextFormat::FieldValuePrinter::PrintBytes(const string& val) const { 1372 return PrintString(val); 1373 } PrintEnum(int32 val,const string & name) const1374 string TextFormat::FieldValuePrinter::PrintEnum(int32 val, 1375 const string& name) const { 1376 return name; 1377 } PrintFieldName(const Message & message,const Reflection * reflection,const FieldDescriptor * field) const1378 string TextFormat::FieldValuePrinter::PrintFieldName( 1379 const Message& message, 1380 const Reflection* reflection, 1381 const FieldDescriptor* field) const { 1382 if (field->is_extension()) { 1383 // We special-case MessageSet elements for compatibility with proto1. 1384 if (field->containing_type()->options().message_set_wire_format() 1385 && field->type() == FieldDescriptor::TYPE_MESSAGE 1386 && field->is_optional() 1387 && field->extension_scope() == field->message_type()) { 1388 return StrCat("[", field->message_type()->full_name(), "]"); 1389 } else { 1390 return StrCat("[", field->full_name(), "]"); 1391 } 1392 } else if (field->type() == FieldDescriptor::TYPE_GROUP) { 1393 // Groups must be serialized with their original capitalization. 1394 return field->message_type()->name(); 1395 } else { 1396 return field->name(); 1397 } 1398 } PrintMessageStart(const Message & message,int field_index,int field_count,bool single_line_mode) const1399 string TextFormat::FieldValuePrinter::PrintMessageStart( 1400 const Message& message, 1401 int field_index, 1402 int field_count, 1403 bool single_line_mode) const { 1404 return single_line_mode ? " { " : " {\n"; 1405 } PrintMessageEnd(const Message & message,int field_index,int field_count,bool single_line_mode) const1406 string TextFormat::FieldValuePrinter::PrintMessageEnd( 1407 const Message& message, 1408 int field_index, 1409 int field_count, 1410 bool single_line_mode) const { 1411 return single_line_mode ? "} " : "}\n"; 1412 } 1413 1414 namespace { 1415 // Our own specialization: for UTF8 escaped strings. 1416 class FieldValuePrinterUtf8Escaping : public TextFormat::FieldValuePrinter { 1417 public: PrintString(const string & val) const1418 virtual string PrintString(const string& val) const { 1419 return StrCat("\"", strings::Utf8SafeCEscape(val), "\""); 1420 } PrintBytes(const string & val) const1421 virtual string PrintBytes(const string& val) const { 1422 return TextFormat::FieldValuePrinter::PrintString(val); 1423 } 1424 }; 1425 1426 } // namespace 1427 Printer()1428 TextFormat::Printer::Printer() 1429 : initial_indent_level_(0), 1430 single_line_mode_(false), 1431 use_field_number_(false), 1432 use_short_repeated_primitives_(false), 1433 hide_unknown_fields_(false), 1434 print_message_fields_in_index_order_(false), 1435 expand_any_(false), 1436 truncate_string_field_longer_than_(0LL) { 1437 SetUseUtf8StringEscaping(false); 1438 } 1439 ~Printer()1440 TextFormat::Printer::~Printer() { 1441 STLDeleteValues(&custom_printers_); 1442 } 1443 SetUseUtf8StringEscaping(bool as_utf8)1444 void TextFormat::Printer::SetUseUtf8StringEscaping(bool as_utf8) { 1445 SetDefaultFieldValuePrinter(as_utf8 1446 ? new FieldValuePrinterUtf8Escaping() 1447 : new FieldValuePrinter()); 1448 } 1449 SetDefaultFieldValuePrinter(const FieldValuePrinter * printer)1450 void TextFormat::Printer::SetDefaultFieldValuePrinter( 1451 const FieldValuePrinter* printer) { 1452 default_field_value_printer_.reset(printer); 1453 } 1454 RegisterFieldValuePrinter(const FieldDescriptor * field,const FieldValuePrinter * printer)1455 bool TextFormat::Printer::RegisterFieldValuePrinter( 1456 const FieldDescriptor* field, 1457 const FieldValuePrinter* printer) { 1458 return field != NULL && printer != NULL && 1459 custom_printers_.insert(std::make_pair(field, printer)).second; 1460 } 1461 PrintToString(const Message & message,string * output) const1462 bool TextFormat::Printer::PrintToString(const Message& message, 1463 string* output) const { 1464 GOOGLE_DCHECK(output) << "output specified is NULL"; 1465 1466 output->clear(); 1467 io::StringOutputStream output_stream(output); 1468 1469 return Print(message, &output_stream); 1470 } 1471 PrintUnknownFieldsToString(const UnknownFieldSet & unknown_fields,string * output) const1472 bool TextFormat::Printer::PrintUnknownFieldsToString( 1473 const UnknownFieldSet& unknown_fields, 1474 string* output) const { 1475 GOOGLE_DCHECK(output) << "output specified is NULL"; 1476 1477 output->clear(); 1478 io::StringOutputStream output_stream(output); 1479 return PrintUnknownFields(unknown_fields, &output_stream); 1480 } 1481 Print(const Message & message,io::ZeroCopyOutputStream * output) const1482 bool TextFormat::Printer::Print(const Message& message, 1483 io::ZeroCopyOutputStream* output) const { 1484 TextGenerator generator(output, initial_indent_level_); 1485 1486 Print(message, generator); 1487 1488 // Output false if the generator failed internally. 1489 return !generator.failed(); 1490 } 1491 PrintUnknownFields(const UnknownFieldSet & unknown_fields,io::ZeroCopyOutputStream * output) const1492 bool TextFormat::Printer::PrintUnknownFields( 1493 const UnknownFieldSet& unknown_fields, 1494 io::ZeroCopyOutputStream* output) const { 1495 TextGenerator generator(output, initial_indent_level_); 1496 1497 PrintUnknownFields(unknown_fields, generator); 1498 1499 // Output false if the generator failed internally. 1500 return !generator.failed(); 1501 } 1502 1503 namespace { 1504 // Comparison functor for sorting FieldDescriptors by field index. 1505 struct FieldIndexSorter { operator ()google::protobuf::__anon6b098f920311::FieldIndexSorter1506 bool operator()(const FieldDescriptor* left, 1507 const FieldDescriptor* right) const { 1508 return left->index() < right->index(); 1509 } 1510 }; 1511 1512 } // namespace 1513 PrintAny(const Message & message,TextGenerator & generator) const1514 bool TextFormat::Printer::PrintAny(const Message& message, 1515 TextGenerator& generator) const { 1516 const FieldDescriptor* type_url_field; 1517 const FieldDescriptor* value_field; 1518 if (!internal::GetAnyFieldDescriptors(message, &type_url_field, 1519 &value_field)) { 1520 return false; 1521 } 1522 1523 const Reflection* reflection = message.GetReflection(); 1524 1525 // Extract the full type name from the type_url field. 1526 const string& type_url = reflection->GetString(message, type_url_field); 1527 string full_type_name; 1528 if (!internal::ParseAnyTypeUrl(type_url, &full_type_name)) { 1529 return false; 1530 } 1531 1532 // Print the "value" in text. 1533 const google::protobuf::Descriptor* value_descriptor = 1534 message.GetDescriptor()->file()->pool()->FindMessageTypeByName( 1535 full_type_name); 1536 if (value_descriptor == NULL) { 1537 GOOGLE_LOG(WARNING) << "Proto type " << type_url << " not found"; 1538 return false; 1539 } 1540 DynamicMessageFactory factory; 1541 google::protobuf::scoped_ptr<google::protobuf::Message> value_message( 1542 factory.GetPrototype(value_descriptor)->New()); 1543 string serialized_value = reflection->GetString(message, value_field); 1544 if (!value_message->ParseFromString(serialized_value)) { 1545 GOOGLE_LOG(WARNING) << type_url << ": failed to parse contents"; 1546 return false; 1547 } 1548 generator.Print(StrCat("[", type_url, "]")); 1549 const FieldValuePrinter* printer = FindWithDefault( 1550 custom_printers_, value_field, default_field_value_printer_.get()); 1551 generator.Print( 1552 printer->PrintMessageStart(message, -1, 0, single_line_mode_)); 1553 generator.Indent(); 1554 Print(*value_message, generator); 1555 generator.Outdent(); 1556 generator.Print(printer->PrintMessageEnd(message, -1, 0, single_line_mode_)); 1557 return true; 1558 } 1559 Print(const Message & message,TextGenerator & generator) const1560 void TextFormat::Printer::Print(const Message& message, 1561 TextGenerator& generator) const { 1562 const Descriptor* descriptor = message.GetDescriptor(); 1563 const Reflection* reflection = message.GetReflection(); 1564 if (descriptor->full_name() == internal::kAnyFullTypeName && expand_any_ && 1565 PrintAny(message, generator)) { 1566 return; 1567 } 1568 vector<const FieldDescriptor*> fields; 1569 reflection->ListFields(message, &fields); 1570 if (print_message_fields_in_index_order_) { 1571 std::sort(fields.begin(), fields.end(), FieldIndexSorter()); 1572 } 1573 for (int i = 0; i < fields.size(); i++) { 1574 PrintField(message, reflection, fields[i], generator); 1575 } 1576 if (!hide_unknown_fields_) { 1577 PrintUnknownFields(reflection->GetUnknownFields(message), generator); 1578 } 1579 } 1580 PrintFieldValueToString(const Message & message,const FieldDescriptor * field,int index,string * output) const1581 void TextFormat::Printer::PrintFieldValueToString( 1582 const Message& message, 1583 const FieldDescriptor* field, 1584 int index, 1585 string* output) const { 1586 1587 GOOGLE_DCHECK(output) << "output specified is NULL"; 1588 1589 output->clear(); 1590 io::StringOutputStream output_stream(output); 1591 TextGenerator generator(&output_stream, initial_indent_level_); 1592 1593 PrintFieldValue(message, message.GetReflection(), field, index, generator); 1594 } 1595 1596 class MapEntryMessageComparator { 1597 public: MapEntryMessageComparator(const Descriptor * descriptor)1598 explicit MapEntryMessageComparator(const Descriptor* descriptor) 1599 : field_(descriptor->field(0)) {} 1600 operator ()(const Message * a,const Message * b)1601 bool operator()(const Message* a, const Message* b) { 1602 const Reflection* reflection = a->GetReflection(); 1603 switch (field_->cpp_type()) { 1604 case FieldDescriptor::CPPTYPE_BOOL: { 1605 bool first = reflection->GetBool(*a, field_); 1606 bool second = reflection->GetBool(*b, field_); 1607 return first < second; 1608 } 1609 case FieldDescriptor::CPPTYPE_INT32: { 1610 int32 first = reflection->GetInt32(*a, field_); 1611 int32 second = reflection->GetInt32(*b, field_); 1612 return first < second; 1613 } 1614 case FieldDescriptor::CPPTYPE_INT64: { 1615 int64 first = reflection->GetInt64(*a, field_); 1616 int64 second = reflection->GetInt64(*b, field_); 1617 return first < second; 1618 } 1619 case FieldDescriptor::CPPTYPE_UINT32: { 1620 uint32 first = reflection->GetUInt32(*a, field_); 1621 uint32 second = reflection->GetUInt32(*b, field_); 1622 return first < second; 1623 } 1624 case FieldDescriptor::CPPTYPE_UINT64: { 1625 uint64 first = reflection->GetUInt64(*a, field_); 1626 uint64 second = reflection->GetUInt64(*b, field_); 1627 return first < second; 1628 } 1629 case FieldDescriptor::CPPTYPE_STRING: { 1630 string first = reflection->GetString(*a, field_); 1631 string second = reflection->GetString(*b, field_); 1632 return first < second; 1633 } 1634 default: 1635 GOOGLE_LOG(DFATAL) << "Invalid key for map field."; 1636 return true; 1637 } 1638 } 1639 1640 private: 1641 const FieldDescriptor* field_; 1642 }; 1643 PrintField(const Message & message,const Reflection * reflection,const FieldDescriptor * field,TextGenerator & generator) const1644 void TextFormat::Printer::PrintField(const Message& message, 1645 const Reflection* reflection, 1646 const FieldDescriptor* field, 1647 TextGenerator& generator) const { 1648 if (use_short_repeated_primitives_ && 1649 field->is_repeated() && 1650 field->cpp_type() != FieldDescriptor::CPPTYPE_STRING && 1651 field->cpp_type() != FieldDescriptor::CPPTYPE_MESSAGE) { 1652 PrintShortRepeatedField(message, reflection, field, generator); 1653 return; 1654 } 1655 1656 int count = 0; 1657 1658 if (field->is_repeated()) { 1659 count = reflection->FieldSize(message, field); 1660 } else if (reflection->HasField(message, field)) { 1661 count = 1; 1662 } 1663 1664 std::vector<const Message*> sorted_map_field; 1665 if (field->is_map()) { 1666 const RepeatedPtrField<Message>& map_field = 1667 reflection->GetRepeatedPtrField<Message>(message, field); 1668 for (RepeatedPtrField<Message>::const_pointer_iterator it = 1669 map_field.pointer_begin(); 1670 it != map_field.pointer_end(); ++it) { 1671 sorted_map_field.push_back(*it); 1672 } 1673 1674 MapEntryMessageComparator comparator(field->message_type()); 1675 std::stable_sort(sorted_map_field.begin(), sorted_map_field.end(), 1676 comparator); 1677 } 1678 1679 for (int j = 0; j < count; ++j) { 1680 const int field_index = field->is_repeated() ? j : -1; 1681 1682 PrintFieldName(message, reflection, field, generator); 1683 1684 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) { 1685 const FieldValuePrinter* printer = FindWithDefault( 1686 custom_printers_, field, default_field_value_printer_.get()); 1687 const Message& sub_message = 1688 field->is_repeated() 1689 ? (field->is_map() 1690 ? *sorted_map_field[j] 1691 : reflection->GetRepeatedMessage(message, field, j)) 1692 : reflection->GetMessage(message, field); 1693 generator.Print( 1694 printer->PrintMessageStart( 1695 sub_message, field_index, count, single_line_mode_)); 1696 generator.Indent(); 1697 Print(sub_message, generator); 1698 generator.Outdent(); 1699 generator.Print( 1700 printer->PrintMessageEnd( 1701 sub_message, field_index, count, single_line_mode_)); 1702 } else { 1703 generator.Print(": "); 1704 // Write the field value. 1705 PrintFieldValue(message, reflection, field, field_index, generator); 1706 if (single_line_mode_) { 1707 generator.Print(" "); 1708 } else { 1709 generator.Print("\n"); 1710 } 1711 } 1712 } 1713 } 1714 PrintShortRepeatedField(const Message & message,const Reflection * reflection,const FieldDescriptor * field,TextGenerator & generator) const1715 void TextFormat::Printer::PrintShortRepeatedField( 1716 const Message& message, 1717 const Reflection* reflection, 1718 const FieldDescriptor* field, 1719 TextGenerator& generator) const { 1720 // Print primitive repeated field in short form. 1721 PrintFieldName(message, reflection, field, generator); 1722 1723 int size = reflection->FieldSize(message, field); 1724 generator.Print(": ["); 1725 for (int i = 0; i < size; i++) { 1726 if (i > 0) generator.Print(", "); 1727 PrintFieldValue(message, reflection, field, i, generator); 1728 } 1729 if (single_line_mode_) { 1730 generator.Print("] "); 1731 } else { 1732 generator.Print("]\n"); 1733 } 1734 } 1735 PrintFieldName(const Message & message,const Reflection * reflection,const FieldDescriptor * field,TextGenerator & generator) const1736 void TextFormat::Printer::PrintFieldName(const Message& message, 1737 const Reflection* reflection, 1738 const FieldDescriptor* field, 1739 TextGenerator& generator) const { 1740 // if use_field_number_ is true, prints field number instead 1741 // of field name. 1742 if (use_field_number_) { 1743 generator.Print(SimpleItoa(field->number())); 1744 return; 1745 } 1746 1747 const FieldValuePrinter* printer = FindWithDefault( 1748 custom_printers_, field, default_field_value_printer_.get()); 1749 generator.Print(printer->PrintFieldName(message, reflection, field)); 1750 } 1751 PrintFieldValue(const Message & message,const Reflection * reflection,const FieldDescriptor * field,int index,TextGenerator & generator) const1752 void TextFormat::Printer::PrintFieldValue( 1753 const Message& message, 1754 const Reflection* reflection, 1755 const FieldDescriptor* field, 1756 int index, 1757 TextGenerator& generator) const { 1758 GOOGLE_DCHECK(field->is_repeated() || (index == -1)) 1759 << "Index must be -1 for non-repeated fields"; 1760 1761 const FieldValuePrinter* printer 1762 = FindWithDefault(custom_printers_, field, 1763 default_field_value_printer_.get()); 1764 1765 switch (field->cpp_type()) { 1766 #define OUTPUT_FIELD(CPPTYPE, METHOD) \ 1767 case FieldDescriptor::CPPTYPE_##CPPTYPE: \ 1768 generator.Print(printer->Print##METHOD(field->is_repeated() \ 1769 ? reflection->GetRepeated##METHOD(message, field, index) \ 1770 : reflection->Get##METHOD(message, field))); \ 1771 break 1772 1773 OUTPUT_FIELD( INT32, Int32); 1774 OUTPUT_FIELD( INT64, Int64); 1775 OUTPUT_FIELD(UINT32, UInt32); 1776 OUTPUT_FIELD(UINT64, UInt64); 1777 OUTPUT_FIELD( FLOAT, Float); 1778 OUTPUT_FIELD(DOUBLE, Double); 1779 OUTPUT_FIELD( BOOL, Bool); 1780 #undef OUTPUT_FIELD 1781 1782 case FieldDescriptor::CPPTYPE_STRING: { 1783 string scratch; 1784 const string& value = field->is_repeated() 1785 ? reflection->GetRepeatedStringReference( 1786 message, field, index, &scratch) 1787 : reflection->GetStringReference(message, field, &scratch); 1788 const string* value_to_print = &value; 1789 string truncated_value; 1790 if (truncate_string_field_longer_than_ > 0 && 1791 truncate_string_field_longer_than_ < value.size()) { 1792 truncated_value = value.substr(0, truncate_string_field_longer_than_) + 1793 "...<truncated>..."; 1794 value_to_print = &truncated_value; 1795 } 1796 if (field->type() == FieldDescriptor::TYPE_STRING) { 1797 generator.Print(printer->PrintString(*value_to_print)); 1798 } else { 1799 GOOGLE_DCHECK_EQ(field->type(), FieldDescriptor::TYPE_BYTES); 1800 generator.Print(printer->PrintBytes(*value_to_print)); 1801 } 1802 break; 1803 } 1804 1805 case FieldDescriptor::CPPTYPE_ENUM: { 1806 int enum_value = field->is_repeated() 1807 ? reflection->GetRepeatedEnumValue(message, field, index) 1808 : reflection->GetEnumValue(message, field); 1809 const EnumValueDescriptor* enum_desc = 1810 field->enum_type()->FindValueByNumber(enum_value); 1811 if (enum_desc != NULL) { 1812 generator.Print(printer->PrintEnum(enum_value, enum_desc->name())); 1813 } else { 1814 // Ordinarily, enum_desc should not be null, because proto2 has the 1815 // invariant that set enum field values must be in-range, but with the 1816 // new integer-based API for enums (or the RepeatedField<int> loophole), 1817 // it is possible for the user to force an unknown integer value. So we 1818 // simply use the integer value itself as the enum value name in this 1819 // case. 1820 generator.Print(printer->PrintEnum(enum_value, 1821 StringPrintf("%d", enum_value))); 1822 } 1823 break; 1824 } 1825 1826 case FieldDescriptor::CPPTYPE_MESSAGE: 1827 Print(field->is_repeated() 1828 ? reflection->GetRepeatedMessage(message, field, index) 1829 : reflection->GetMessage(message, field), 1830 generator); 1831 break; 1832 } 1833 } 1834 Print(const Message & message,io::ZeroCopyOutputStream * output)1835 /* static */ bool TextFormat::Print(const Message& message, 1836 io::ZeroCopyOutputStream* output) { 1837 return Printer().Print(message, output); 1838 } 1839 PrintUnknownFields(const UnknownFieldSet & unknown_fields,io::ZeroCopyOutputStream * output)1840 /* static */ bool TextFormat::PrintUnknownFields( 1841 const UnknownFieldSet& unknown_fields, 1842 io::ZeroCopyOutputStream* output) { 1843 return Printer().PrintUnknownFields(unknown_fields, output); 1844 } 1845 PrintToString(const Message & message,string * output)1846 /* static */ bool TextFormat::PrintToString( 1847 const Message& message, string* output) { 1848 return Printer().PrintToString(message, output); 1849 } 1850 PrintUnknownFieldsToString(const UnknownFieldSet & unknown_fields,string * output)1851 /* static */ bool TextFormat::PrintUnknownFieldsToString( 1852 const UnknownFieldSet& unknown_fields, string* output) { 1853 return Printer().PrintUnknownFieldsToString(unknown_fields, output); 1854 } 1855 PrintFieldValueToString(const Message & message,const FieldDescriptor * field,int index,string * output)1856 /* static */ void TextFormat::PrintFieldValueToString( 1857 const Message& message, 1858 const FieldDescriptor* field, 1859 int index, 1860 string* output) { 1861 return Printer().PrintFieldValueToString(message, field, index, output); 1862 } 1863 ParseFieldValueFromString(const string & input,const FieldDescriptor * field,Message * message)1864 /* static */ bool TextFormat::ParseFieldValueFromString( 1865 const string& input, 1866 const FieldDescriptor* field, 1867 Message* message) { 1868 return Parser().ParseFieldValueFromString(input, field, message); 1869 } 1870 1871 // Prints an integer as hex with a fixed number of digits dependent on the 1872 // integer type. 1873 template<typename IntType> PaddedHex(IntType value)1874 static string PaddedHex(IntType value) { 1875 string result; 1876 result.reserve(sizeof(value) * 2); 1877 for (int i = sizeof(value) * 2 - 1; i >= 0; i--) { 1878 result.push_back(int_to_hex_digit(value >> (i*4) & 0x0F)); 1879 } 1880 return result; 1881 } 1882 PrintUnknownFields(const UnknownFieldSet & unknown_fields,TextGenerator & generator) const1883 void TextFormat::Printer::PrintUnknownFields( 1884 const UnknownFieldSet& unknown_fields, TextGenerator& generator) const { 1885 for (int i = 0; i < unknown_fields.field_count(); i++) { 1886 const UnknownField& field = unknown_fields.field(i); 1887 string field_number = SimpleItoa(field.number()); 1888 1889 switch (field.type()) { 1890 case UnknownField::TYPE_VARINT: 1891 generator.Print(field_number); 1892 generator.Print(": "); 1893 generator.Print(SimpleItoa(field.varint())); 1894 if (single_line_mode_) { 1895 generator.Print(" "); 1896 } else { 1897 generator.Print("\n"); 1898 } 1899 break; 1900 case UnknownField::TYPE_FIXED32: { 1901 generator.Print(field_number); 1902 generator.Print(": 0x"); 1903 generator.Print( 1904 StrCat(strings::Hex(field.fixed32(), strings::ZERO_PAD_8))); 1905 if (single_line_mode_) { 1906 generator.Print(" "); 1907 } else { 1908 generator.Print("\n"); 1909 } 1910 break; 1911 } 1912 case UnknownField::TYPE_FIXED64: { 1913 generator.Print(field_number); 1914 generator.Print(": 0x"); 1915 generator.Print( 1916 StrCat(strings::Hex(field.fixed64(), strings::ZERO_PAD_16))); 1917 if (single_line_mode_) { 1918 generator.Print(" "); 1919 } else { 1920 generator.Print("\n"); 1921 } 1922 break; 1923 } 1924 case UnknownField::TYPE_LENGTH_DELIMITED: { 1925 generator.Print(field_number); 1926 const string& value = field.length_delimited(); 1927 UnknownFieldSet embedded_unknown_fields; 1928 if (!value.empty() && embedded_unknown_fields.ParseFromString(value)) { 1929 // This field is parseable as a Message. 1930 // So it is probably an embedded message. 1931 if (single_line_mode_) { 1932 generator.Print(" { "); 1933 } else { 1934 generator.Print(" {\n"); 1935 generator.Indent(); 1936 } 1937 PrintUnknownFields(embedded_unknown_fields, generator); 1938 if (single_line_mode_) { 1939 generator.Print("} "); 1940 } else { 1941 generator.Outdent(); 1942 generator.Print("}\n"); 1943 } 1944 } else { 1945 // This field is not parseable as a Message. 1946 // So it is probably just a plain string. 1947 string printed(": \""); 1948 CEscapeAndAppend(value, &printed); 1949 printed.append(single_line_mode_ ? "\" " : "\"\n"); 1950 generator.Print(printed); 1951 } 1952 break; 1953 } 1954 case UnknownField::TYPE_GROUP: 1955 generator.Print(field_number); 1956 if (single_line_mode_) { 1957 generator.Print(" { "); 1958 } else { 1959 generator.Print(" {\n"); 1960 generator.Indent(); 1961 } 1962 PrintUnknownFields(field.group(), generator); 1963 if (single_line_mode_) { 1964 generator.Print("} "); 1965 } else { 1966 generator.Outdent(); 1967 generator.Print("}\n"); 1968 } 1969 break; 1970 } 1971 } 1972 } 1973 1974 } // namespace protobuf 1975 } // namespace google 1976