1
0
Fork 0
mirror of https://github.com/KingDuckZ/kamokan.git synced 2025-01-13 19:56:40 +00:00
kamokan/docs/source-highlight.cc
2017-05-10 20:49:40 +01:00

594 lines
20 KiB
C++

/*
* Copyright (C) 1999-2009 Lorenzo Bettini <http://www.lorenzobettini.it>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 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 General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
// this is glib related so don't use it when compiling with qmake
#include "progname.h"
#else
#define set_program_name(x) ;
#endif
#include <cstdlib>
#include <iostream>
#include <exception>
#include <boost/shared_ptr.hpp>
#include "srchilite/sourcehighlight.h"
#include "srchilite/fileutil.h"
#include "srchilite/verbosity.h"
#include "srchilite/langmap.h"
#include "srchilite/languageinfer.h"
#include "srchilite/utils.h"
#include "srchilite/highlightbuilderexception.h"
#include "srchilite/parserexception.h"
#include "srchilite/debuglistener.h"
#include "srchilite/ctagsmanager.h"
#include "srchilite/stopwatch.h"
#include "srchilite/lineranges.h"
#include "srchilite/regexranges.h"
#include "srchilite/versions.h"
#include "srchilite/settings.h"
#include "cmdline.h"
using namespace std;
using namespace srchilite;
/**
* how language inference should be made: NOINFERENCE = no inference at all,
* INFERFIRST = inference has priority (e.g., if --infer-lang is specified),
* INFERATLAST = try inference as the last chance
*/
enum InferPolicy {
NOINFERENCE = 0, INFERFIRST, INFERATLAST
};
static void print_copyright();
static void print_reportbugs();
static void print_version();
static string inferLang(const string &inputFileName);
static string getMappedLang(const string &s, LangMap &langmap);
static void printError(const string &s);
static void print_cgi_header();
/**
* Exits for a generic error (i.e., file not found)
*/
static void exitError(const string &s);
static string getLangFileName(InferPolicy infer, const string &inputFileName,
const string &langFileName, const string &langName, LangMap &langMap);
static bool verbose = false;
static bool failsafe = false;
#ifdef BUILD_AS_CGI
#include "envmapper.h"
#endif // BUILD_AS_CGI
/**
* Print progress status information (provided --quiet is not specified)
* @param message
*/
#define PROGRESSINFO(message) if (!args_info.quiet_given) cerr << message;
int main(int argc, char * argv[]) {
set_program_name(argv[0]);
gengetopt_args_info args_info; // command line structure
bool is_cgi = false;
#ifdef BUILD_AS_CGI
// map environment to parameters if used as CGI
char **temp_argv;
temp_argv = map_environment(&argc, argv);
is_cgi = temp_argv != argv;
argv = temp_argv;
#endif // BUILD_AS_CGI
if (cmdline_parser(argc, argv, &args_info) != 0) {
// calls cmdline parser. The user gived bag args if it doesn't return -1
return EXIT_FAILURE;
}
if (args_info.version_given) {
print_version();
print_copyright();
cmdline_parser_free(&args_info);
return EXIT_SUCCESS;
}
if (args_info.help_given || args_info.detailed_help_given) {
cout << "GNU ";
if (args_info.detailed_help_given) {
cmdline_parser_print_detailed_help();
} else {
cmdline_parser_print_help();
}
print_reportbugs();
cmdline_parser_free(&args_info);
return EXIT_SUCCESS;
}
// set possible line ranges
boost::shared_ptr<LineRanges> lineRanges;
if (args_info.line_range_given) {
lineRanges = boost::shared_ptr<LineRanges>(new LineRanges);
for (unsigned int i = 0; i < args_info.line_range_given; ++i) {
if (lineRanges->addRange(args_info.line_range_arg[i]) != NO_ERROR) {
string invalid_range = args_info.line_range_arg[i];
cmdline_parser_free(&args_info);
exitError("invalid line range: " + invalid_range);
}
}
}
// set possible regex ranges
boost::shared_ptr<RegexRanges> regexRanges;
if (args_info.regex_range_given) {
regexRanges = boost::shared_ptr<RegexRanges>(new RegexRanges);
for (unsigned int i = 0; i < args_info.regex_range_given; ++i) {
if (!regexRanges->addRegexRange(args_info.regex_range_arg[i])) {
string invalid_range = args_info.regex_range_arg[i];
cmdline_parser_free(&args_info);
exitError("invalid regex range: " + invalid_range);
}
}
}
if (args_info.range_context_given) {
lineRanges->setContextLines(args_info.range_context_arg);
}
// if a stopwatch is created, when it is deleted (automatically
// since we're using a shared pointer, it will print the
// elapsed seconds.
boost::shared_ptr<StopWatch> stopwatch;
if (args_info.statistics_given) {
stopwatch = boost::shared_ptr<StopWatch>(new StopWatch);
}
verbose = args_info.verbose_given;
failsafe = args_info.failsafe_given;
string inputFile;
string outputFile;
string srcLang;
string outputFormat;
string langFile;
string outLangFile;
string dataDir;
string outputDir;
Verbosity::setVerbosity(verbose);
if (args_info.input_given)
inputFile = args_info.input_arg;
if (args_info.output_given)
outputFile = args_info.output_arg;
// for cgi force output to standard output
if (is_cgi)
outputFile = "STDOUT";
if (args_info.src_lang_given)
srcLang = args_info.src_lang_arg;
if (args_info.lang_def_given)
langFile = args_info.lang_def_arg;
if (args_info.outlang_def_given)
outLangFile = args_info.outlang_def_arg;
// the default for output format is html
outputFormat = args_info.out_format_arg;
if (args_info.data_dir_given) {
dataDir = args_info.data_dir_arg;
}
if (args_info.output_dir_given) {
outputDir = args_info.output_dir_arg;
}
/*
the starting default path to search for files is computed at
run-time: it is
the path of the binary + ".." + RELATIVEDATADIR
this should make the package relocable (i.e., not stuck
with a fixed installation directory).
Of course, the GNU standards for installation directories
should be followed, but this is not a problem if you use
configure and make install features.
If no path is specified in the running program we go back to
the absolute datadir.
*/
// this is defined in fileutil.cc
string prefix_dir = get_file_path(argv[0]);
if (prefix_dir.size())
start_path = get_file_path(argv[0]) + RELATIVEDATADIR;
else
start_path = Settings::retrieveDataDir();
// if datadir is not specified, we rely on start_path?
try {
// initialize map files
LangMap langmap(dataDir, args_info.lang_map_arg);
// just print the input language list and associations
if (args_info.lang_list_given) {
langmap.open();
langmap.print();
return (EXIT_SUCCESS);
}
LangMap outlangmap(dataDir, args_info.outlang_map_arg);
// just print the input language list and associations
if (args_info.outlang_list_given) {
outlangmap.open();
outlangmap.print();
return (EXIT_SUCCESS);
}
string cssUrl;
if (args_info.css_given)
cssUrl = args_info.css_arg;
string docTitle;
if (args_info.title_given)
docTitle = args_info.title_arg;
if (args_info.check_lang_given || args_info.check_outlang_given
|| args_info.show_regex_given
|| args_info.show_lang_elements_given
|| args_info.lang_list_given || args_info.outlang_list_given) {
// FIXME just a bogus outlang file specification not to search in outlang.map
outLangFile = "html.outlang";
}
// initialize the main source highlight object
SourceHighlight sourcehighlight(getLangFileName(NOINFERENCE,
outputFile, outLangFile, outputFormat
+ (cssUrl.size() ? +"-css" : ""), outlangmap));
// and sets all its properties according to the command line args
sourcehighlight.setDataDir(dataDir);
boost::shared_ptr<DebugListener> debugListener;
// if a simple check is required...
if (args_info.check_lang_given) {
cout << "checking " << args_info.check_lang_arg << "... ";
sourcehighlight.checkLangDef(args_info.check_lang_arg);
// if we're here, everything went fine!
cout << "OK" << endl;
// otherwise an exception has already been thrown
return (EXIT_SUCCESS);
}
if (args_info.check_outlang_given) {
cout << "checking " << args_info.check_outlang_arg << "... ";
sourcehighlight.checkOutLangDef(args_info.check_outlang_arg);
// if we're here, everything went fine!
cout << "OK" << endl;
// otherwise an exception has already been thrown
return (EXIT_SUCCESS);
}
// just print the highlightstate if show-regex is specified
if (args_info.show_regex_given) {
sourcehighlight.printHighlightState(args_info.show_regex_arg, cout);
return (EXIT_SUCCESS);
}
// just print the lang elems if show-lang-elements is specified
if (args_info.show_lang_elements_given) {
sourcehighlight.printLangElems(args_info.show_lang_elements_arg,
cout);
return (EXIT_SUCCESS);
}
sourcehighlight.setGenerateEntireDoc((!args_info.no_doc_given)
&& (args_info.doc_given || (docTitle.size()) || cssUrl.size()));
sourcehighlight.setStyleDefaultFile(args_info.style_defaults_arg);
sourcehighlight.setStyleFile(args_info.style_file_arg);
if (args_info.style_css_file_given)
sourcehighlight.setStyleCssFile(args_info.style_css_file_arg);
sourcehighlight.setGenerateVersion(args_info.gen_version_flag != 0);
sourcehighlight.setCss(cssUrl);
sourcehighlight.setTitle(docTitle);
sourcehighlight.setOutputDir(outputDir);
sourcehighlight.setLineRanges(lineRanges.get());
sourcehighlight.setRegexRanges(regexRanges.get());
sourcehighlight.setBinaryOutput(args_info.binary_output_given);
if (args_info.debug_langdef_given) {
debugListener = boost::shared_ptr<DebugListener>(new DebugListener);
const string debugType = args_info.debug_langdef_arg;
if (debugType == "interactive")
debugListener->setInteractive(true);
sourcehighlight.setHighlightEventListener(debugListener.get());
// during debugging disable optimizations
sourcehighlight.setOptimize(false);
}
int tabs = 0;
if (args_info.tab_given) {
tabs = args_info.tab_arg;
}
if (args_info.header_given)
sourcehighlight.setHeaderFileName(args_info.header_arg);
if (args_info.footer_given)
sourcehighlight.setFooterFileName(args_info.footer_arg);
bool generateLineNumbers = args_info.line_number_given
|| args_info.line_number_ref_given;
sourcehighlight.setGenerateLineNumbers(generateLineNumbers);
if (args_info.line_number_given) {
// set the line number padding char (default is always '0')
sourcehighlight.setLineNumberPad(args_info.line_number_arg[0]);
}
if (args_info.line_number_ref_given) {
sourcehighlight.setGenerateLineNumberRefs(true);
sourcehighlight.setLineNumberAnchorPrefix(
args_info.line_number_ref_arg);
}
if (generateLineNumbers) {
// when generating line numbers tabs must be translated
// otherwise the output line will not be aligned
// due to the presence of line numbers
sourcehighlight.setTabSpaces(tabs ? tabs : 8);
}
if (tabs) {
sourcehighlight.setTabSpaces(tabs);
}
// whether to give precedence to language inference
InferPolicy inferPolicy = (args_info.infer_lang_given ? INFERFIRST
: INFERATLAST);
RefPosition refposition = INLINE;
string gen_references_arg = args_info.gen_references_arg;
if (gen_references_arg == "inline")
refposition = INLINE;
else if (gen_references_arg == "postline")
refposition = POSTLINE;
else if (gen_references_arg == "postdoc")
refposition = POSTDOC;
boost::shared_ptr<CTagsManager> ctagsManager;
if (args_info.gen_references_given) {
string ctags = args_info.ctags_arg;
string ctags_file = args_info.ctags_file_arg;
// build the additional arguments for the ctags command
if (args_info.gen_references_given && ctags != "") {
if (inputFile.size()) {
ctags += " ";
ctags += inputFile;
} else if (args_info.inputs_num) {
for (unsigned int i = 0; i < (args_info.inputs_num); ++i) {
ctags += " ";
ctags += args_info.inputs[i];
}
}
}
// the ctags command must be executed if --ctags is specified with an empty string
ctagsManager = boost::shared_ptr<CTagsManager>(new CTagsManager(
ctags_file, ctags, ctags != "", refposition));
sourcehighlight.setCTagsManager(ctagsManager.get());
}
if (args_info.range_separator_given) {
sourcehighlight.setRangeSeparator(args_info.range_separator_arg);
}
// OK, let's highlight!
if (args_info.inputs_num && (args_info.input_given
|| args_info.output_given)) {
// do not mix command line invocation modes
cerr << "Please, use one of the two syntaxes for invocation: "
<< endl;
cerr
<< "source-highlight [OPTIONS]... -i input_file -o output_file"
<< endl;
cerr << "source-highlight [OPTIONS]... [FILES]..." << endl;
exit(EXIT_FAILURE);
}
// for cgi we can only process one file specified with --input
if (args_info.inputs_num && !is_cgi) {
// the output file name is empty, but we don't want to generate to stdout
sourcehighlight.setCanUseStdOut(false);
// in case multiple input files were specified (without --input)
for (unsigned int i = 0; i < (args_info.inputs_num); ++i) {
PROGRESSINFO(string("Processing ")+ args_info.inputs[i] + " ... ");
sourcehighlight.highlight(args_info.inputs[i], "",
getLangFileName(inferPolicy, args_info.inputs[i],
langFile, srcLang, langmap));
PROGRESSINFO("created " + sourcehighlight.createOutputFileName(args_info.inputs[i]) + "\n");
}
} else {
if (is_cgi)
print_cgi_header();
// in case only one input file was specified with --input
sourcehighlight.highlight(inputFile, outputFile, getLangFileName(
inferPolicy, inputFile, langFile, srcLang, langmap));
}
} catch (const HighlightBuilderException &e) {
cerr << e << endl;
exit(EXIT_FAILURE);
} catch (const ParserException &e) {
cerr << e << endl;
exit(EXIT_FAILURE);
} catch (const exception &e) {
exitError(e.what());
}
cmdline_parser_free(&args_info);
return 0;
}
void print_copyright() {
int i;
int copyright_text_length = 5;
const char *copyright_text[] = {
"copyright.text",
"Copyright (C) 1999-2008 Lorenzo Bettini <http://www.lorenzobettini.it>",
"This program comes with ABSOLUTELY NO WARRANTY.",
"This is free software; you may redistribute copies of the program",
"under the terms of the GNU General Public License.",
"For more information about these matters, see the file named COPYING.",
0 };
for (i = 1; i <= copyright_text_length; ++i)
cout << copyright_text[i] << endl;;
}
void print_reportbugs() {
int i;
int reportbugs_text_length = 3;
const char *reportbugs_text[] = {
"reportbugs.text",
"",
"Maintained by Lorenzo Bettini <http://www.lorenzobettini.it>",
"Report bugs to <bug-source-highlight at gnu.org>",
0 };
for (i = 1; i <= reportbugs_text_length; ++i)
cout << reportbugs_text[i] << endl;
}
void print_version() {
cout << Versions::getCompleteVersion() << endl;
}
string inferLang(const string &inputFileName) {
VERBOSELN("inferring input language...");
if (!inputFileName.size()) {
printError("missing feature: language inference requires input file");
return "";
}
LanguageInfer languageInfer;
const string &result = languageInfer.infer(inputFileName);
if (result.size()) {
VERBOSELN("inferred input language: " + result);
} else {
VERBOSELN("couldn't infer input language");
}
return result;
}
string getMappedLang(const string &s, LangMap &langmap) {
// OK now map it into a .lang file
string mapped_lang = langmap.getMappedFileName(s);
if (!mapped_lang.size()) {
// try the lower version
mapped_lang = langmap.getFileName(Utils::tolower(s));
}
return mapped_lang;
}
string getLangFileName(InferPolicy infer, const string &inputFileName,
const string &langFileName, const string &langName, LangMap &langMap) {
string langFile;
// language inference has the precedence
if (infer == INFERFIRST) {
langFile = inferLang(inputFileName);
langFile = getMappedLang(langFile, langMap);
if (langFile.size())
return langFile;
}
// then if a lang file name was specified we're done
if (langFileName.size())
return langFileName;
// now try with the langName
langFile = getMappedLang(langName, langMap);
if (langFile.size())
return langFile;
// otherwise try with the inputFileName (its file extension
// and its name)
langFile = langMap.getMappedFileNameFromFileName(inputFileName);
if (langFile.size())
return langFile;
// OK, as a last chance let's try with language infer
if (infer == INFERATLAST) {
langFile = getMappedLang(inferLang(inputFileName), langMap);
if (langFile.size())
return langFile;
}
// if we're here we failed all checks
// if failsafe is specified we default to default.lang
if (failsafe)
langFile = "default.lang";
if (!langFile.size()) {
// if we're here we must exit with failure
if (langName.size())
exitError("could not find a language definition for " + langName);
else
exitError("could not find a language definition for input file "
+ inputFileName);
}
return langFile;
}
void printError(const string &s) {
cerr << PACKAGE << ": " << s << endl;
}
void exitError(const string &s) {
printError(s);
exit(EXIT_FAILURE);
}
void print_cgi_header() {
cout << "Content-type: text/html\n";
cout << "\n";
}