|  | /* | 
|  | * 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_PROGRAM_H | 
|  | #define T_PROGRAM_H | 
|  |  | 
|  | #include <map> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | // For program_name() | 
|  | #include "main.h" | 
|  |  | 
|  | #include "t_doc.h" | 
|  | #include "t_scope.h" | 
|  | #include "t_base_type.h" | 
|  | #include "t_typedef.h" | 
|  | #include "t_enum.h" | 
|  | #include "t_const.h" | 
|  | #include "t_struct.h" | 
|  | #include "t_service.h" | 
|  | #include "t_list.h" | 
|  | #include "t_map.h" | 
|  | #include "t_set.h" | 
|  | #include "generate/t_generator_registry.h" | 
|  | //#include "t_doc.h" | 
|  |  | 
|  | /** | 
|  | * Top level class representing an entire thrift program. A program consists | 
|  | * fundamentally of the following: | 
|  | * | 
|  | *   Typedefs | 
|  | *   Enumerations | 
|  | *   Constants | 
|  | *   Structs | 
|  | *   Exceptions | 
|  | *   Services | 
|  | * | 
|  | * The program module also contains the definitions of the base types. | 
|  | * | 
|  | */ | 
|  | class t_program : public t_doc { | 
|  | public: | 
|  | t_program(std::string path, std::string name) : | 
|  | path_(path), | 
|  | name_(name), | 
|  | out_path_("./"), | 
|  | out_path_is_absolute_(false) { | 
|  | scope_ = new t_scope(); | 
|  | } | 
|  |  | 
|  | t_program(std::string path) : | 
|  | path_(path), | 
|  | out_path_("./"), | 
|  | out_path_is_absolute_(false) { | 
|  | name_ = program_name(path); | 
|  | scope_ = new t_scope(); | 
|  | } | 
|  |  | 
|  | ~t_program() | 
|  | { | 
|  | if(scope_) | 
|  | { | 
|  | delete scope_; | 
|  | scope_ = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Path accessor | 
|  | const std::string& get_path() const { return path_; } | 
|  |  | 
|  | // Output path accessor | 
|  | const std::string& get_out_path() const { return out_path_; } | 
|  |  | 
|  | // Create gen-* dir accessor | 
|  | const bool is_out_path_absolute() const { return out_path_is_absolute_; } | 
|  |  | 
|  | // Name accessor | 
|  | const std::string& get_name() const { return name_; } | 
|  |  | 
|  | // Namespace | 
|  | const std::string& get_namespace() const { return namespace_; } | 
|  |  | 
|  | // Include prefix accessor | 
|  | const std::string& get_include_prefix() const { return include_prefix_; } | 
|  |  | 
|  | // Accessors for program elements | 
|  | const std::vector<t_typedef*>& get_typedefs()  const { return typedefs_;  } | 
|  | const std::vector<t_enum*>&    get_enums()     const { return enums_;     } | 
|  | const std::vector<t_const*>&   get_consts()    const { return consts_;    } | 
|  | const std::vector<t_struct*>&  get_structs()   const { return structs_;   } | 
|  | const std::vector<t_struct*>&  get_xceptions() const { return xceptions_; } | 
|  | const std::vector<t_struct*>&  get_objects()   const { return objects_;   } | 
|  | const std::vector<t_service*>& get_services()  const { return services_;  } | 
|  |  | 
|  | // Program elements | 
|  | void add_typedef  (t_typedef* td) { typedefs_.push_back(td);  } | 
|  | void add_enum     (t_enum*    te) { enums_.push_back(te);     } | 
|  | void add_const    (t_const*   tc) { consts_.push_back(tc);    } | 
|  | void add_struct   (t_struct*  ts) { objects_.push_back(ts); | 
|  | structs_.push_back(ts);   } | 
|  | void add_xception (t_struct*  tx) { objects_.push_back(tx); | 
|  | xceptions_.push_back(tx); } | 
|  | void add_service  (t_service* ts) { services_.push_back(ts);  } | 
|  |  | 
|  | // Programs to include | 
|  | const std::vector<t_program*>& get_includes() const { return includes_; } | 
|  |  | 
|  | void set_out_path(std::string out_path, bool out_path_is_absolute) { | 
|  | out_path_ = out_path; | 
|  | out_path_is_absolute_ = out_path_is_absolute; | 
|  | // Ensure that it ends with a trailing '/' (or '\' for windows machines) | 
|  | char c = out_path_.at(out_path_.size() - 1); | 
|  | if (!(c == '/' || c == '\\')) { | 
|  | out_path_.push_back('/'); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Typename collision detection | 
|  | /** | 
|  | * Search for typename collisions | 
|  | * @param t    the type to test for collisions | 
|  | * @return     true if a certain collision was found, otherwise false | 
|  | */ | 
|  | bool is_unique_typename(t_type * t) { | 
|  | int occurances = program_typename_count(this, t); | 
|  | for (std::vector<t_program*>::iterator it = includes_.begin(); | 
|  | it != includes_.end(); ++it) { | 
|  | occurances += program_typename_count(*it, t); | 
|  | } | 
|  | return 0 == occurances; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Search all type collections for duplicate typenames | 
|  | * @param prog the program to search | 
|  | * @param t    the type to test for collisions | 
|  | * @return     the number of certain typename collisions | 
|  | */ | 
|  | int program_typename_count(t_program * prog, t_type * t) { | 
|  | int occurances = 0; | 
|  | occurances += collection_typename_count(prog, prog->typedefs_, t); | 
|  | occurances += collection_typename_count(prog, prog->enums_, t); | 
|  | occurances += collection_typename_count(prog, prog->objects_, t); | 
|  | occurances += collection_typename_count(prog, prog->services_, t); | 
|  | return occurances; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Search a type collection for duplicate typenames | 
|  | * @param prog            the program to search | 
|  | * @param type_collection the type collection to search | 
|  | * @param t               the type to test for collisions | 
|  | * @return                the number of certain typename collisions | 
|  | */ | 
|  | template <class T> | 
|  | int collection_typename_count(t_program * prog, T type_collection, t_type * t) { | 
|  | int occurances = 0; | 
|  | for (typename T::iterator it = type_collection.begin(); it != type_collection.end(); ++it) | 
|  | if (t != *it && 0 == t->get_name().compare((*it)->get_name()) && is_common_namespace(prog, t)) | 
|  | ++occurances; | 
|  | return occurances; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Determine whether identical typenames will collide based on namespaces. | 
|  | * | 
|  | * Because we do not know which languages the user will generate code for, | 
|  | * collisions within programs (IDL files) having namespace declarations can be | 
|  | * difficult to determine. Only guaranteed collisions return true (cause an error). | 
|  | * Possible collisions involving explicit namespace declarations produce a warning. | 
|  | * Other possible collisions go unreported. | 
|  | * @param prog the program containing the preexisting typename | 
|  | * @param t    the type containing the typename match | 
|  | * @return     true if a collision within namespaces is found, otherwise false | 
|  | */ | 
|  | bool is_common_namespace(t_program * prog, t_type * t) { | 
|  | //Case 1: Typenames are in the same program [collision] | 
|  | if (prog == t->get_program()) { | 
|  | pwarning(1, "Duplicate typename %s found in %s", | 
|  | t->get_name().c_str(), t->get_program()->get_name().c_str()); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | //Case 2: Both programs have identical namespace scope/name declarations [collision] | 
|  | bool match = true; | 
|  | for (std::map<std::string, std::string>::iterator it = prog->namespaces_.begin(); | 
|  | it != prog->namespaces_.end(); ++it) { | 
|  | if (0 == it->second.compare(t->get_program()->get_namespace(it->first))) { | 
|  | pwarning(1, "Duplicate typename %s found in %s,%s,%s and %s,%s,%s [file,scope,ns]", | 
|  | t->get_name().c_str(), | 
|  | t->get_program()->get_name().c_str(), it->first.c_str(), it->second.c_str(), | 
|  | prog->get_name().c_str(), it->first.c_str(), it->second.c_str()); | 
|  | } else { | 
|  | match = false; | 
|  | } | 
|  | } | 
|  | for (std::map<std::string, std::string>::iterator it = t->get_program()->namespaces_.begin(); | 
|  | it != t->get_program()->namespaces_.end(); ++it) { | 
|  | if (0 == it->second.compare(prog->get_namespace(it->first))) { | 
|  | pwarning(1, "Duplicate typename %s found in %s,%s,%s and %s,%s,%s [file,scope,ns]", | 
|  | t->get_name().c_str(), | 
|  | t->get_program()->get_name().c_str(), it->first.c_str(), it->second.c_str(), | 
|  | prog->get_name().c_str(), it->first.c_str(), it->second.c_str()); | 
|  | } else { | 
|  | match = false; | 
|  | } | 
|  | } | 
|  | if (0 == prog->namespaces_.size() && 0 == t->get_program()->namespaces_.size()) { | 
|  | pwarning(1, "Duplicate typename %s found in %s and %s", | 
|  | t->get_name().c_str(), t->get_program()->get_name().c_str(),  prog->get_name().c_str()); | 
|  | } | 
|  | return match; | 
|  | } | 
|  |  | 
|  | // Scoping and namespacing | 
|  | void set_namespace(std::string name) { | 
|  | namespace_ = name; | 
|  | } | 
|  |  | 
|  | // Scope accessor | 
|  | t_scope* scope() { | 
|  | return scope_; | 
|  | } | 
|  |  | 
|  | // Includes | 
|  |  | 
|  | void add_include(std::string path, std::string include_site) { | 
|  | t_program* program = new t_program(path); | 
|  |  | 
|  | // include prefix for this program is the site at which it was included | 
|  | // (minus the filename) | 
|  | std::string include_prefix; | 
|  | std::string::size_type last_slash = std::string::npos; | 
|  | if ((last_slash = include_site.rfind("/")) != std::string::npos) { | 
|  | include_prefix = include_site.substr(0, last_slash); | 
|  | } | 
|  |  | 
|  | program->set_include_prefix(include_prefix); | 
|  | includes_.push_back(program); | 
|  | } | 
|  |  | 
|  | std::vector<t_program*>& get_includes() { | 
|  | return includes_; | 
|  | } | 
|  |  | 
|  | void set_include_prefix(std::string include_prefix) { | 
|  | include_prefix_ = include_prefix; | 
|  |  | 
|  | // this is intended to be a directory; add a trailing slash if necessary | 
|  | int len = include_prefix_.size(); | 
|  | if (len > 0 && include_prefix_[len - 1] != '/') { | 
|  | include_prefix_ += '/'; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Language neutral namespace / packaging | 
|  | void set_namespace(std::string language, std::string name_space) { | 
|  | if (language != "*") { | 
|  | size_t sub_index = language.find('.'); | 
|  | std::string base_language = language.substr(0, sub_index); | 
|  | std::string sub_namespace; | 
|  |  | 
|  | if(base_language == "smalltalk") { | 
|  | pwarning(1, "Namespace 'smalltalk' is deprecated. Use 'st' instead"); | 
|  | base_language = "st"; | 
|  | } | 
|  |  | 
|  | t_generator_registry::gen_map_t my_copy = t_generator_registry::get_generator_map(); | 
|  |  | 
|  | t_generator_registry::gen_map_t::iterator it; | 
|  | it=my_copy.find(base_language); | 
|  |  | 
|  | if (it == my_copy.end()) { | 
|  | std::string warning = "No generator named '" + base_language + "' could be found!"; | 
|  | pwarning(1, warning.c_str()); | 
|  | } else { | 
|  | if (sub_index != std::string::npos) { | 
|  | std::string sub_namespace = language.substr(sub_index+1); | 
|  | if ( ! it->second->is_valid_namespace(sub_namespace)) { | 
|  | std::string warning = base_language + " generator does not accept '" + sub_namespace + "' as sub-namespace!"; | 
|  | pwarning(1, warning.c_str()); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | namespaces_[language] = name_space; | 
|  | } | 
|  |  | 
|  | std::string get_namespace(std::string language) const { | 
|  | std::map<std::string, std::string>::const_iterator iter; | 
|  | if ((iter = namespaces_.find(language)) != namespaces_.end() || | 
|  | (iter = namespaces_.find("*"     )) != namespaces_.end()) { | 
|  | return iter->second; | 
|  | } | 
|  | return std::string(); | 
|  | } | 
|  |  | 
|  | // Language specific namespace / packaging | 
|  |  | 
|  | void add_cpp_include(std::string path) { | 
|  | cpp_includes_.push_back(path); | 
|  | } | 
|  |  | 
|  | const std::vector<std::string>& get_cpp_includes() { | 
|  | return cpp_includes_; | 
|  | } | 
|  |  | 
|  | void add_c_include(std::string path) { | 
|  | c_includes_.push_back(path); | 
|  | } | 
|  |  | 
|  | const std::vector<std::string>& get_c_includes() { | 
|  | return c_includes_; | 
|  | } | 
|  |  | 
|  | private: | 
|  |  | 
|  | // File path | 
|  | std::string path_; | 
|  |  | 
|  | // Name | 
|  | std::string name_; | 
|  |  | 
|  | // Output directory | 
|  | std::string out_path_; | 
|  |  | 
|  | // Output directory is absolute location for generated source (no gen-*) | 
|  | bool out_path_is_absolute_; | 
|  |  | 
|  | // Namespace | 
|  | std::string namespace_; | 
|  |  | 
|  | // Included programs | 
|  | std::vector<t_program*> includes_; | 
|  |  | 
|  | // Include prefix for this program, if any | 
|  | std::string include_prefix_; | 
|  |  | 
|  | // Identifier lookup scope | 
|  | t_scope* scope_; | 
|  |  | 
|  | // Components to generate code for | 
|  | std::vector<t_typedef*> typedefs_; | 
|  | std::vector<t_enum*>    enums_; | 
|  | std::vector<t_const*>   consts_; | 
|  | std::vector<t_struct*>  objects_; | 
|  | std::vector<t_struct*>  structs_; | 
|  | std::vector<t_struct*>  xceptions_; | 
|  | std::vector<t_service*> services_; | 
|  |  | 
|  | // Dynamic namespaces | 
|  | std::map<std::string, std::string> namespaces_; | 
|  |  | 
|  | // C++ extra includes | 
|  | std::vector<std::string> cpp_includes_; | 
|  |  | 
|  | // C extra includes | 
|  | std::vector<std::string> c_includes_; | 
|  |  | 
|  | }; | 
|  |  | 
|  | #endif |