mirror of
https://github.com/pocoproject/poco.git
synced 2025-10-29 04:17:55 +01:00
added PageCompiler
This commit is contained in:
515
PageCompiler/src/PageCompiler.cpp
Normal file
515
PageCompiler/src/PageCompiler.cpp
Normal file
@@ -0,0 +1,515 @@
|
||||
//
|
||||
// PageCompiler.cpp
|
||||
//
|
||||
// $Id: //poco/Main/PageCompiler/src/PageCompiler.cpp#1 $
|
||||
//
|
||||
// A compiler that compiler HTML pages containing JSP directives into C++ classes.
|
||||
//
|
||||
// Copyright (c) 2007, Applied Informatics Software Engineering GmbH.
|
||||
// and Contributors.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person or organization
|
||||
// obtaining a copy of the software and accompanying documentation covered by
|
||||
// this license (the "Software") to use, reproduce, display, distribute,
|
||||
// execute, and transmit the Software, and to prepare derivative works of the
|
||||
// Software, and to permit third-parties to whom the Software is furnished to
|
||||
// do so, all subject to the following:
|
||||
//
|
||||
// The copyright notices in the Software and this entire statement, including
|
||||
// the above license grant, this restriction and the following disclaimer,
|
||||
// must be included in all copies of the Software, in whole or in part, and
|
||||
// all derivative works of the Software, unless such copies or derivative
|
||||
// works are solely in the form of machine-executable object code generated by
|
||||
// a source language processor.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
||||
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
||||
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
||||
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
//
|
||||
|
||||
|
||||
#include "Poco/Util/Application.h"
|
||||
#include "Poco/Util/Option.h"
|
||||
#include "Poco/Util/OptionSet.h"
|
||||
#include "Poco/Util/HelpFormatter.h"
|
||||
#include "Poco/Util/AbstractConfiguration.h"
|
||||
#include "Poco/AutoPtr.h"
|
||||
#include "Poco/FileStream.h"
|
||||
#include "Poco/Path.h"
|
||||
#include "Poco/DateTime.h"
|
||||
#include "Poco/DateTimeFormatter.h"
|
||||
#include "Poco/DateTimeFormat.h"
|
||||
#include "Poco/StringTokenizer.h"
|
||||
#include "Poco/LineEndingConverter.h"
|
||||
#include "Page.h"
|
||||
#include <cctype>
|
||||
#include <sstream>
|
||||
#include <iostream>
|
||||
|
||||
|
||||
using Poco::Util::Application;
|
||||
using Poco::Util::Option;
|
||||
using Poco::Util::OptionSet;
|
||||
using Poco::Util::HelpFormatter;
|
||||
using Poco::Util::AbstractConfiguration;
|
||||
using Poco::Util::OptionCallback;
|
||||
using Poco::AutoPtr;
|
||||
using Poco::FileInputStream;
|
||||
using Poco::FileOutputStream;
|
||||
using Poco::Path;
|
||||
using Poco::DateTime;
|
||||
using Poco::DateTimeFormatter;
|
||||
using Poco::DateTimeFormat;
|
||||
using Poco::StringTokenizer;
|
||||
using Poco::OutputLineEndingConverter;
|
||||
|
||||
|
||||
class CompilerApp: public Application
|
||||
{
|
||||
public:
|
||||
CompilerApp():
|
||||
_helpRequested(false),
|
||||
_generateOSPCode(false),
|
||||
_generateApacheCode(false)
|
||||
{
|
||||
}
|
||||
|
||||
protected:
|
||||
void initialize(Application& self)
|
||||
{
|
||||
loadConfiguration(); // load default configuration files, if present
|
||||
Application::initialize(self);
|
||||
}
|
||||
|
||||
void defineOptions(OptionSet& options)
|
||||
{
|
||||
Application::defineOptions(options);
|
||||
|
||||
options.addOption(
|
||||
Option("help", "h", "display help information on command line arguments")
|
||||
.required(false)
|
||||
.repeatable(false)
|
||||
.callback(OptionCallback<CompilerApp>(this, &CompilerApp::handleHelp)));
|
||||
|
||||
options.addOption(
|
||||
Option("define", "D", "define a property")
|
||||
.required(false)
|
||||
.repeatable(true)
|
||||
.argument("name=value")
|
||||
.callback(OptionCallback<CompilerApp>(this, &CompilerApp::handleDefine)));
|
||||
|
||||
options.addOption(
|
||||
Option("config-file", "f", "load configuration data from a file")
|
||||
.required(false)
|
||||
.repeatable(true)
|
||||
.argument("file")
|
||||
.callback(OptionCallback<CompilerApp>(this, &CompilerApp::handleConfig)));
|
||||
|
||||
options.addOption(
|
||||
Option("osp", "O", "add factory class definition/implementation for use with OSP")
|
||||
.required(false)
|
||||
.repeatable(false)
|
||||
.callback(OptionCallback<CompilerApp>(this, &CompilerApp::handleOSP)));
|
||||
|
||||
options.addOption(
|
||||
Option("apache", "A", "add factory class definition/implementation and shared library manifest for use with ApacheConnector")
|
||||
.required(false)
|
||||
.repeatable(false)
|
||||
.callback(OptionCallback<CompilerApp>(this, &CompilerApp::handleApache)));
|
||||
}
|
||||
|
||||
void handleHelp(const std::string& name, const std::string& value)
|
||||
{
|
||||
_helpRequested = true;
|
||||
displayHelp();
|
||||
stopOptionsProcessing();
|
||||
}
|
||||
|
||||
void handleDefine(const std::string& name, const std::string& value)
|
||||
{
|
||||
defineProperty(value);
|
||||
}
|
||||
|
||||
void handleConfig(const std::string& name, const std::string& value)
|
||||
{
|
||||
loadConfiguration(value);
|
||||
}
|
||||
|
||||
void handleOSP(const std::string& name, const std::string& value)
|
||||
{
|
||||
_generateOSPCode = true;
|
||||
}
|
||||
|
||||
void handleApache(const std::string& name, const std::string& value)
|
||||
{
|
||||
_generateApacheCode = true;
|
||||
}
|
||||
|
||||
void displayHelp()
|
||||
{
|
||||
HelpFormatter helpFormatter(options());
|
||||
helpFormatter.setCommand(commandName());
|
||||
helpFormatter.setUsage("OPTIONS PAGES");
|
||||
helpFormatter.setHeader("The POCO C++ Server Page Compiler.");
|
||||
helpFormatter.format(std::cout);
|
||||
}
|
||||
|
||||
void defineProperty(const std::string& def)
|
||||
{
|
||||
std::string name;
|
||||
std::string value;
|
||||
std::string::size_type pos = def.find('=');
|
||||
if (pos != std::string::npos)
|
||||
{
|
||||
name.assign(def, 0, pos);
|
||||
value.assign(def, pos + 1, def.length() - pos);
|
||||
}
|
||||
else name = def;
|
||||
config().setString(name, value);
|
||||
}
|
||||
|
||||
int main(const std::vector<std::string>& args)
|
||||
{
|
||||
if (_helpRequested)
|
||||
return Application::EXIT_OK;
|
||||
|
||||
for (std::vector<std::string>::const_iterator it = args.begin(); it != args.end(); ++it)
|
||||
{
|
||||
compile(*it);
|
||||
}
|
||||
|
||||
return Application::EXIT_OK;
|
||||
}
|
||||
|
||||
void compile(const std::string& path)
|
||||
{
|
||||
Page page(path);
|
||||
|
||||
FileInputStream srcStream(path);
|
||||
page.parse(srcStream);
|
||||
|
||||
Path p(path);
|
||||
config().setString("inputFileName", p.getFileName());
|
||||
config().setString("inputFilePath", p.toString());
|
||||
|
||||
DateTime now;
|
||||
config().setString("dateTime", DateTimeFormatter::format(now, DateTimeFormat::SORTABLE_FORMAT));
|
||||
|
||||
std::string clazz;
|
||||
if (page.has("page.class"))
|
||||
{
|
||||
clazz = page.get("page.class");
|
||||
p.setBaseName(clazz);
|
||||
}
|
||||
else
|
||||
{
|
||||
clazz = p.getBaseName() + "Handler";
|
||||
clazz[0] = std::toupper(clazz[0]);
|
||||
}
|
||||
|
||||
p.setExtension("h");
|
||||
std::string headerPath = p.toString();
|
||||
std::string headerFileName = p.getFileName();
|
||||
config().setString("outputFileName", p.getFileName());
|
||||
config().setString("outputFilePath", headerPath);
|
||||
FileOutputStream headerStream(headerPath);
|
||||
OutputLineEndingConverter headerLEC(headerStream);
|
||||
writeHeader(headerLEC, page, clazz, p.getFileName());
|
||||
|
||||
p.setExtension("cpp");
|
||||
std::string implPath = p.toString();
|
||||
config().setString("outputFileName", p.getFileName());
|
||||
config().setString("outputFilePath", implPath);
|
||||
FileOutputStream implStream(implPath);
|
||||
OutputLineEndingConverter implLEC(implStream);
|
||||
writeImpl(implLEC, page, clazz, headerFileName);
|
||||
}
|
||||
|
||||
void writeHeader(std::ostream& ostr, Page& page, const std::string& clazz, const std::string& headerFileName)
|
||||
{
|
||||
std::string fileHeader = config().getString("PageCompiler.fileHeader", "");
|
||||
if (!fileHeader.empty())
|
||||
{
|
||||
ostr << fileHeader << std::endl;
|
||||
ostr << "\n\n";
|
||||
}
|
||||
Path p(headerFileName);
|
||||
std::string guard(p.getBaseName());
|
||||
guard += "_INCLUDED";
|
||||
ostr << "#ifndef " << guard << "\n";
|
||||
ostr << "#define " << guard << "\n";
|
||||
ostr << "\n\n";
|
||||
ostr << "#include \"Poco/Net/HTTPRequestHandler.h\"\n";
|
||||
|
||||
if (_generateOSPCode)
|
||||
{
|
||||
ostr << "#include \"Poco/OSP/Web/WebRequestHandlerFactory.h\"\n";
|
||||
ostr << "#include \"Poco/OSP/BundleContext.h\"\n";
|
||||
}
|
||||
else if (_generateApacheCode)
|
||||
{
|
||||
ostr << "#include \"Poco/Net/HTTPRequestHandlerFactory.h\"\n";
|
||||
}
|
||||
|
||||
ostr << "\n\n";
|
||||
|
||||
const std::string& decls = page.headerDecls();
|
||||
if (!decls.empty())
|
||||
{
|
||||
ostr << decls;
|
||||
ostr << "\n\n";
|
||||
}
|
||||
|
||||
beginNamespace(ostr, page);
|
||||
|
||||
std::string base(page.get("page.baseClass", "Poco::Net::HTTPRequestHandler"));
|
||||
std::string ctorArg;
|
||||
if (_generateOSPCode)
|
||||
ctorArg = "Poco::OSP::BundleContext::Ptr";
|
||||
else
|
||||
ctorArg = page.get("page.ctorArg", "");
|
||||
|
||||
std::string exprt(page.get("page.export", ""));
|
||||
if (!exprt.empty()) exprt += ' ';
|
||||
|
||||
ostr << "class " << exprt << clazz << ": public " << base << "\n";
|
||||
ostr << "{\n";
|
||||
ostr << "public:\n";
|
||||
if (!ctorArg.empty())
|
||||
{
|
||||
ostr << "\t" << clazz << "(" << ctorArg << ");\n";
|
||||
ostr << "\n";
|
||||
}
|
||||
ostr << "\tvoid handleRequest(Poco::Net::HTTPServerRequest& request, Poco::Net::HTTPServerResponse& response);\n";
|
||||
|
||||
if (_generateOSPCode)
|
||||
writeOSPHeader(ostr);
|
||||
|
||||
ostr << "};\n";
|
||||
|
||||
if (_generateOSPCode)
|
||||
{
|
||||
ostr << "\n\n";
|
||||
writeFactoryHeader(ostr, page, clazz, "Poco::OSP::Web::WebRequestHandlerFactory");
|
||||
}
|
||||
else if (_generateApacheCode)
|
||||
{
|
||||
ostr << "\n\n";
|
||||
writeFactoryHeader(ostr, page, clazz, "Poco::Net::HTTPRequestHandlerFactory");
|
||||
}
|
||||
|
||||
endNamespace(ostr, page);
|
||||
|
||||
ostr << "\n\n";
|
||||
ostr << "#endif // " << guard << "\n";
|
||||
}
|
||||
|
||||
void writeImpl(std::ostream& ostr, Page& page, const std::string& clazz, const std::string& headerFileName)
|
||||
{
|
||||
std::string fileHeader = config().getString("PageCompiler.fileHeader", "");
|
||||
if (!fileHeader.empty())
|
||||
{
|
||||
ostr << fileHeader << std::endl;
|
||||
ostr << "\n\n";
|
||||
}
|
||||
ostr << "#include \"" << headerFileName << "\"\n";
|
||||
ostr << "#include \"Poco/Net/HTTPServerRequest.h\"\n";
|
||||
ostr << "#include \"Poco/Net/HTTPServerResponse.h\"\n";
|
||||
ostr << "#include \"Poco/Net/HTMLForm.h\"\n";
|
||||
|
||||
if (_generateOSPCode)
|
||||
{
|
||||
if (page.has("page.session"))
|
||||
{
|
||||
ostr << "#include \"Poco/OSP/Web/WebSession.h\"\n";
|
||||
ostr << "#include \"Poco/OSP/Web/WebSessionManager.h\"\n";
|
||||
ostr << "#include \"Poco/OSP/ServiceRegistry.h\"\n";
|
||||
}
|
||||
}
|
||||
else if (_generateApacheCode)
|
||||
{
|
||||
ostr << "#include \"Poco/ClassLibrary.h\"\n";
|
||||
}
|
||||
|
||||
ostr << "\n\n";
|
||||
|
||||
const std::string& decls = page.implDecls();
|
||||
if (!decls.empty())
|
||||
{
|
||||
ostr << decls;
|
||||
ostr << "\n\n";
|
||||
}
|
||||
|
||||
beginNamespace(ostr, page);
|
||||
|
||||
Path p(headerFileName);
|
||||
std::string base(page.get("page.baseClass", "Poco::Net::HTTPRequestHandler"));
|
||||
std::string ctorArg(page.get("page.ctorArg", ""));
|
||||
|
||||
if (_generateOSPCode)
|
||||
{
|
||||
ostr << clazz << "::" << clazz << "(Poco::OSP::BundleContext::Ptr pContext):\n";
|
||||
ostr << "\t_pContext(pContext)\n";
|
||||
ostr << "{\n}\n";
|
||||
ostr << "\n\n";
|
||||
}
|
||||
else if (!ctorArg.empty())
|
||||
{
|
||||
ostr << clazz << "::" << clazz << "(" << ctorArg << " arg):\n";
|
||||
ostr << "\t" << base << "(arg)\n";
|
||||
ostr << "{\n}\n";
|
||||
ostr << "\n\n";
|
||||
}
|
||||
|
||||
ostr << "void " << clazz << "::handleRequest(Poco::Net::HTTPServerRequest& request, Poco::Net::HTTPServerResponse& response)\n";
|
||||
ostr << "{\n";
|
||||
|
||||
if (_generateOSPCode && page.has("page.session"))
|
||||
{
|
||||
writeOSPSession(ostr, page);
|
||||
}
|
||||
|
||||
if (page.get("page.form", "true") != "false")
|
||||
{
|
||||
std::string partHandler(page.get("page.formPartHandler", ""));
|
||||
if (!partHandler.empty())
|
||||
{
|
||||
ostr << "\t" << partHandler << " cpspPartHandler(*this);\n";
|
||||
}
|
||||
ostr << "\tPoco::Net::HTMLForm form(request, request.stream()";
|
||||
if (!partHandler.empty())
|
||||
{
|
||||
ostr << ", cpspPartHandler";
|
||||
}
|
||||
ostr << ");\n";
|
||||
}
|
||||
|
||||
std::string contentType(page.get("page.contentType", "text/html"));
|
||||
std::string chunked(page.get("page.chunked", "true"));
|
||||
|
||||
if (chunked != "false")
|
||||
{
|
||||
ostr << "\tresponse.setChunkedTransferEncoding(true);\n";
|
||||
}
|
||||
|
||||
ostr << "\tresponse.setContentType(\"" << contentType << "\");\n";
|
||||
ostr << "\n";
|
||||
|
||||
ostr << "\tstd::ostream& ostr = response.send();\n";
|
||||
ostr << page.handler();
|
||||
|
||||
ostr << "}\n";
|
||||
|
||||
if (_generateOSPCode)
|
||||
{
|
||||
ostr << "\n\n";
|
||||
writeFactoryImpl(ostr, page, clazz, "context()");
|
||||
}
|
||||
else if (_generateApacheCode)
|
||||
{
|
||||
ostr << "\n\n";
|
||||
writeFactoryImpl(ostr, page, clazz, "");
|
||||
}
|
||||
|
||||
endNamespace(ostr, page);
|
||||
|
||||
if (_generateApacheCode)
|
||||
writeManifest(ostr, page, clazz);
|
||||
}
|
||||
|
||||
void beginNamespace(std::ostream& ostr, Page& page)
|
||||
{
|
||||
std::string ns = page.get("page.namespace", "");
|
||||
if (!ns.empty())
|
||||
{
|
||||
StringTokenizer tok(ns, ":", StringTokenizer::TOK_IGNORE_EMPTY | StringTokenizer::TOK_TRIM);
|
||||
for (StringTokenizer::Iterator it = tok.begin(); it != tok.end(); ++it)
|
||||
{
|
||||
ostr << "namespace " << *it << " {\n";
|
||||
}
|
||||
ostr << "\n\n";
|
||||
}
|
||||
}
|
||||
|
||||
void endNamespace(std::ostream& ostr, Page& page)
|
||||
{
|
||||
std::string ns = page.get("page.namespace", "");
|
||||
if (!ns.empty())
|
||||
{
|
||||
ostr << "\n\n";
|
||||
StringTokenizer tok(ns, ":", StringTokenizer::TOK_IGNORE_EMPTY | StringTokenizer::TOK_TRIM);
|
||||
for (StringTokenizer::Iterator it = tok.begin(); it != tok.end(); ++it)
|
||||
{
|
||||
ostr << "} ";
|
||||
}
|
||||
ostr << "// namespace " << ns << "\n";
|
||||
}
|
||||
}
|
||||
|
||||
void writeOSPSession(std::ostream& ostr, Page& page)
|
||||
{
|
||||
ostr << "\tPoco::OSP::Web::WebSession::Ptr session;\n";
|
||||
ostr << "\t{\n";
|
||||
ostr << "\t\tPoco::OSP::ServiceRef::Ptr pWebSessionManagerRef = context()->registry().findByName(Poco::OSP::Web::WebSessionManager::SERVICE_NAME);\n";
|
||||
ostr << "\t\tif (pWebSessionManagerRef)\n";
|
||||
ostr << "\t\t{\n";
|
||||
ostr << "\t\t\tPoco::OSP::Web::WebSessionManager::Ptr pWebSessionManager = pWebSessionManagerRef->castedInstance<Poco::OSP::Web::WebSessionManager>();\n";
|
||||
ostr << "\t\t\tsession = pWebSessionManager->find(\"" << page.get("page.session") << "\", request);\n";
|
||||
ostr << "\t\t\tif (session.isNull())\n";
|
||||
ostr << "\t\t\t\tsession = pWebSessionManager->get(\"" << page.get("page.session") << "\", request, " << page.get("page.sessionTimeout", "30") << ", context());\n";
|
||||
ostr << "\t\t}\n";
|
||||
ostr << "\t}\n";
|
||||
}
|
||||
|
||||
void writeOSPHeader(std::ostream& ostr)
|
||||
{
|
||||
ostr << "\n";
|
||||
ostr << "protected:\n";
|
||||
ostr << "\tPoco::OSP::BundleContext::Ptr context() const\n";
|
||||
ostr << "\t{\n";
|
||||
ostr << "\t\treturn _pContext;\n";
|
||||
ostr << "\t}\n";
|
||||
ostr << "\n";
|
||||
ostr << "private:\n";
|
||||
ostr << "\tPoco::OSP::BundleContext::Ptr _pContext;\n";
|
||||
}
|
||||
|
||||
void writeFactoryHeader(std::ostream& ostr, Page& page, const std::string& clazz, const std::string& base)
|
||||
{
|
||||
ostr << "class " << clazz << "Factory: public " << base << "\n";
|
||||
ostr << "{\n";
|
||||
ostr << "public:\n";
|
||||
ostr << "\tPoco::Net::HTTPRequestHandler* createRequestHandler(const Poco::Net::HTTPServerRequest& request);\n";
|
||||
ostr << "};\n";
|
||||
}
|
||||
|
||||
void writeFactoryImpl(std::ostream& ostr, Page& page, const std::string& clazz, const std::string& arg)
|
||||
{
|
||||
ostr << "Poco::Net::HTTPRequestHandler* " << clazz << "Factory::createRequestHandler(const Poco::Net::HTTPServerRequest& request)\n";
|
||||
ostr << "{\n";
|
||||
ostr << "\treturn new " << clazz << "(" << arg << ");\n";
|
||||
ostr << "}\n";
|
||||
}
|
||||
|
||||
void writeManifest(std::ostream& ostr, Page& page, const std::string& clazz)
|
||||
{
|
||||
std::string ns = page.get("page.namespace", "");
|
||||
if (!ns.empty()) ns += "::";
|
||||
ostr << "\n\n";
|
||||
ostr << "POCO_BEGIN_MANIFEST(Poco::Net::HTTPRequestHandlerFactory)\n";
|
||||
ostr << "\tPOCO_EXPORT_CLASS(" << ns << clazz << "Factory)\n";
|
||||
ostr << "POCO_END_MANIFEST\n";
|
||||
}
|
||||
|
||||
private:
|
||||
bool _helpRequested;
|
||||
bool _generateOSPCode;
|
||||
bool _generateApacheCode;
|
||||
};
|
||||
|
||||
|
||||
POCO_APP_MAIN(CompilerApp)
|
||||
Reference in New Issue
Block a user