parser.h 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522
  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. // Author: kenton@google.com (Kenton Varda)
  31. // Based on original Protocol Buffers design by
  32. // Sanjay Ghemawat, Jeff Dean, and others.
  33. //
  34. // Implements parsing of .proto files to FileDescriptorProtos.
  35. #ifndef GOOGLE_PROTOBUF_COMPILER_PARSER_H__
  36. #define GOOGLE_PROTOBUF_COMPILER_PARSER_H__
  37. #include <map>
  38. #include <string>
  39. #include <utility>
  40. #include <google/protobuf/descriptor.h>
  41. #include <google/protobuf/descriptor.pb.h>
  42. #include <google/protobuf/repeated_field.h>
  43. #include <google/protobuf/io/tokenizer.h>
  44. namespace google {
  45. namespace protobuf { class Message; }
  46. namespace protobuf {
  47. namespace compiler {
  48. // Defined in this file.
  49. class Parser;
  50. class SourceLocationTable;
  51. // Implements parsing of protocol definitions (such as .proto files).
  52. //
  53. // Note that most users will be more interested in the Importer class.
  54. // Parser is a lower-level class which simply converts a single .proto file
  55. // to a FileDescriptorProto. It does not resolve import directives or perform
  56. // many other kinds of validation needed to construct a complete
  57. // FileDescriptor.
  58. class LIBPROTOBUF_EXPORT Parser {
  59. public:
  60. Parser();
  61. ~Parser();
  62. // Parse the entire input and construct a FileDescriptorProto representing
  63. // it. Returns true if no errors occurred, false otherwise.
  64. bool Parse(io::Tokenizer* input, FileDescriptorProto* file);
  65. // Optional fetaures:
  66. // DEPRECATED: New code should use the SourceCodeInfo embedded in the
  67. // FileDescriptorProto.
  68. //
  69. // Requests that locations of certain definitions be recorded to the given
  70. // SourceLocationTable while parsing. This can be used to look up exact line
  71. // and column numbers for errors reported by DescriptorPool during validation.
  72. // Set to NULL (the default) to discard source location information.
  73. void RecordSourceLocationsTo(SourceLocationTable* location_table) {
  74. source_location_table_ = location_table;
  75. }
  76. // Requests that errors be recorded to the given ErrorCollector while
  77. // parsing. Set to NULL (the default) to discard error messages.
  78. void RecordErrorsTo(io::ErrorCollector* error_collector) {
  79. error_collector_ = error_collector;
  80. }
  81. // Returns the identifier used in the "syntax = " declaration, if one was
  82. // seen during the last call to Parse(), or the empty string otherwise.
  83. const string& GetSyntaxIdentifier() { return syntax_identifier_; }
  84. // If set true, input files will be required to begin with a syntax
  85. // identifier. Otherwise, files may omit this. If a syntax identifier
  86. // is provided, it must be 'syntax = "proto2";' and must appear at the
  87. // top of this file regardless of whether or not it was required.
  88. void SetRequireSyntaxIdentifier(bool value) {
  89. require_syntax_identifier_ = value;
  90. }
  91. // Call SetStopAfterSyntaxIdentifier(true) to tell the parser to stop
  92. // parsing as soon as it has seen the syntax identifier, or lack thereof.
  93. // This is useful for quickly identifying the syntax of the file without
  94. // parsing the whole thing. If this is enabled, no error will be recorded
  95. // if the syntax identifier is something other than "proto2" (since
  96. // presumably the caller intends to deal with that), but other kinds of
  97. // errors (e.g. parse errors) will still be reported. When this is enabled,
  98. // you may pass a NULL FileDescriptorProto to Parse().
  99. void SetStopAfterSyntaxIdentifier(bool value) {
  100. stop_after_syntax_identifier_ = value;
  101. }
  102. private:
  103. class LocationRecorder;
  104. // =================================================================
  105. // Error recovery helpers
  106. // Consume the rest of the current statement. This consumes tokens
  107. // until it sees one of:
  108. // ';' Consumes the token and returns.
  109. // '{' Consumes the brace then calls SkipRestOfBlock().
  110. // '}' Returns without consuming.
  111. // EOF Returns (can't consume).
  112. // The Parser often calls SkipStatement() after encountering a syntax
  113. // error. This allows it to go on parsing the following lines, allowing
  114. // it to report more than just one error in the file.
  115. void SkipStatement();
  116. // Consume the rest of the current block, including nested blocks,
  117. // ending after the closing '}' is encountered and consumed, or at EOF.
  118. void SkipRestOfBlock();
  119. // -----------------------------------------------------------------
  120. // Single-token consuming helpers
  121. //
  122. // These make parsing code more readable.
  123. // True if the current token is TYPE_END.
  124. inline bool AtEnd();
  125. // True if the next token matches the given text.
  126. inline bool LookingAt(const char* text);
  127. // True if the next token is of the given type.
  128. inline bool LookingAtType(io::Tokenizer::TokenType token_type);
  129. // If the next token exactly matches the text given, consume it and return
  130. // true. Otherwise, return false without logging an error.
  131. bool TryConsume(const char* text);
  132. // These attempt to read some kind of token from the input. If successful,
  133. // they return true. Otherwise they return false and add the given error
  134. // to the error list.
  135. // Consume a token with the exact text given.
  136. bool Consume(const char* text, const char* error);
  137. // Same as above, but automatically generates the error "Expected \"text\".",
  138. // where "text" is the expected token text.
  139. bool Consume(const char* text);
  140. // Consume a token of type IDENTIFIER and store its text in "output".
  141. bool ConsumeIdentifier(string* output, const char* error);
  142. // Consume an integer and store its value in "output".
  143. bool ConsumeInteger(int* output, const char* error);
  144. // Consume a signed integer and store its value in "output".
  145. bool ConsumeSignedInteger(int* output, const char* error);
  146. // Consume a 64-bit integer and store its value in "output". If the value
  147. // is greater than max_value, an error will be reported.
  148. bool ConsumeInteger64(uint64 max_value, uint64* output, const char* error);
  149. // Consume a number and store its value in "output". This will accept
  150. // tokens of either INTEGER or FLOAT type.
  151. bool ConsumeNumber(double* output, const char* error);
  152. // Consume a string literal and store its (unescaped) value in "output".
  153. bool ConsumeString(string* output, const char* error);
  154. // Consume a token representing the end of the statement. Comments between
  155. // this token and the next will be harvested for documentation. The given
  156. // LocationRecorder should refer to the declaration that was just parsed;
  157. // it will be populated with these comments.
  158. //
  159. // TODO(kenton): The LocationRecorder is const because historically locations
  160. // have been passed around by const reference, for no particularly good
  161. // reason. We should probably go through and change them all to mutable
  162. // pointer to make this more intuitive.
  163. bool TryConsumeEndOfDeclaration(const char* text,
  164. const LocationRecorder* location);
  165. bool ConsumeEndOfDeclaration(const char* text,
  166. const LocationRecorder* location);
  167. // -----------------------------------------------------------------
  168. // Error logging helpers
  169. // Invokes error_collector_->AddError(), if error_collector_ is not NULL.
  170. void AddError(int line, int column, const string& error);
  171. // Invokes error_collector_->AddError() with the line and column number
  172. // of the current token.
  173. void AddError(const string& error);
  174. // Records a location in the SourceCodeInfo.location table (see
  175. // descriptor.proto). We use RAII to ensure that the start and end locations
  176. // are recorded -- the constructor records the start location and the
  177. // destructor records the end location. Since the parser is
  178. // recursive-descent, this works out beautifully.
  179. class LIBPROTOBUF_EXPORT LocationRecorder {
  180. public:
  181. // Construct the file's "root" location.
  182. LocationRecorder(Parser* parser);
  183. // Construct a location that represents a declaration nested within the
  184. // given parent. E.g. a field's location is nested within the location
  185. // for a message type. The parent's path will be copied, so you should
  186. // call AddPath() only to add the path components leading from the parent
  187. // to the child (as opposed to leading from the root to the child).
  188. LocationRecorder(const LocationRecorder& parent);
  189. // Convenience constructors that call AddPath() one or two times.
  190. LocationRecorder(const LocationRecorder& parent, int path1);
  191. LocationRecorder(const LocationRecorder& parent, int path1, int path2);
  192. ~LocationRecorder();
  193. // Add a path component. See SourceCodeInfo.Location.path in
  194. // descriptor.proto.
  195. void AddPath(int path_component);
  196. // By default the location is considered to start at the current token at
  197. // the time the LocationRecorder is created. StartAt() sets the start
  198. // location to the given token instead.
  199. void StartAt(const io::Tokenizer::Token& token);
  200. // Start at the same location as some other LocationRecorder.
  201. void StartAt(const LocationRecorder& other);
  202. // By default the location is considered to end at the previous token at
  203. // the time the LocationRecorder is destroyed. EndAt() sets the end
  204. // location to the given token instead.
  205. void EndAt(const io::Tokenizer::Token& token);
  206. // Records the start point of this location to the SourceLocationTable that
  207. // was passed to RecordSourceLocationsTo(), if any. SourceLocationTable
  208. // is an older way of keeping track of source locations which is still
  209. // used in some places.
  210. void RecordLegacyLocation(const Message* descriptor,
  211. DescriptorPool::ErrorCollector::ErrorLocation location);
  212. // Attaches leading and trailing comments to the location. The two strings
  213. // will be swapped into place, so after this is called *leading and
  214. // *trailing will be empty.
  215. //
  216. // TODO(kenton): See comment on TryConsumeEndOfDeclaration(), above, for
  217. // why this is const.
  218. void AttachComments(string* leading, string* trailing) const;
  219. private:
  220. Parser* parser_;
  221. SourceCodeInfo::Location* location_;
  222. void Init(const LocationRecorder& parent);
  223. };
  224. // =================================================================
  225. // Parsers for various language constructs
  226. // Parses the "syntax = \"proto2\";" line at the top of the file. Returns
  227. // false if it failed to parse or if the syntax identifier was not
  228. // recognized.
  229. bool ParseSyntaxIdentifier();
  230. // These methods parse various individual bits of code. They return
  231. // false if they completely fail to parse the construct. In this case,
  232. // it is probably necessary to skip the rest of the statement to recover.
  233. // However, if these methods return true, it does NOT mean that there
  234. // were no errors; only that there were no *syntax* errors. For instance,
  235. // if a service method is defined using proper syntax but uses a primitive
  236. // type as its input or output, ParseMethodField() still returns true
  237. // and only reports the error by calling AddError(). In practice, this
  238. // makes logic much simpler for the caller.
  239. // Parse a top-level message, enum, service, etc.
  240. bool ParseTopLevelStatement(FileDescriptorProto* file,
  241. const LocationRecorder& root_location);
  242. // Parse various language high-level language construrcts.
  243. bool ParseMessageDefinition(DescriptorProto* message,
  244. const LocationRecorder& message_location,
  245. const FileDescriptorProto* containing_file);
  246. bool ParseEnumDefinition(EnumDescriptorProto* enum_type,
  247. const LocationRecorder& enum_location,
  248. const FileDescriptorProto* containing_file);
  249. bool ParseServiceDefinition(ServiceDescriptorProto* service,
  250. const LocationRecorder& service_location,
  251. const FileDescriptorProto* containing_file);
  252. bool ParsePackage(FileDescriptorProto* file,
  253. const LocationRecorder& root_location,
  254. const FileDescriptorProto* containing_file);
  255. bool ParseImport(RepeatedPtrField<string>* dependency,
  256. RepeatedField<int32>* public_dependency,
  257. RepeatedField<int32>* weak_dependency,
  258. const LocationRecorder& root_location,
  259. const FileDescriptorProto* containing_file);
  260. bool ParseOption(Message* options,
  261. const LocationRecorder& options_location,
  262. const FileDescriptorProto* containing_file);
  263. // These methods parse the contents of a message, enum, or service type and
  264. // add them to the given object. They consume the entire block including
  265. // the beginning and ending brace.
  266. bool ParseMessageBlock(DescriptorProto* message,
  267. const LocationRecorder& message_location,
  268. const FileDescriptorProto* containing_file);
  269. bool ParseEnumBlock(EnumDescriptorProto* enum_type,
  270. const LocationRecorder& enum_location,
  271. const FileDescriptorProto* containing_file);
  272. bool ParseServiceBlock(ServiceDescriptorProto* service,
  273. const LocationRecorder& service_location,
  274. const FileDescriptorProto* containing_file);
  275. // Parse one statement within a message, enum, or service block, inclunding
  276. // final semicolon.
  277. bool ParseMessageStatement(DescriptorProto* message,
  278. const LocationRecorder& message_location,
  279. const FileDescriptorProto* containing_file);
  280. bool ParseEnumStatement(EnumDescriptorProto* message,
  281. const LocationRecorder& enum_location,
  282. const FileDescriptorProto* containing_file);
  283. bool ParseServiceStatement(ServiceDescriptorProto* message,
  284. const LocationRecorder& service_location,
  285. const FileDescriptorProto* containing_file);
  286. // Parse a field of a message. If the field is a group, its type will be
  287. // added to "messages".
  288. //
  289. // parent_location and location_field_number_for_nested_type are needed when
  290. // parsing groups -- we need to generate a nested message type within the
  291. // parent and record its location accordingly. Since the parent could be
  292. // either a FileDescriptorProto or a DescriptorProto, we must pass in the
  293. // correct field number to use.
  294. bool ParseMessageField(FieldDescriptorProto* field,
  295. RepeatedPtrField<DescriptorProto>* messages,
  296. const LocationRecorder& parent_location,
  297. int location_field_number_for_nested_type,
  298. const LocationRecorder& field_location,
  299. const FileDescriptorProto* containing_file);
  300. // Like ParseMessageField() but expects the label has already been filled in
  301. // by the caller.
  302. bool ParseMessageFieldNoLabel(FieldDescriptorProto* field,
  303. RepeatedPtrField<DescriptorProto>* messages,
  304. const LocationRecorder& parent_location,
  305. int location_field_number_for_nested_type,
  306. const LocationRecorder& field_location,
  307. const FileDescriptorProto* containing_file);
  308. // Parse an "extensions" declaration.
  309. bool ParseExtensions(DescriptorProto* message,
  310. const LocationRecorder& extensions_location,
  311. const FileDescriptorProto* containing_file);
  312. // Parse an "extend" declaration. (See also comments for
  313. // ParseMessageField().)
  314. bool ParseExtend(RepeatedPtrField<FieldDescriptorProto>* extensions,
  315. RepeatedPtrField<DescriptorProto>* messages,
  316. const LocationRecorder& parent_location,
  317. int location_field_number_for_nested_type,
  318. const LocationRecorder& extend_location,
  319. const FileDescriptorProto* containing_file);
  320. // Parse a "oneof" declaration. The caller is responsible for setting
  321. // oneof_decl->label() since it will have had to parse the label before it
  322. // knew it was parsing a oneof.
  323. bool ParseOneof(OneofDescriptorProto* oneof_decl,
  324. DescriptorProto* containing_type,
  325. int oneof_index,
  326. const LocationRecorder& oneof_location,
  327. const LocationRecorder& containing_type_location,
  328. const FileDescriptorProto* containing_file);
  329. // Parse a single enum value within an enum block.
  330. bool ParseEnumConstant(EnumValueDescriptorProto* enum_value,
  331. const LocationRecorder& enum_value_location,
  332. const FileDescriptorProto* containing_file);
  333. // Parse enum constant options, i.e. the list in square brackets at the end
  334. // of the enum constant value definition.
  335. bool ParseEnumConstantOptions(EnumValueDescriptorProto* value,
  336. const LocationRecorder& enum_value_location,
  337. const FileDescriptorProto* containing_file);
  338. // Parse a single method within a service definition.
  339. bool ParseServiceMethod(MethodDescriptorProto* method,
  340. const LocationRecorder& method_location,
  341. const FileDescriptorProto* containing_file);
  342. // Parse options of a single method or stream.
  343. bool ParseOptions(const LocationRecorder& parent_location,
  344. const FileDescriptorProto* containing_file,
  345. const int optionsFieldNumber,
  346. Message* mutable_options);
  347. // Parse "required", "optional", or "repeated" and fill in "label"
  348. // with the value.
  349. bool ParseLabel(FieldDescriptorProto::Label* label,
  350. const FileDescriptorProto* containing_file);
  351. // Parse a type name and fill in "type" (if it is a primitive) or
  352. // "type_name" (if it is not) with the type parsed.
  353. bool ParseType(FieldDescriptorProto::Type* type,
  354. string* type_name);
  355. // Parse a user-defined type and fill in "type_name" with the name.
  356. // If a primitive type is named, it is treated as an error.
  357. bool ParseUserDefinedType(string* type_name);
  358. // Parses field options, i.e. the stuff in square brackets at the end
  359. // of a field definition. Also parses default value.
  360. bool ParseFieldOptions(FieldDescriptorProto* field,
  361. const LocationRecorder& field_location,
  362. const FileDescriptorProto* containing_file);
  363. // Parse the "default" option. This needs special handling because its
  364. // type is the field's type.
  365. bool ParseDefaultAssignment(FieldDescriptorProto* field,
  366. const LocationRecorder& field_location,
  367. const FileDescriptorProto* containing_file);
  368. enum OptionStyle {
  369. OPTION_ASSIGNMENT, // just "name = value"
  370. OPTION_STATEMENT // "option name = value;"
  371. };
  372. // Parse a single option name/value pair, e.g. "ctype = CORD". The name
  373. // identifies a field of the given Message, and the value of that field
  374. // is set to the parsed value.
  375. bool ParseOption(Message* options,
  376. const LocationRecorder& options_location,
  377. const FileDescriptorProto* containing_file,
  378. OptionStyle style);
  379. // Parses a single part of a multipart option name. A multipart name consists
  380. // of names separated by dots. Each name is either an identifier or a series
  381. // of identifiers separated by dots and enclosed in parentheses. E.g.,
  382. // "foo.(bar.baz).qux".
  383. bool ParseOptionNamePart(UninterpretedOption* uninterpreted_option,
  384. const LocationRecorder& part_location,
  385. const FileDescriptorProto* containing_file);
  386. // Parses a string surrounded by balanced braces. Strips off the outer
  387. // braces and stores the enclosed string in *value.
  388. // E.g.,
  389. // { foo } *value gets 'foo'
  390. // { foo { bar: box } } *value gets 'foo { bar: box }'
  391. // {} *value gets ''
  392. //
  393. // REQUIRES: LookingAt("{")
  394. // When finished successfully, we are looking at the first token past
  395. // the ending brace.
  396. bool ParseUninterpretedBlock(string* value);
  397. // =================================================================
  398. io::Tokenizer* input_;
  399. io::ErrorCollector* error_collector_;
  400. SourceCodeInfo* source_code_info_;
  401. SourceLocationTable* source_location_table_; // legacy
  402. bool had_errors_;
  403. bool require_syntax_identifier_;
  404. bool stop_after_syntax_identifier_;
  405. string syntax_identifier_;
  406. // Leading doc comments for the next declaration. These are not complete
  407. // yet; use ConsumeEndOfDeclaration() to get the complete comments.
  408. string upcoming_doc_comments_;
  409. GOOGLE_DISALLOW_EVIL_CONSTRUCTORS(Parser);
  410. };
  411. // A table mapping (descriptor, ErrorLocation) pairs -- as reported by
  412. // DescriptorPool when validating descriptors -- to line and column numbers
  413. // within the original source code.
  414. //
  415. // This is semi-obsolete: FileDescriptorProto.source_code_info now contains
  416. // far more complete information about source locations. However, as of this
  417. // writing you still need to use SourceLocationTable when integrating with
  418. // DescriptorPool.
  419. class LIBPROTOBUF_EXPORT SourceLocationTable {
  420. public:
  421. SourceLocationTable();
  422. ~SourceLocationTable();
  423. // Finds the precise location of the given error and fills in *line and
  424. // *column with the line and column numbers. If not found, sets *line to
  425. // -1 and *column to 0 (since line = -1 is used to mean "error has no exact
  426. // location" in the ErrorCollector interface). Returns true if found, false
  427. // otherwise.
  428. bool Find(const Message* descriptor,
  429. DescriptorPool::ErrorCollector::ErrorLocation location,
  430. int* line, int* column) const;
  431. // Adds a location to the table.
  432. void Add(const Message* descriptor,
  433. DescriptorPool::ErrorCollector::ErrorLocation location,
  434. int line, int column);
  435. // Clears the contents of the table.
  436. void Clear();
  437. private:
  438. typedef map<
  439. pair<const Message*, DescriptorPool::ErrorCollector::ErrorLocation>,
  440. pair<int, int> > LocationMap;
  441. LocationMap location_map_;
  442. };
  443. } // namespace compiler
  444. } // namespace protobuf
  445. } // namespace google
  446. #endif // GOOGLE_PROTOBUF_COMPILER_PARSER_H__