| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| |
| #ifndef T_GENERATOR_H |
| #define T_GENERATOR_H |
| |
| #include <string> |
| #include <iostream> |
| #include <fstream> |
| #include <sstream> |
| #include "parse/t_program.h" |
| #include "globals.h" |
| #include "t_generator_registry.h" |
| |
| /** |
| * Base class for a thrift code generator. This class defines the basic |
| * routines for code generation and contains the top level method that |
| * dispatches code generation across various components. |
| * |
| */ |
| class t_generator { |
| public: |
| t_generator(t_program* program) { |
| tmp_ = 0; |
| indent_ = 0; |
| program_ = program; |
| program_name_ = get_program_name(program); |
| escape_['\n'] = "\\n"; |
| escape_['\r'] = "\\r"; |
| escape_['\t'] = "\\t"; |
| escape_['"'] = "\\\""; |
| escape_['\\'] = "\\\\"; |
| } |
| |
| virtual ~t_generator() {} |
| |
| /** |
| * Framework generator method that iterates over all the parts of a program |
| * and performs general actions. This is implemented by the base class and |
| * should not normally be overwritten in the subclasses. |
| */ |
| virtual void generate_program(); |
| |
| const t_program* get_program() const { return program_; } |
| |
| void generate_docstring_comment(std::ofstream& out, |
| const std::string& comment_start, |
| const std::string& line_prefix, |
| const std::string& contents, |
| const std::string& comment_end); |
| |
| /** |
| * check whether sub-namespace declaraction is used by generator. |
| * e.g. allow |
| * namespace py.twisted bar |
| * to specify namespace to use when -gen py:twisted is specified. |
| * Will be called with subnamespace, i.e. is_valid_namespace("twisted") |
| * will be called for the above example. |
| */ |
| static bool is_valid_namespace(const std::string& sub_namespace) { |
| (void) sub_namespace; |
| return false; |
| } |
| |
| /** |
| * Escape string to use one in generated sources. |
| */ |
| virtual std::string escape_string(const std::string &in) const; |
| |
| std::string get_escaped_string(t_const_value* constval) { |
| return escape_string(constval->get_string()); |
| } |
| |
| protected: |
| |
| /** |
| * Optional methods that may be imlemented by subclasses to take necessary |
| * steps at the beginning or end of code generation. |
| */ |
| |
| virtual void init_generator() {} |
| virtual void close_generator() {} |
| |
| virtual void generate_consts(std::vector<t_const*> consts); |
| |
| /** |
| * Pure virtual methods implemented by the generator subclasses. |
| */ |
| |
| virtual void generate_typedef (t_typedef* ttypedef) = 0; |
| virtual void generate_enum (t_enum* tenum) = 0; |
| virtual void generate_const (t_const* tconst) { |
| (void) tconst; |
| } |
| virtual void generate_struct (t_struct* tstruct) = 0; |
| virtual void generate_service (t_service* tservice) = 0; |
| virtual void generate_xception (t_struct* txception) { |
| // By default exceptions are the same as structs |
| generate_struct(txception); |
| } |
| |
| /** |
| * Method to get the program name, may be overridden |
| */ |
| virtual std::string get_program_name(t_program* tprogram) { |
| return tprogram->get_name(); |
| } |
| |
| /** |
| * Method to get the service name, may be overridden |
| */ |
| virtual std::string get_service_name(t_service* tservice) { |
| return tservice->get_name(); |
| } |
| |
| /** |
| * Get the current output directory |
| */ |
| virtual std::string get_out_dir() const { |
| if (program_->is_out_path_absolute()) { |
| return program_->get_out_path() + "/"; |
| } |
| |
| return program_->get_out_path() + out_dir_base_ + "/"; |
| } |
| |
| /** |
| * Creates a unique temporary variable name, which is just "name" with a |
| * number appended to it (i.e. name35) |
| */ |
| std::string tmp(std::string name) { |
| std::ostringstream out; |
| out << name << tmp_++; |
| return out.str(); |
| } |
| |
| /** |
| * Indentation level modifiers |
| */ |
| |
| void indent_up(){ |
| ++indent_; |
| } |
| |
| void indent_down() { |
| --indent_; |
| } |
| |
| /** |
| * Indentation print function |
| */ |
| std::string indent() { |
| std::string ind = ""; |
| int i; |
| for (i = 0; i < indent_; ++i) { |
| ind += " "; |
| } |
| return ind; |
| } |
| |
| /** |
| * Indentation utility wrapper |
| */ |
| std::ostream& indent(std::ostream &os) { |
| return os << indent(); |
| } |
| |
| /** |
| * Capitalization helpers |
| */ |
| std::string capitalize(std::string in) { |
| in[0] = toupper(in[0]); |
| return in; |
| } |
| std::string decapitalize(std::string in) { |
| in[0] = tolower(in[0]); |
| return in; |
| } |
| std::string lowercase(std::string in) { |
| for (size_t i = 0; i < in.size(); ++i) { |
| in[i] = tolower(in[i]); |
| } |
| return in; |
| } |
| /** |
| * Transforms a camel case string to an equivalent one separated by underscores |
| * e.g. aMultiWord -> a_multi_word |
| * someName -> some_name |
| * CamelCase -> camel_case |
| * name -> name |
| * Name -> name |
| */ |
| std::string underscore(std::string in) { |
| in[0] = tolower(in[0]); |
| for (size_t i = 1; i < in.size(); ++i) { |
| if (isupper(in[i])) { |
| in[i] = tolower(in[i]); |
| in.insert(i, "_"); |
| } |
| } |
| return in; |
| } |
| /** |
| * Transforms a string with words separated by underscores to a camel case equivalent |
| * e.g. a_multi_word -> aMultiWord |
| * some_name -> someName |
| * name -> name |
| */ |
| std::string camelcase(std::string in) { |
| std::ostringstream out; |
| bool underscore = false; |
| |
| for (size_t i = 0; i < in.size(); i++) { |
| if (in[i] == '_') { |
| underscore = true; |
| continue; |
| } |
| if (underscore) { |
| out << (char) toupper(in[i]); |
| underscore = false; |
| continue; |
| } |
| out << in[i]; |
| } |
| |
| return out.str(); |
| } |
| |
| /** |
| * Get the true type behind a series of typedefs. |
| */ |
| static t_type* get_true_type(t_type* type) { |
| return type->get_true_type(); |
| } |
| |
| protected: |
| /** |
| * The program being generated |
| */ |
| t_program* program_; |
| |
| /** |
| * Quick accessor for formatted program name that is currently being |
| * generated. |
| */ |
| std::string program_name_; |
| |
| /** |
| * Quick accessor for formatted service name that is currently being |
| * generated. |
| */ |
| std::string service_name_; |
| |
| /** |
| * Output type-specifc directory name ("gen-*") |
| */ |
| std::string out_dir_base_; |
| |
| /** |
| * Map of characters to escape in string literals. |
| */ |
| std::map<char, std::string> escape_; |
| |
| private: |
| /** |
| * Current code indentation level |
| */ |
| int indent_; |
| |
| /** |
| * Temporary variable counter, for making unique variable names |
| */ |
| int tmp_; |
| }; |
| |
| #endif |