15 #include <libxml/HTMLtree.h>
24 static const char *stylesheet_default =
25 ".bold { font-weight: bold }\n"
26 ".maint { color: blue }\n"
27 ".offline { color: red }\n"
28 ".online { color: green }\n"
29 ".rsc-failed { color: red }\n"
30 ".rsc-failure-ignored { color: yellow }\n"
31 ".rsc-managed { color: yellow }\n"
32 ".rsc-multiple { color: orange }\n"
33 ".rsc-ok { color: green }\n"
34 ".standby { color: orange }\n"
35 ".warning { color: red, font-weight: bold }";
37 static gboolean cgi_output = FALSE;
38 static char *stylesheet_link = NULL;
39 static char *title = NULL;
40 static GSList *extra_headers = NULL;
43 {
"html-cgi", 0, 0, G_OPTION_ARG_NONE, &cgi_output,
44 "Add CGI headers (requires --output-as=html)",
47 {
"html-stylesheet", 0, 0, G_OPTION_ARG_STRING, &stylesheet_link,
48 "Link to an external stylesheet (requires --output-as=html)",
51 {
"html-title", 0, 0, G_OPTION_ARG_STRING, &title,
52 "Specify a page title (requires --output-as=html)",
58 typedef struct private_data_s {
72 xmlFreeNode(priv->root);
73 g_queue_free(priv->parent_q);
74 g_slist_free(priv->errors);
84 if (out->
priv != NULL) {
88 if (out->
priv == NULL) {
95 priv->parent_q = g_queue_new();
98 xmlCreateIntSubset(priv->root->doc, (
pcmkXmlStr)
"html", NULL, NULL);
101 g_queue_push_tail(priv->parent_q, priv->root);
110 add_error_node(gpointer
data, gpointer user_data) {
111 char *str = (
char *)
data;
119 htmlNodePtr head_node = NULL;
120 htmlNodePtr charset_node = NULL;
122 if (cgi_output && print) {
123 fprintf(out->
dest,
"Content-Type: text/html\n\n");
130 head_node = xmlNewNode(NULL, (
pcmkXmlStr)
"head");
132 if (title != NULL ) {
134 }
else if (out->
request != NULL) {
142 for (
int i = 0; i < g_slist_length(extra_headers); i++) {
143 xmlAddChild(head_node, xmlCopyNode(g_slist_nth_data(extra_headers, i), 1));
154 if (stylesheet_link != NULL) {
160 xmlAddPrevSibling(priv->root->children, head_node);
162 if (g_slist_length(priv->errors) > 0) {
164 g_slist_foreach(priv->errors, add_error_node, (gpointer) out);
169 htmlDocDump(out->
dest, priv->root->doc);
180 if (priv == NULL || priv->root == NULL) {
184 finish_reset_common(out, exit_status, print);
186 if (copy_dest != NULL) {
190 g_slist_free_full(extra_headers, (GDestroyNotify) xmlFreeNode);
197 out->
dest = freopen(NULL,
"w", out->
dest);
200 if (out->
priv != NULL) {
201 finish_reset_common(out,
CRM_EX_OK,
true);
210 const char *proc_stdout,
const char *proc_stderr) {
220 if (proc_stdout != NULL) {
224 if (proc_stderr != NULL) {
254 va_start(ap, format);
255 len = vasprintf(&buf, format, ap);
259 priv->errors = g_slist_append(priv->errors, buf);
270 htmlNodePtr node = NULL;
282 const
char *plural_noun, const
char *format, ...) {
285 xmlNodePtr node = NULL;
293 q_len = g_queue_get_length(priv->parent_q);
298 if (format != NULL) {
303 va_start(ap, format);
304 len = vasprintf(&buf, format, ap);
318 g_queue_push_tail(priv->parent_q, node);
325 htmlNodePtr item_node = NULL;
332 va_start(ap, format);
333 len = vasprintf(&buf, format, ap);
357 g_queue_pop_tail(priv->parent_q);
361 if (g_queue_get_length(priv->parent_q) > 2) {
375 if (retval == NULL) {
380 retval->
request = argv == NULL ? NULL : g_strjoinv(
" ", argv);
382 retval->
init = html_init;
384 retval->
finish = html_finish;
385 retval->
reset = html_reset;
391 retval->
version = html_version;
392 retval->
info = html_info;
393 retval->
err = html_err;
408 const char *class_name,
const char *text) {
411 if (class_name != NULL) {
424 htmlNodePtr header_node;
431 char *key = va_arg(ap,
char *);
438 value = va_arg(ap,
char *);
442 extra_headers = g_slist_append(extra_headers, header_node);