/* Support for dumping to the file on startup (w/o bfu) */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include /* NetBSD flavour */ #ifdef HAVE_SYS_SIGNAL_H #include #endif #ifdef HAVE_FCNTL_H #include /* OS/2 needs this after sys/types.h */ #endif #ifdef HAVE_UNISTD_H #include #endif #include "elinks.h" #include "cache/cache.h" #include "config/options.h" #include "document/document.h" #include "document/options.h" #include "document/renderer.h" #include "document/view.h" #include "intl/charsets.h" #include "intl/gettext/libintl.h" #include "main/select.h" #include "main/main.h" #include "network/connection.h" #include "network/state.h" #include "osdep/ascii.h" #include "osdep/osdep.h" #include "protocol/protocol.h" #include "protocol/uri.h" #include "session/download.h" #include "terminal/color.h" #include "terminal/hardio.h" #include "terminal/terminal.h" #include "util/memory.h" #include "util/string.h" #include "viewer/dump/dump.h" #include "viewer/text/view.h" #include "viewer/text/vs.h" static int dump_pos; static struct download dump_download; static int dump_redir_count = 0; static int dump_to_file_16(struct document *document, int fd); #if defined(CONFIG_88_COLORS) || defined(CONFIG_256_COLORS) static int dump_to_file_256(struct document *document, int fd); #endif #ifdef CONFIG_TRUE_COLOR static int dump_to_file_true_color(struct document *document, int fd); #endif /* This dumps the given @cached's source onto @fd nothing more. It returns 0 if it * all went fine and 1 if something isn't quite right and we should terminate * ourselves ASAP. */ static int dump_source(int fd, struct download *download, struct cache_entry *cached) { struct fragment *frag; if (!cached) return 0; nextfrag: foreach (frag, cached->frag) { int d = dump_pos - frag->offset; int l, w; if (d < 0 || frag->length <= d) continue; l = frag->length - d; w = hard_write(fd, frag->data + d, l); if (w != l) { detach_connection(download, dump_pos); if (w < 0) ERROR(gettext("Can't write to stdout: %s"), (unsigned char *) strerror(errno)); else ERROR(gettext("Can't write to stdout.")); program.retval = RET_ERROR; return 1; } dump_pos += w; detach_connection(download, dump_pos); goto nextfrag; } return 0; } /* This dumps the given @cached's formatted output onto @fd. */ static void dump_formatted(int fd, struct download *download, struct cache_entry *cached) { struct document_options o; struct document_view formatted; struct view_state vs; int width; if (!cached) return; memset(&formatted, 0, sizeof(formatted)); init_document_options(&o); width = get_opt_int("document.dump.width"); set_box(&o.box, 0, 1, width, DEFAULT_TERMINAL_HEIGHT); o.cp = get_opt_codepage("document.dump.codepage"); o.color_mode = get_opt_int("document.dump.color_mode"); o.plain = 0; o.frames = 0; o.links_numbering = get_opt_bool("document.dump.numbering"); init_vs(&vs, cached->uri, -1); render_document(&vs, &formatted, &o); switch(o.color_mode) { case COLOR_MODE_DUMP: case COLOR_MODE_MONO: /* FIXME: inversion */ dump_to_file(formatted.document, fd); break; default: /* If the desired color mode was not compiled in, * use 16 colors. */ case COLOR_MODE_16: dump_to_file_16(formatted.document, fd); break; #ifdef CONFIG_88_COLORS case COLOR_MODE_88: dump_to_file_256(formatted.document, fd); break; #endif #ifdef CONFIG_256_COLORS case COLOR_MODE_256: dump_to_file_256(formatted.document, fd); break; #endif #ifdef CONFIG_TRUE_COLOR case COLOR_MODE_TRUE_COLOR: dump_to_file_true_color(formatted.document, fd); break; #endif } detach_formatted(&formatted); destroy_vs(&vs, 1); } static unsigned char * subst_url(unsigned char *str, struct string *url) { struct string string; if (!init_string(&string)) return NULL; while (*str) { int p; for (p = 0; str[p] && str[p] != '%' && str[p] != '\\'; p++); add_bytes_to_string(&string, str, p); str += p; if (*str == '\\') { unsigned char ch; str++; switch (*str) { case 'f': ch = '\f'; break; case 'n': ch = '\n'; break; case 't': ch = '\t'; break; default: ch = *str; } if (*str) { add_char_to_string(&string, ch); str++; } continue; } else if (*str != '%') { break; } str++; switch (*str) { case 'u': if (url) add_string_to_string(&string, url); break; } if (*str) str++; } return string.source; } static void dump_print(unsigned char *option, struct string *url) { unsigned char *str = get_opt_str(option); if (str) { unsigned char *realstr = subst_url(str, url); if (realstr) { printf("%s", realstr); fflush(stdout); mem_free(realstr); } } } static void dump_loading_callback(struct download *download, void *p) { struct cache_entry *cached = download->cached; int fd = get_output_handle(); if (fd == -1) return; if (cached && cached->redirect && dump_redir_count++ < MAX_REDIRECTS) { struct uri *uri = cached->redirect; cancel_download(download, 0); load_uri(uri, cached->uri, download, PRI_MAIN, 0, -1); return; } if (is_in_queued_state(download->state)) return; if (get_cmd_opt_bool("dump")) { if (is_in_transfering_state(download->state)) return; dump_formatted(fd, download, cached); } else { if (dump_source(fd, download, cached) > 0) goto terminate; if (is_in_progress_state(download->state)) return; } if (!is_in_state(download->state, S_OK)) { usrerror(get_state_message(download->state, NULL)); program.retval = RET_ERROR; goto terminate; } terminate: program.terminate = 1; dump_next(NULL); } static void dump_start(unsigned char *url) { unsigned char *wd = get_cwd(); struct uri *uri = get_translated_uri(url, wd); mem_free_if(wd); if (!uri || get_protocol_external_handler(NULL, uri)) { usrerror(gettext("URL protocol not supported (%s)."), url); goto terminate; } dump_download.callback = (download_callback_T *) dump_loading_callback; dump_pos = 0; if (load_uri(uri, NULL, &dump_download, PRI_MAIN, 0, -1)) { terminate: dump_next(NULL); program.terminate = 1; program.retval = RET_SYNTAX; } if (uri) done_uri(uri); } void dump_next(LIST_OF(struct string_list_item) *url_list) { static INIT_LIST_OF(struct string_list_item, todo_list); static INIT_LIST_OF(struct string_list_item, done_list); struct string_list_item *item; if (url_list) { /* Steal all them nice list items but keep the same order */ while (!list_empty(*url_list)) { item = url_list->next; del_from_list(item); add_to_list_end(todo_list, item); } } /* Dump each url list item one at a time */ if (!list_empty(todo_list)) { static int first = 1; program.terminate = 0; item = todo_list.next; del_from_list(item); add_to_list(done_list, item); if (!first) { dump_print("document.dump.separator", NULL); } else { first = 0; } dump_print("document.dump.header", &item->string); dump_start(item->string.source); /* XXX: I think it ought to print footer at the end of * the whole dump (not only this file). Testing required. * --pasky */ dump_print("document.dump.footer", &item->string); } else { free_string_list(&done_list); program.terminate = 1; } } /* Using this function in dump_to_file() is unfortunately slightly slower than * the current code. However having this here instead of in the scripting * backends is better. */ struct string * add_document_to_string(struct string *string, struct document *document) { int y; assert(string && document); if_assert_failed return NULL; #ifdef CONFIG_UTF8 if (is_cp_utf8(document->options.cp)) goto utf8; #endif /* CONFIG_UTF8 */ for (y = 0; y < document->height; y++) { int white = 0; int x; for (x = 0; x < document->data[y].length; x++) { struct screen_char *pos = &document->data[y].chars[x]; unsigned char data = pos->data; unsigned int frame = (pos->attr & SCREEN_ATTR_FRAME); if (!isscreensafe(data)) { white++; continue; } else { if (frame && data >= 176 && data < 224) data = frame_dumb[data - 176]; if (data <= ' ') { /* Count spaces. */ white++; } else { /* Print spaces if any. */ if (white) { add_xchar_to_string(string, ' ', white); white = 0; } add_char_to_string(string, data); } } } add_char_to_string(string, '\n'); } #ifdef CONFIG_UTF8 goto end; utf8: for (y = 0; y < document->height; y++) { int white = 0; int x; for (x = 0; x < document->data[y].length; x++) { struct screen_char *pos = &document->data[y].chars[x]; unicode_val_T data = pos->data; unsigned int frame = (pos->attr & SCREEN_ATTR_FRAME); if (!isscreensafe_ucs(data)) { white++; continue; } else { if (frame && data >= 176 && data < 224) data = frame_dumb[data - 176]; if (data <= ' ') { /* Count spaces. */ white++; } else if (data == UCS_NO_CHAR) { /* This is the second cell of * a double-cell character. */ } else { /* Print spaces if any. */ if (white) { add_xchar_to_string(string, ' ', white); white = 0; } add_to_string(string, encode_utf8(data)); } } } add_char_to_string(string, '\n'); } end: #endif /* CONFIG_UTF8 */ return string; } #define D_BUF 65536 static int write_char(unsigned char c, int fd, unsigned char *buf, int *bptr) { buf[(*bptr)++] = c; if ((*bptr) >= D_BUF) { if (hard_write(fd, buf, (*bptr)) != (*bptr)) return -1; (*bptr) = 0; } return 0; } static int write_color_16(unsigned char color, int fd, unsigned char *buf, int *bptr) { unsigned char bufor[] = "\033[0;30;40m"; unsigned char *data = bufor; int background = (color >> 4) & 7; int foreground = color & 7; bufor[5] += foreground; if (background) bufor[8] += background; else { bufor[6] = 'm'; bufor[7] = '\0'; } while(*data) { if (write_char(*data++, fd, buf, bptr)) return -1; } return 0; } #define DUMP_COLOR_MODE_16 #include "dump-color-mode.h" #undef DUMP_COLOR_MODE_16 /* configure --enable-debug uses gcc -Wall -Werror, and -Wall includes * -Wunused-function, so declaring or defining any unused function * would break the build. */ #if defined(CONFIG_88_COLORS) || defined(CONFIG_256_COLORS) static int write_color_256(unsigned char *str, unsigned char color, int fd, unsigned char *buf, int *bptr) { unsigned char bufor[16]; unsigned char *data = bufor; snprintf(bufor, 16, "\033[%s;5;%dm", str, color); while(*data) { if (write_char(*data++, fd, buf, bptr)) return -1; } return 0; } #define DUMP_COLOR_MODE_256 #include "dump-color-mode.h" #undef DUMP_COLOR_MODE_256 #endif /* defined(CONFIG_88_COLORS) || defined(CONFIG_256_COLORS) */ #ifdef CONFIG_TRUE_COLOR static int write_true_color(unsigned char *str, unsigned char *color, int fd, unsigned char *buf, int *bptr) { unsigned char bufor[24]; unsigned char *data = bufor; snprintf(bufor, 24, "\033[%s;2;%d;%d;%dm", str, color[0], color[1], color[2]); while(*data) { if (write_char(*data++, fd, buf, bptr)) return -1; } return 0; } #define DUMP_COLOR_MODE_TRUE #include "dump-color-mode.h" #undef DUMP_COLOR_MODE_TRUE #endif /* CONFIG_TRUE_COLOR */ #include "dump-color-mode.h" #undef D_BUF