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::__anon395507550311::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