/*************************************************************************** GenClientMethod.cpp - generate class with client methods ------------------- begin : Thu Juk 19 2007 copyright : (C) 2002-2007 by Ewald Arnold email : ulxmlrpcpp@ewald-arnold.de $Id: GenClientMethod.cpp 1005 2007-07-19 07:33:46Z ewald-arnold $ ***************************************************************************/ /************************************************************************** * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2 of the License, * or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * ***************************************************************************/ #include "GenClientFunction.h" #include #include #include GenerateClientFunctions::GenerateClientFunctions(const std::vector &methods) : theMethods(methods) { Method::resolveOverloaded(theMethods); } void GenerateClientFunctions::generate_H(const std::string &destdir, const std::string &name) { const std::string h_name = destdir + name + "_ulxr_client.h"; std::ofstream h_file(h_name.c_str()); std::cout << "Header file will be created: " << h_name << std::endl; generateHeaderHead(h_file, name + "UlxrClient"); generateHeaderMethods(h_file, name); h_file << "#endif // " << name + "UlxrClient" + "_H\n\n"; } void GenerateClientFunctions::generate_CPP(const std::string &destdir, const std::string &name) { const std::string h_name = name + "_ulxr_client.h"; const std::string cpp_name = destdir + name + "_ulxr_client.cpp"; std::ofstream cpp_file(cpp_name.c_str()); std::cout << "Source file will be created: " << cpp_name << std::endl; generateSourceHead(cpp_file, h_name); generateSourceMethods(destdir, cpp_file, name); } void GenerateClientFunctions::generate_CPP_USER(const std::string &destdir, const std::string &name) { const std::string h_name = name + "_ulxr_client.h"; std::string cpp_name = destdir + name + "_ulxr_client_user.cpp"; struct stat statbuf; if (stat(cpp_name.c_str(), &statbuf) >= 0) { std::cout << "User file already exists: " << cpp_name << std::endl; cpp_name += ".new"; std::cout << "New template will be created: " << cpp_name << std::endl; } std::ofstream cpp_file(cpp_name.c_str()); std::cout << "User file will be created: " << cpp_name << std::endl; generateUserSourceHead(cpp_file, h_name); cpp_file << "// #include \n"; cpp_file << "// #include \n"; cpp_file << "// #include \n\n"; cpp_file << "// #include \"" << name + "_ulxr_names.h" << "\"\n\n"; } void GenerateClientFunctions::generate(const std::string &destdir, const std::string &name) { generate_NameDefines(destdir, name, theMethods, name + "_"); generate_H(destdir, name); generate_CPP(destdir, name); generate_CPP_USER(destdir, name); } void GenerateClientFunctions::generateHeaderMethods(std::ostream & h_file, const std::string &name) { h_file << "namespace ulxr\n" "{\n" " class Requester;\n" "}\n\n"; for (unsigned i = 0; i < theMethods.size(); ++i) { Method method = theMethods[i]; std::string nspart = method.extractNamespace(); if (nspart.length() != 0) h_file << "namespace " << nspart << " {\n"; h_file << "// mapped to " << method.getOverloadName(); if (method.getName() != method.getOverloadName()) h_file << " (there are overloaded methods)"; h_file << "\n" << method.getCppString(0, false, "") << "\n"; if (nspart.length() != 0) h_file << "}\n"; h_file << "\n"; } h_file << "void " << name << "_setClientCredentials\n" << " (ulxr::Requester &in_requester, const ulxr::CppString &realm,\n" " const ulxr::CppString &user, const ulxr::CppString &pass);\n\n"; } void GenerateClientFunctions::generateSourceMethods(const std::string &destdir, std::ostream & cpp_file, const std::string &name) { cpp_file << "#include \n"; cpp_file << "#include \n"; cpp_file << "#include \n\n"; cpp_file << "#include \n"; cpp_file << "#include \n\n"; cpp_file << "#include \"" << name + "_ulxr_names.h" << "\"\n\n"; cpp_file << "namespace {\n\n" << "ulxr::Requester *requester = 0;\n" "ulxr::CppString rpc_realm;\n" "ulxr::CppString rpc_user;\n" "ulxr::CppString rpc_pass;\n\n" << "ulxr::MethodResponse\n " << "sendCall(const ulxr::MethodCall &calldata)\n" << "{\n" << " assert(requester != 0);\n" << " ulxr::MethodResponse resp = requester->call(calldata, rpc_realm, rpc_user, rpc_pass);\n" << "\n" << " if (!resp.isOK())\n" << " {\n" << " ulxr::Struct st = resp.getResult();\n" << " int ec = ulxr::Integer(st.getMember(ULXR_PCHAR(\"faultCode\"))).getInteger();\n" << " ulxr::CppString es = ulxr::RpcString(st.getMember(ULXR_PCHAR(\"faultString\"))).getString();\n" << " throw ulxr::RuntimeException(ec, es);\n" << " }\n" << "\n" << " return resp;\n" << "}\n\n"; for (unsigned i = 0; i < theMethods.size(); ++i) { const Method &method = theMethods[i]; const bool reftype = method.getType().isReference(); const bool ptrtype = method.getType().isPointer(); if (reftype || ptrtype) cpp_file << method.getType().getProxyType() << " ulxr_refFor_" << name << method.getOverloadName() << ";\n"; } for (unsigned i = 0; i < theMethods.size(); ++i) { const Method &method = theMethods[i]; cpp_file << "ulxr::MethodCall callTo_" << method.getOverloadName() << "(ULXR_CALLTO_" << name << "_" << method.getOverloadName(false, "", "_") << "); // mapped to: " << method.getCppString(0, true, ""); if (method.getName() != method.getOverloadName()) cpp_file << " (there are overloaded methods)"; cpp_file << "\n"; } cpp_file << "\n} // namespace\n\n\n" << "void " << name << "_setClientCredentials\n" << " (ulxr::Requester &in_requester, const ulxr::CppString &realm,\n" " const ulxr::CppString &user, const ulxr::CppString &pass)\n" "{\n" " requester = &in_requester;\n" " rpc_realm = realm;\n" " rpc_user = user;\n" " rpc_pass = pass;\n" "}\n\n"; for (unsigned i = 0; i < theMethods.size(); ++i) { Method method = theMethods[i]; std::string nspart = method.extractNamespace(); if (nspart.length() != 0) cpp_file << "namespace " << nspart << " {\n\n"; cpp_file << "// mapped to " << method.getOverloadName(false); if (method.getName() != method.getOverloadName()) cpp_file << " (there are overloaded methods)"; cpp_file << "\n" << method.getCppString(0, true, "") << "\n" << "{\n"; cpp_file << " callTo_" << method.getOverloadName() << ".clear();\n"; for (unsigned iarg = 0; iarg < method.numArgs(); ++iarg) { std::string adap = method.getArg(iarg).getType().getInversTypeAdapter(); std::string adap2; if (adap.length() != 0) { adap += "("; adap2 = ")"; } cpp_file << " callTo_" << method.getOverloadName() << ".addParam(" << method.getArg(iarg).getType().getRpcName() << "(" << adap << method.getArg(iarg).getType().getTypeDereference() << method.getArg(iarg).getName() << adap2 << "));\n"; } bool have_retval = false; if (method.getType().getName() != "void" || method.getType().getLeft() != "" || method.getType().getRight() != "") have_retval = true; if (have_retval) cpp_file << " ulxr::MethodResponse resp = sendCall(callTo_" << method.getOverloadName() << ");\n"; else cpp_file << " sendCall(callTo_" << method.getOverloadName() << ");\n"; if (have_retval) { std::string adap = method.getType().getTypeAdapter(); std::string adap2; if (adap.length() != 0) { adap += "("; adap2 = ")"; } const bool reftype = method.getType().isReference(); const bool ptrtype = method.getType().isPointer(); if (reftype || ptrtype) cpp_file << " ulxr_refFor_" << name << method.getOverloadName() << " = "; else cpp_file << " return "; cpp_file << "(" << method.getType().getProxyType() << ") " << adap + method.getType().getRpcName() << "(resp.getResult())." << method.getType().getRpcAccessor() << "()" << adap2 << ";\n"; if (reftype) cpp_file << " return ulxr_refFor_" << name << method.getOverloadName() << ";\n"; else if (ptrtype) { std::string acc = "&"; std::string acc2 = method.getType().getTypeAccessor(); if (acc2.length() != 0) acc = ""; cpp_file << " return " << "(" << method.getType().getCppString() << ") " << acc << "ulxr_refFor_" << name << method.getOverloadName() << acc2 << ";\n"; } } cpp_file << "}\n\n"; if (nspart.length() != 0) cpp_file << "}\n"; cpp_file << "\n"; } }