/ .. / / -> download
#include <stdio.h>
#include <unistd.h>
#include <getopt.h>
#include <stdbool.h>
#include <string.h>
#include <libxml/tree.h>
#include <libxml/xpath.h>
#include <libxslt/transform.h>
#include <libexslt/exslt.h>
#include "s1kd_tools.h"
#include "resources.h"

#define PROG_NAME "s1kd-aspp"
#define VERSION "5.0.1"

#define ERR_PREFIX PROG_NAME ": ERROR: "
#define WRN_PREFIX PROG_NAME ": WARNING: "
#define INF_PREFIX PROG_NAME ": INFO: "

#define E_BAD_LIST ERR_PREFIX "Could not read list: %s\n"

#define W_MISSING_REF WRN_PREFIX "Could not read referenced object: %s\n"

#define I_PROCESS INF_PREFIX "Processing %s...\n"

/* ID for the inline <applic> element representing the whole data module's
 * applicability. */
#define DEFAULT_DM_APPLIC_ID BAD_CAST "app-0000"
static xmlChar *dmApplicId;

/* XPath to select all elements which may have applicability annotations.
 *
 * Read from elements_list.h*/
static xmlChar *applicElemsXPath;

/* Search for ACTs/CCTs recursively. */
static bool recursive_search = false;

/* Directory to start search for ACTs/CCTs in. */
static char *search_dir;

/* Overwrite existing display text in annotations. */
static bool overwriteDispText = true;

/* Verbose output. */
static enum verbosity { QUIET, NORMAL, VERBOSE } verbosity = NORMAL;

/* Assume objects were created with -N. */
static bool no_issue = false;

/* Delimiter for format strings. */
#define FMTSTR_DELIM '%'

/* Return the first node matching an XPath expression. */
static xmlNodePtr firstXPathNode(xmlDocPtr doc, xmlNodePtr node, const char *xpath)
{
	xmlXPathContextPtr ctx;
	xmlXPathObjectPtr obj;
	xmlNodePtr first;

	if (doc) {
		ctx = xmlXPathNewContext(doc);
	} else if (node) {
		ctx = xmlXPathNewContext(node->doc);
	} else {
		return NULL;
	}

	ctx->node = node;

	obj = xmlXPathEvalExpression(BAD_CAST xpath, ctx);

	if (xmlXPathNodeSetIsEmpty(obj->nodesetval)) {
		first = NULL;
	} else {
		first = obj->nodesetval->nodeTab[0];
	}

	xmlXPathFreeObject(obj);
	xmlXPathFreeContext(ctx);

	return first;
}

/* Return the last node matching an XPath expression. */
static xmlNodePtr lastXPathNode(xmlDocPtr doc, xmlNodePtr node, const char *xpath)
{
	xmlXPathContextPtr ctx;
	xmlXPathObjectPtr obj;
	xmlNodePtr last;

	if (doc) {
		ctx = xmlXPathNewContext(doc);
	} else if (node) {
		ctx = xmlXPathNewContext(node->doc);
	} else {
		return NULL;
	}

	ctx->node = node;

	obj = xmlXPathEvalExpression(BAD_CAST xpath, ctx);

	if (xmlXPathNodeSetIsEmpty(obj->nodesetval)) {
		last = NULL;
	} else {
		last = obj->nodesetval->nodeTab[obj->nodesetval->nodeNr - 1];
	}

	xmlXPathFreeObject(obj);
	xmlXPathFreeContext(ctx);

	return last;
}

/* Return the value of the first node matching an XPath expression. */
static xmlChar *first_xpath_value(xmlDocPtr doc, xmlNodePtr node, const char *xpath)
{
	return xmlNodeGetContent(firstXPathNode(doc, node, xpath));
}

/* Set an explicit applicability annotation on a node.
 *
 * Nodes with the attribute @applicRefId already have an explicit annotation.
 *
 * Nodes without @applicRefId have an implicit annotation inherited from their
 * parent/ancestor which does have an explicit annotation, or ultimately from
 * the applicability of the whole data module. */
static void processNode(xmlNodePtr node)
{
	xmlNodePtr attr;

	attr = firstXPathNode(NULL, node, "@applicRefId|@refapplic");

	if (!attr) {
		/* Inherit applicability from
		 * - an ancestor, or
		 * - the whole data module level */
		xmlNodePtr ancestor;
		xmlChar *name;

		ancestor = lastXPathNode(NULL, node, "ancestor::*[@applicRefId]|ancestor::*[@refapplic]");

		name = BAD_CAST (firstXPathNode(NULL, node, "//idstatus") ? "refapplic" : "applicRefId");

		if (!ancestor) {
			xmlSetProp(node, name, dmApplicId);
		} else {
			xmlChar *ancestorApplic = xmlGetProp(ancestor, name);
			xmlSetProp(node, name, ancestorApplic);
			xmlFree(ancestorApplic);
		}
	}
}

/* Set explicit applicability on all nodes in a nodeset. */
static void processNodeSet(xmlNodeSetPtr nodes)
{
	int i;

	for (i = 0; i < nodes->nodeNr; ++i) {
		processNode(nodes->nodeTab[i]);
	}
}

/* Remove duplicate applicability annotations in document-order so that an
 * annotation is only shown when applicability changes. */
static void removeDuplicates(xmlNodeSetPtr nodes)
{
	int i;

	xmlChar *applic = xmlStrdup(dmApplicId);

	for (i = 0; i < nodes->nodeNr; ++i) {
		xmlNodePtr attr;
		xmlChar *applicRefId;

		attr = firstXPathNode(NULL, nodes->nodeTab[i], "@applicRefId|@refapplic");
		applicRefId = xmlNodeGetContent(attr);

		if (xmlStrcmp(applicRefId, applic) == 0) {
			xmlUnsetProp(nodes->nodeTab[i], attr->name);
		} else {
			xmlFree(applic);
			applic = xmlStrdup(applicRefId);
		}

		xmlFree(applicRefId);
	}

	xmlFree(applic);
}

/* Insert a new inline <applic> element representing the applicability of the
 * whole data module, based on the dmStatus/applic element. */
static void addDmApplic(xmlNodePtr dmodule)
{
	xmlNodePtr referencedApplicGroup;

	if ((referencedApplicGroup = firstXPathNode(NULL, dmodule, ".//referencedApplicGroup|.//inlineapplics"))) {
		xmlNodePtr applic;
		xmlNodePtr wholeDmApplic;

		wholeDmApplic = firstXPathNode(NULL, dmodule, ".//dmStatus/applic|.//status/applic");
		applic = xmlAddChild(referencedApplicGroup, xmlCopyNode(wholeDmApplic, 1));

		xmlSetProp(applic, BAD_CAST "id", dmApplicId);
	}
}

static xmlDocPtr parse_disptext(xmlDocPtr config)
{
	xmlDocPtr doc1, doc2, res;
	xsltStylesheetPtr style;

	doc1 = read_xml_mem((const char *) disptext_xsl, disptext_xsl_len);
	style = xsltParseStylesheetDoc(doc1);

	doc2 = xsltApplyStylesheet(style, config, NULL);
	xsltFreeStylesheet(style);

	res = xmlCopyDoc(doc2, 1);
	xmlFreeDoc(doc2);

	return res;
}

static void dumpGenDispTextXsl(void)
{
	xmlDocPtr doc1, doc2;
	doc1 = read_xml_mem((const char *) disptext_xml, disptext_xml_len);
	doc2 = parse_disptext(doc1);
	xmlFreeDoc(doc1);
	save_xml_doc(doc2, "-");
	xmlFreeDoc(doc2);
}

static void dumpDispText(void)
{
	printf("%.*s", disptext_xml_len, disptext_xml);
}

/* Customize the display text based on a format string. */
static void apply_format_str(xmlDocPtr doc, const char *fmt)
{
	xmlNodePtr t, c;
	int i;

	/* Get the assert text template. */
	t = firstXPathNode(doc, NULL, "//*[@match='assert' and @mode='text']");

	if (!t) {
		return;
	}

	/* Clear the template. */
	c = t->children;
	while (c) {
		xmlNodePtr n;
		n = c->next;
		xmlUnlinkNode(c);
		xmlFreeNode(c);
		c = n;
	}

	/* Parse the format string and generate the new template. */
	for (i = 0; fmt[i]; ++i) {
		xmlChar s[2] = {0};

		if (fmt[i] == FMTSTR_DELIM) {
			if (fmt[i + 1] == FMTSTR_DELIM) {
				s[0] = FMTSTR_DELIM;
				xmlNewChild(t, t->nsDef, BAD_CAST "text", s);
				++i;
			} else {
				const char *k, *e;
				int n;

				k = fmt + i + 1;
				e = strchr(k, FMTSTR_DELIM);
				if (!e) break;
				n = e - k;

				if (strncmp(k, "name", n) == 0) {
					c = xmlNewChild(t, t->nsDef, BAD_CAST "call-template", NULL);
					xmlSetProp(c, BAD_CAST "name", BAD_CAST "applicPropertyName");
				} else if (strncmp(k, "values", n) == 0) {
					c = xmlNewChild(t, t->nsDef, BAD_CAST "call-template", NULL);
					xmlSetProp(c, BAD_CAST "name", BAD_CAST "applicPropertyVal");
				}

				i += n + 1;
			}
		} else {
			if (fmt[i] == '\\') {
				switch (fmt[i + 1]) {
					case 'n': s[0] = '\n'; ++i; break;
					case 't': s[0] = '\t'; ++i; break;
					default: s[0] = fmt[i]; break;
				}
			} else {
				s[0] = fmt[i];
			}
			xmlNewChild(t, t->nsDef, BAD_CAST "text", s);
		}
	}
}

static void generateDisplayText(xmlDocPtr doc, xmlNodePtr acts, xmlNodePtr ccts, xsltStylesheetPtr style)
{
	xmlDocPtr res, muxdoc;
	xmlNodePtr mux, cur, muxacts, muxccts, new, old;
	const char *params[3];

	muxdoc = xmlNewDoc(BAD_CAST "1.0");
	mux = xmlNewNode(NULL, BAD_CAST "mux");
	xmlDocSetRootElement(muxdoc, mux);

	xmlAddChild(mux, xmlCopyNode(xmlDocGetRootElement(doc), 1));
	muxacts = xmlNewChild(mux, NULL, BAD_CAST "acts", NULL);
	muxccts = xmlNewChild(mux, NULL, BAD_CAST "ccts", NULL);
	for (cur = acts->children; cur; cur = cur->next) {
		xmlDocPtr act;
		xmlChar *path;
		path = xmlNodeGetContent(cur);
		act = read_xml_doc((char *) path);
		xmlAddChild(muxacts, xmlCopyNode(xmlDocGetRootElement(act), 1));
		xmlFreeDoc(act);
		xmlFree(path);
	}
	for (cur = ccts->children; cur; cur = cur->next) {
		xmlDocPtr cct;
		xmlChar *path;
		path = xmlNodeGetContent(cur);
		cct = read_xml_doc((char *) path);
		xmlAddChild(muxccts, xmlCopyNode(xmlDocGetRootElement(cct), 1));
		xmlFreeDoc(cct);
		xmlFree(path);
	}

	params[0] = "overwrite-display-text";
	params[1] = overwriteDispText ? "true()" : "false()";
	params[2] = NULL;

	res = xsltApplyStylesheet(style, muxdoc, params);

	new = xmlCopyNode(firstXPathNode(res, NULL, "/mux/dmodule"), 1);
	old = xmlDocSetRootElement(doc, new);
	xmlFreeNode(old);

	xmlFreeDoc(res);
	xmlFreeDoc(muxdoc);
}

static void processDmodule(xmlNodePtr dmodule)
{
	xmlXPathContextPtr ctx;
	xmlXPathObjectPtr obj;

	ctx = xmlXPathNewContext(dmodule->doc);
	ctx->node = dmodule;
	obj = xmlXPathEvalExpression(applicElemsXPath, ctx);

	processNodeSet(obj->nodesetval);
	removeDuplicates(obj->nodesetval);
	addDmApplic(dmodule);

	xmlXPathFreeObject(obj);
	xmlXPathFreeContext(ctx);
}

static void processDmodules(xmlNodeSetPtr dmodules)
{
	int i;

	for (i = 0; i < dmodules->nodeNr; ++i) {
		processDmodule(dmodules->nodeTab[i]);
	}
}

/* Find a data module filename in the current directory based on the dmRefIdent
 * element. */
static bool find_dmod_fname(char *dst, xmlNodePtr dmRefIdent)
{
	char *model_ident_code;
	char *system_diff_code;
	char *system_code;
	char *sub_system_code;
	char *sub_sub_system_code;
	char *assy_code;
	char *disassy_code;
	char *disassy_code_variant;
	char *info_code;
	char *info_code_variant;
	char *item_location_code;
	char *learn_code;
	char *learn_event_code;
	char code[64];
	xmlNodePtr dmCode, issueInfo, language;

	dmCode = firstXPathNode(NULL, dmRefIdent, "dmCode|avee");
	issueInfo = firstXPathNode(NULL, dmRefIdent, "issueInfo|issno");
	language = firstXPathNode(NULL, dmRefIdent, "language");

	model_ident_code     = (char *) first_xpath_value(NULL, dmCode, "modelic|@modelIdentCode");
	system_diff_code     = (char *) first_xpath_value(NULL, dmCode, "sdc|@systemDiffCode");
	system_code          = (char *) first_xpath_value(NULL, dmCode, "chapnum|@systemCode");
	sub_system_code      = (char *) first_xpath_value(NULL, dmCode, "section|@subSystemCode");
	sub_sub_system_code  = (char *) first_xpath_value(NULL, dmCode, "subsect|@subSubSystemCode");
	assy_code            = (char *) first_xpath_value(NULL, dmCode, "subject|@assyCode");
	disassy_code         = (char *) first_xpath_value(NULL, dmCode, "discode|@disassyCode");
	disassy_code_variant = (char *) first_xpath_value(NULL, dmCode, "discodev|@disassyCodeVariant");
	info_code            = (char *) first_xpath_value(NULL, dmCode, "incode|@infoCode");
	info_code_variant    = (char *) first_xpath_value(NULL, dmCode, "incodev|@infoCodeVariant");
	item_location_code   = (char *) first_xpath_value(NULL, dmCode, "itemloc|@itemLocationCode");
	learn_code           = (char *) first_xpath_value(NULL, dmCode, "@learnCode");
	learn_event_code     = (char *) first_xpath_value(NULL, dmCode, "@learnEventCode");

	snprintf(code, 64, "DMC-%s-%s-%s-%s%s-%s-%s%s-%s%s-%s",
		model_ident_code,
		system_diff_code,
		system_code,
		sub_system_code,
		sub_sub_system_code,
		assy_code,
		disassy_code,
		disassy_code_variant,
		info_code,
		info_code_variant,
		item_location_code);

	xmlFree(model_ident_code);
	xmlFree(system_diff_code);
	xmlFree(system_code);
	xmlFree(sub_system_code);
	xmlFree(sub_sub_system_code);
	xmlFree(assy_code);
	xmlFree(disassy_code);
	xmlFree(disassy_code_variant);
	xmlFree(info_code);
	xmlFree(info_code_variant);
	xmlFree(item_location_code);

	if (learn_code) {
		char learn[8];
		snprintf(learn, 8, "-%s%s", learn_code, learn_event_code);
		strcat(code, learn);
	}

	xmlFree(learn_code);
	xmlFree(learn_event_code);

	if (!no_issue) {
		if (issueInfo) {
			char *issue_number;
			char *in_work;
			char iss[8];

			issue_number = (char *) first_xpath_value(NULL, issueInfo, "@issno|@issueNumber");
			in_work      = (char *) first_xpath_value(NULL, issueInfo, "@inwork|@inWork");

			snprintf(iss, 8, "_%s-%s", issue_number, in_work ? in_work : "00");
			strcat(code, iss);

			xmlFree(issue_number);
			xmlFree(in_work);
		} else if (language) {
			strcat(code, "_\?\?\?-\?\?");
		}
	}

	if (language) {
		char *language_iso_code;
		char *country_iso_code;
		char lang[8];

		language_iso_code = (char *) first_xpath_value(NULL, language, "@language|@languageIsoCode");
		country_iso_code  = (char *) first_xpath_value(NULL, language, "@country|@countryIsoCode");

		snprintf(lang, 8, "_%s-%s", language_iso_code, country_iso_code);
		strcat(code, lang);

		xmlFree(language_iso_code);
		xmlFree(country_iso_code);
	}

	if (find_csdb_object(dst, search_dir, code, is_dm, recursive_search)) {
		return true;
	}

	if (verbosity >= NORMAL) {
		fprintf(stderr, W_MISSING_REF, code);
	}
	return false;
}

/* Find the filename of a referenced ACT data module. */
static bool find_act_fname(char *dst, xmlDocPtr doc)
{
	xmlNodePtr actref;
	actref = firstXPathNode(doc, NULL, "//applicCrossRefTableRef/dmRef/dmRefIdent|//actref/refdm");
	return actref && find_dmod_fname(dst, actref);
}

/* Find the filename of a referenced PCT data module via the ACT. */
static bool find_cct_fname(char *dst, xmlDocPtr act)
{
	xmlNodePtr pctref;
	bool found;

	pctref = firstXPathNode(act, NULL, "//condCrossRefTableRef/dmRef/dmRefIdent|//cctref/refdm");
	found = pctref && find_dmod_fname(dst, pctref);

	return found;
}

/* Add cross-reference tables by searching for them in the current directory. */
static void find_cross_ref_tables(xmlDocPtr doc, xmlNodePtr acts, xmlNodePtr ccts)
{
	char act_fname[PATH_MAX];
	xmlDocPtr act = NULL;

	if (find_act_fname(act_fname, doc) && (act = read_xml_doc(act_fname))) {
		char cct_fname[PATH_MAX];

		xmlNewChild(acts, NULL, BAD_CAST "act", BAD_CAST act_fname);

		if (find_cct_fname(cct_fname, act)) {
			xmlNewChild(ccts, NULL, BAD_CAST "cct", BAD_CAST cct_fname);
		}

		xmlFreeDoc(act);
	}
}

/* Add tags containing the display text of the referenecd applic annotation. */
static void addTags(xmlDocPtr doc, const char *tags)
{
	xmlDocPtr styledoc, src, res;
	xsltStylesheetPtr style;
	const char *params[3];
	xmlNodePtr old;

	styledoc = read_xml_mem((const char *) addTags_xsl, addTags_xsl_len);
	style = xsltParseStylesheetDoc(styledoc);

	params[0] = "mode";
	if (strcmp(tags, "comment") == 0) {
		params[1] = "'comment'";
	} else if (strcmp(tags, "pi") == 0) {
		params[1] = "'pi'";
	} else {
		params[1] = "'remove'";
	}
	params[2] = NULL;

	src = xmlCopyDoc(doc, 1);
	res = xsltApplyStylesheet(style, src, params);
	xmlFreeDoc(src);

	old = xmlDocSetRootElement(doc, xmlCopyNode(xmlDocGetRootElement(res), 1));
	xmlFreeNode(old);

	xmlFreeDoc(res);
	xsltFreeStylesheet(style);
}

/* Recursively delete display text nodes. */
static void deleteDisplayTextNode(xmlNodePtr node)
{
	xmlNodePtr cur;

	/* 3.0-: displaytext
	 * 4.0+: displayText
	 *
	 * Ignore display text nodes without any siblings. */
	if ((xmlStrcmp(node->name, BAD_CAST "displayText") == 0 || xmlStrcmp(node->name, BAD_CAST "displaytext") == 0) && xmlChildElementCount(node->parent) > 1) {
		xmlUnlinkNode(node);
		xmlFreeNode(node);
		return;
	}

	/* Call recursively on all children. */
	cur = node->children;
	while (cur) {
		xmlNodePtr next = cur->next;
		deleteDisplayTextNode(cur);
		cur = next;
	}

}

/* Delete all display text nodes in a document. */
static void deleteDisplayText(xmlDocPtr doc)
{
	deleteDisplayTextNode(xmlDocGetRootElement(doc));
}

static void processFile(const char *in, const char *out, bool process,
	bool genDispText, bool delDispText, xmlNodePtr acts, xmlNodePtr ccts,
	bool findcts, xsltStylesheetPtr style, const char *tags)
{
	xmlDocPtr doc;
	xmlXPathContextPtr ctx;
	xmlXPathObjectPtr obj;
	xmlNodePtr all_acts, all_ccts;

	if (verbosity >= VERBOSE) {
		fprintf(stderr, I_PROCESS, in);
	}

	doc = read_xml_doc(in);

	if (findcts) {
		/* Copy the user-defined ACTs/CCTs. */
		all_acts = xmlCopyNode(acts, 1);
		all_ccts = xmlCopyNode(ccts, 1);
		/* Find the ACT/CCT referenced by the current DM. */
		find_cross_ref_tables(doc, all_acts, all_ccts);
	} else {
		/* Only use the user-defined ACTs/CCTs. */
		all_acts = acts;
		all_ccts = ccts;
	}

	if (process) {
		ctx = xmlXPathNewContext(doc);
		obj = xmlXPathEvalExpression(BAD_CAST "//dmodule", ctx);

		processDmodules(obj->nodesetval);

		xmlXPathFreeObject(obj);
		xmlXPathFreeContext(ctx);
	}

	if (delDispText) {
		deleteDisplayText(doc);
	} else if (genDispText) {
		generateDisplayText(doc, all_acts, all_ccts, style);
	}

	if (tags) {
		addTags(doc, tags);
	}

	save_xml_doc(doc, out);

	/* The next data module could reference a different ACT/CCT, so
	 * the list must be cleared. */
	if (findcts) {
		xmlFreeNode(all_acts);
		xmlFreeNode(all_ccts);
	}

	xmlFreeDoc(doc);
}

static void process_list(const char *path, bool overwrite, bool process,
	bool genDispText, bool delDispText, xmlNodePtr acts, xmlNodePtr ccts,
	bool findcts, xsltStylesheetPtr style, const char *tags)
{
	FILE *f;
	char line[PATH_MAX];

	if (path) {
		if (!(f = fopen(path, "r"))) {
			if (verbosity >= NORMAL) {
				fprintf(stderr, E_BAD_LIST, path);
			}
			return;
		}
	} else {
		f = stdin;
	}

	while (fgets(line, PATH_MAX, f)) {
		strtok(line, "\t\r\n");
		processFile(line, overwrite ? line : "-", process, genDispText, delDispText, acts, ccts, findcts, style, tags);
	}

	if (path) {
		fclose(f);
	}
}

static void show_help(void)
{
	puts("Usage:");
	puts("  " PROG_NAME " [options] [<object> ...]");
	puts("");
	puts("Options:");
	puts("  -., --dump-disptext   Dump the built-in .disptext file.");
	puts("  -,, --dump-xsl        Dump the built-in XSLT for generating display text.");
	puts("  -A, --act <ACT>       Use <ACT> when generating display text.");
	puts("  -a, --id <ID>  Use <ID> for DM-level applic.");
	puts("  -C, --cct <CCT>       Use <CCT> when generating display text.");
	puts("  -c, --search          Search for ACT/CCT data modules.");
	puts("  -D, --delete          Remove all display text.");
	puts("  -d, --dir <dir>       Directory to start search for ACT/CCT in.");
	puts("  -F, --format <fmt>    Use a custom format string for generating display text.");
	puts("  -f, --overwrite       Overwrite input file(s).");
	puts("  -G, --disptext        Specify .disptext file.");
	puts("  -g, --generate        Generate display text for applicability annotations.");
	puts("  -k, --keep            Do not overwrite existing display text.");
	puts("  -l, --list            Treat input as list of modules.");
	puts("  -N, --omit-issue      Assume issue/inwork number are omitted.");
	puts("  -p, --presentation    Convert semantic applicability to presentation applicability.");
	puts("  -q, --quiet           Quiet mode.");
	puts("  -r, --recursive       Search for ACT/CCT recursively.");
	puts("  -t, --tags <mode>     Add display text tags before elements with applicability.");
	puts("  -v, --verbose         Verbose output.");
	puts("  -x, --xsl <XSL>       Use custom XSLT script to generate display text.");
	puts("  -h, -?, --help        Show help/usage message.");
	puts("  --version             Show version information.");
	puts("  <object> ...          CSDB objects to process.");
	LIBXML2_PARSE_LONGOPT_HELP
}

static void show_version(void)
{
	printf("%s (s1kd-tools) %s\n", PROG_NAME, VERSION);
	printf("Using libxml %s, libxslt %s and libexslt %s\n",
		xmlParserVersion, xsltEngineVersion, exsltLibraryVersion);
}

int main(int argc, char **argv)
{
	int i;
	bool overwrite = false;
	bool genDispText = false;
	bool process = false;
	bool delDispText = false;
	bool findcts = false;
	bool islist = false;
	char *format = NULL;
	char *tags = NULL;
	char *customGenDispTextFile = NULL;
	xmlDocPtr disptext = NULL;
	xmlDocPtr styledoc;
	xsltStylesheetPtr style;
	
	xmlNodePtr acts, ccts;

	const char *sopts = ".,A:a:C:cDd:F:fG:gklNpqrt:vx:h?";
	struct option lopts[] = {
		{"version"      , no_argument      , 0, 0},
		{"help"         , no_argument      , 0, 'h'},
		{"dump-disptext", no_argument      , 0, '.'},
		{"dump-xsl"     , no_argument      , 0, ','},
		{"act"          , required_argument, 0, 'A'},
		{"id"           , required_argument, 0, 'a'},
		{"cct"          , required_argument, 0, 'C'},
		{"search"       , no_argument      , 0, 'c'},
		{"delete"       , no_argument      , 0, 'D'},
		{"dir"          , required_argument, 0, 'd'},
		{"format"       , required_argument, 0, 'F'},
		{"overwrite"    , no_argument      , 0, 'f'},
		{"disptext"     , required_argument, 0, 'G'},
		{"generate"     , no_argument      , 0, 'g'},
		{"keep"         , no_argument      , 0, 'k'},
		{"list"         , no_argument      , 0, 'l'},
		{"omit-issue"   , no_argument      , 0, 'N'},
		{"presentation" , no_argument      , 0, 'p'},
		{"quiet"        , no_argument      , 0, 'q'},
		{"recursive"    , no_argument      , 0, 'r'},
		{"tags"         , required_argument, 0, 't'},
		{"verbose"      , no_argument      , 0, 'v'},
		{"xsl"          , required_argument, 0, 'x'},
		LIBXML2_PARSE_LONGOPT_DEFS
		{0, 0, 0, 0}
	};
	int loptind = 0;

	exsltRegisterAll();

	dmApplicId = xmlStrdup(DEFAULT_DM_APPLIC_ID);

	applicElemsXPath = xmlStrndup(elements_list, elements_list_len);

	acts = xmlNewNode(NULL, BAD_CAST "acts");
	ccts = xmlNewNode(NULL, BAD_CAST "ccts");

	search_dir = strdup(".");

	while ((i = getopt_long(argc, argv, sopts, lopts, &loptind)) != -1) {
		switch (i) {
			case 0:
				if (strcmp(lopts[loptind].name, "version") == 0) {
					show_version();
					return 0;
				}
				LIBXML2_PARSE_LONGOPT_HANDLE(lopts, loptind, optarg)
				break;
			case '.':
				dumpDispText();
				return 0;
			case ',':
				dumpGenDispTextXsl();
				return 0;
			case 'A':
				xmlNewChild(acts, NULL, BAD_CAST "act", BAD_CAST optarg);
				findcts = false;
				break;
			case 'a':
				xmlFree(dmApplicId);
				dmApplicId = xmlStrdup(BAD_CAST optarg);
				break;
			case 'C':
				xmlNewChild(ccts, NULL, BAD_CAST "cct", BAD_CAST optarg);
				findcts = false;
				break;
			case 'c':
				findcts = true;
				break;
			case 'D':
				delDispText = true;
				break;
			case 'd':
				free(search_dir);
				search_dir = strdup(optarg);
				break;
			case 'F':
				genDispText = true;
				free(format);
				format = strdup(optarg);
				break;
			case 'f':
				overwrite = true;
				break;
			case 'G':
				genDispText = true;
				xmlFreeDoc(disptext);
				disptext = read_xml_doc(optarg);
				break;
			case 'g':
				genDispText = true;
				break;
			case 'k':
				overwriteDispText = false;
				break;
			case 'l':
				islist = true;
				break;
			case 'N':
				no_issue = true;
				break;
			case 'p':
				process = true;
				break;
			case 'q':
				--verbosity;
				break;
			case 'r':
				recursive_search = true;
				break;
			case 't':
				free(tags);
				tags = strdup(optarg);
				break;
			case 'v':
				++verbosity;
				break;
			case 'x':
				genDispText = true;
				free(customGenDispTextFile);
				customGenDispTextFile = strdup(optarg);
				break;
			case 'h':
			case '?':
				show_help();
				return 0;
		}
	}

	if (customGenDispTextFile == NULL) {
		if (disptext == NULL) {
			char disptext_fname[PATH_MAX];

			if (find_config(disptext_fname, DEFAULT_DISPTEXT_FNAME)) {
				disptext = read_xml_doc(disptext_fname);
			} else {
				disptext = read_xml_mem((const char *) disptext_xml, disptext_xml_len);
			}
		}

		styledoc = parse_disptext(disptext);
	} else {
		styledoc = read_xml_doc(customGenDispTextFile);
	}

	if (format) {
		apply_format_str(styledoc, format);
	}
	style = xsltParseStylesheetDoc(styledoc);

	if (optind >= argc) {
		if (islist) {
			process_list(NULL, overwrite, process, genDispText, delDispText, acts, ccts, findcts, style, tags);
		} else {
			processFile("-", "-", process, genDispText, delDispText, acts, ccts, findcts, style, tags);
		}
	} else {
		for (i = optind; i < argc; ++i) {
			if (islist) {
				process_list(argv[i], overwrite, process, genDispText, delDispText, acts, ccts, findcts, style, tags);
			} else {
				processFile(argv[i], overwrite ? argv[i] : "-", process, genDispText, delDispText, acts, ccts, findcts, style, tags);
			}
		}
	}

	xmlFree(dmApplicId);
	xmlFree(applicElemsXPath);

	xmlFreeNode(acts);
	xmlFreeNode(ccts);

	free(customGenDispTextFile);
	free(search_dir);
	free(format);
	free(tags);
	xmlFreeDoc(disptext);
	xsltFreeStylesheet(style);

	xsltCleanupGlobals();
	xmlCleanupParser();

	return 0;
}


/ gopher://khzae.net/0/s1kd/s1kd-tools/src/tools/s1kd-aspp/s1kd-aspp.c
Styles: Light Dark Classic