/* $Id: backend.c,v 1.15 2010/08/08 09:53:42 htrb Exp $ */ #include <stdio.h> #include <string.h> #include <sys/types.h> #include "fm.h" #include <gc.h> #include "terms.h" /* Prototype declaration of internal functions */ #ifdef HAVE_READLINE #include <readline/readline.h> #else /* ! HAVE_READLINE */ static char *readline(char *); #endif /* ! HAVE_READLINE */ static TextList *split(char *); /* Prototype declaration of command functions */ static void get(TextList *); static void post(TextList *); static void set(TextList *); static void show(TextList *); static void quit(TextList *); static void help(TextList *); /* *INDENT-OFF* */ /* Table of command functions */ struct { const char *name; const char *option_string; const char *help; void (*func)(TextList*); } command_table[] = { {"get", "[-download_only] URL", "Retrieve URL.", get}, {"post", "[-download_only] [-target TARGET] [-charset CHARSET]" " [-enctype ENCTYPE] [-body BODY] [-boundary BOUNDARY] [-length LEN] URL", "Retrieve URL.", post}, {"set", "VARIABLE VALUE", "Set VALUE to VARIABLE.", set}, {"show", "VARIABLE", "Show value of VARIABLE.", show}, {"quit", "", "Quit program.", quit}, {"help", "", "Display help messages.", help}, {NULL, NULL, NULL, NULL}, }; /* *INDENT-ON* */ /* Prototype declaration of functions to manipulate configuration variables */ static void set_column(TextList *); static void show_column(TextList *); /* *INDENT-OFF* */ /* Table of configuration variables */ struct { const char *name; void (*set_func)(TextList*); void (*show_func)(TextList*); } variable_table[] = { {"column", set_column, show_column}, {NULL, NULL, NULL}, }; /* *INDENT-ON* */ static void print_headers(Buffer *buf, int len) { TextListItem *tp; if (buf->document_header) { for (tp = buf->document_header->first; tp; tp = tp->next) printf("%s\n", tp->ptr); } printf("w3m-current-url: %s\n", parsedURL2Str(&buf->currentURL)->ptr); if (buf->baseURL) printf("w3m-base-url: %s\n", parsedURL2Str(buf->baseURL)->ptr); printf("w3m-content-type: %s\n", buf->type); #ifdef USE_M17N if (buf->document_charset) printf("w3m-content-charset: %s\n", wc_ces_to_charset(buf->document_charset)); #endif if (len > 0) printf("w3m-content-length: %d\n", len); } static void internal_get(char *url, int flag, FormList *request) { Buffer *buf; backend_halfdump_buf = NULL; do_download = flag; buf = loadGeneralFile(url, NULL, NO_REFERER, 0, request); do_download = FALSE; if (buf != NULL && buf != NO_BUFFER) { if (is_html_type(buf->type) && backend_halfdump_buf) { TextLineListItem *p; Str first, last; int len = 0; for (p = backend_halfdump_buf->first; p; p = p->next) { p->ptr->line = Str_conv_to_halfdump(p->ptr->line); len += p->ptr->line->length + 1; } first = Strnew_charp("<pre>\n"); last = Strnew_m_charp("</pre><title>", html_quote(buf->buffername), "</title>\n", NULL); print_headers(buf, len + first->length + last->length); printf("\n"); printf("%s", first->ptr); for (p = backend_halfdump_buf->first; p; p = p->next) printf("%s\n", p->ptr->line->ptr); printf("%s", last->ptr); } else { if (!strcasecmp(buf->type, "text/plain")) { Line *lp; int len = 0; for (lp = buf->firstLine; lp; lp = lp->next) { len += lp->len; if (lp->lineBuf[lp->len - 1] != '\n') len++; } print_headers(buf, len); printf("\n"); saveBuffer(buf, stdout, TRUE); } else { print_headers(buf, 0); } } } } /* Command: get */ static void get(TextList *argv) { char *p, *url = NULL; int flag = FALSE; while ((p = popText(argv))) { if (!strcasecmp(p, "-download_only")) flag = TRUE; else url = p; } if (url) { internal_get(url, flag, NULL); } } /* Command: post */ static void post(TextList *argv) { FormList *request; char *p, *target = NULL, *charset = NULL, *enctype = NULL, *body = NULL, *boundary = NULL, *url = NULL; int flag = FALSE, length = 0; while ((p = popText(argv))) { if (!strcasecmp(p, "-download_only")) flag = TRUE; else if (!strcasecmp(p, "-target")) target = popText(argv); else if (!strcasecmp(p, "-charset")) charset = popText(argv); else if (!strcasecmp(p, "-enctype")) enctype = popText(argv); else if (!strcasecmp(p, "-body")) body = popText(argv); else if (!strcasecmp(p, "-boundary")) boundary = popText(argv); else if (!strcasecmp(p, "-length")) length = atol(popText(argv)); else url = p; } if (url) { request = newFormList(NULL, "post", charset, enctype, target, NULL, NULL); request->body = body; request->boundary = boundary; request->length = (length > 0) ? length : (body ? strlen(body) : 0); internal_get(url, flag, request); } } /* Command: set */ static void set(TextList *argv) { if (argv->nitem > 1) { int i; for (i = 0; variable_table[i].name; i++) { if (!strcasecmp(variable_table[i].name, argv->first->ptr)) { popText(argv); if (variable_table[i].set_func) variable_table[i].set_func(argv); break; } } } } /* Command: show */ static void show(TextList *argv) { if (argv->nitem >= 1) { int i; for (i = 0; variable_table[i].name; i++) { if (!strcasecmp(variable_table[i].name, argv->first->ptr)) { popText(argv); if (variable_table[i].show_func) variable_table[i].show_func(argv); break; } } } } /* Command: quit */ static void quit(TextList *argv) { #ifdef USE_COOKIE save_cookies(); #endif /* USE_COOKIE */ w3m_exit(0); } /* Command: help */ static void help(TextList *argv) { int i; for (i = 0; command_table[i].name; i++) printf("%s %s\n %s\n", command_table[i].name, command_table[i].option_string, command_table[i].help); } /* Sub command: set COLS */ static void set_column(TextList *argv) { if (argv->nitem == 1) { COLS = atol(argv->first->ptr); } } /* Sub command: show COLS */ static void show_column(TextList *argv) { fprintf(stdout, "column=%d\n", COLS); } /* Call appropriate command function based on given string */ static void call_command_function(char *str) { int i; TextList *argv = split(str); if (argv->nitem > 0) { for (i = 0; command_table[i].name; i++) { if (!strcasecmp(command_table[i].name, argv->first->ptr)) { popText(argv); if (command_table[i].func) command_table[i].func(argv); break; } } } } /* Main function */ int backend(void) { char *str; w3m_dump = 0; if (COLS == 0) COLS = DEFAULT_COLS; #ifdef USE_MOUSE use_mouse = FALSE; #endif /* USE_MOUSE */ if (backend_batch_commands) { while ((str = popText(backend_batch_commands))) call_command_function(str); } else { while ((str = readline("w3m> "))) call_command_function(str); } quit(NULL); return 0; } /* Dummy function of readline(). */ #ifndef HAVE_READLINE static char * readline(char *prompt) { Str s; fputs(prompt, stdout); fflush(stdout); s = Strfgets(stdin); if (feof(stdin) && (strlen(s->ptr) == 0)) return NULL; else return s->ptr; } #endif /* ! HAVE_READLINE */ /* Splits a string into a list of tokens and returns that list. */ static TextList * split(char *p) { int in_double_quote = FALSE, in_single_quote = FALSE; Str s = Strnew(); TextList *tp = newTextList(); for (; *p; p++) { switch (*p) { case '"': if (in_single_quote) Strcat_char(s, '"'); else in_double_quote = !in_double_quote; break; case '\'': if (in_double_quote) Strcat_char(s, '\''); else in_single_quote = !in_single_quote; break; case '\\': if (!in_single_quote) { /* Process escape characters. */ p++; switch (*p) { case 't': Strcat_char(s, '\t'); break; case 'r': Strcat_char(s, '\r'); break; case 'f': Strcat_char(s, '\f'); break; case 'n': Strcat_char(s, '\n'); break; case '\0': goto LAST; default: Strcat_char(s, *p); } } else { Strcat_char(s, *p); } break; case ' ': case '\t': case '\r': case '\f': case '\n': /* Separators are detected. */ if (in_double_quote || in_single_quote) { Strcat_char(s, *p); } else if (s->length > 0) { pushText(tp, s->ptr); s = Strnew(); } break; default: Strcat_char(s, *p); } } LAST: if (s->length > 0) pushText(tp, s->ptr); return tp; }