#include <HTUtils.h>
#include <HTFile.h>
#include <UCMap.h>
#include <LYUtils.h>
#include <LYStrings.h>
#include <LYStructs.h>
#include <LYGlobalDefs.h>
#include <LYCharSets.h>
#include <LYKeymap.h>
#include <LYJump.h>
#include <LYGetFile.h>
#include <LYCgi.h>
#include <LYCurses.h>
#include <LYSignal.h>
#include <LYBookmark.h>
#include <LYCookie.h>
#include <LYReadCFG.h>
#include <HTAlert.h>
#ifdef DIRED_SUPPORT
#include <LYLocal.h>
#endif /* DIRED_SUPPORT */
#include <LYexit.h>
#include <LYLeaks.h>
extern int HTNewsMaxChunk; /* Max news articles before chunking (HTNews.c) */
extern int HTNewsChunkSize; /* Number of news articles per chunk (HTNews.c) */
PUBLIC BOOLEAN have_read_cfg=FALSE;
PUBLIC BOOLEAN LYUseNoviceLineTwo=TRUE;
/*
* Translate a TRUE/FALSE field in a string buffer.
*/
PRIVATE int is_true ARGS1(
char *, string)
{
if (!strncasecomp(string,"TRUE",4))
return(TRUE);
else
return(FALSE);
}
/*
* Find an unescaped colon in a string buffer.
*/
PRIVATE char *find_colon ARGS1(
char *, buffer)
{
char ch, *buf = buffer;
if (buf == NULL)
return NULL;
while ((ch = *buf) != 0) {
if (ch == ':')
return buf;
if (ch == '\\') {
buf++;
if (*buf == 0)
break;
}
buf++;
}
return NULL;
}
/*
* Function for freeing the DOWNLOADER and UPLOADER menus list. - FM
*/
PRIVATE void free_item_list NOARGS
{
lynx_html_item_type *cur;
lynx_html_item_type *next;
cur = downloaders;
while (cur) {
next = cur->next;
FREE(cur->name);
FREE(cur->command);
FREE(cur);
cur = next;
}
downloaders = NULL;
#ifdef DIRED_SUPPORT
cur = uploaders;
while (cur) {
next = cur->next;
FREE(cur->name);
FREE(cur->command);
FREE(cur);
cur = next;
}
uploaders = NULL;
#endif /* DIRED_SUPPORT */
#ifdef USE_EXTERNALS
cur = externals;
while (cur) {
next = cur->next;
FREE(cur->name);
FREE(cur->command);
FREE(cur);
cur = next;
}
externals = NULL;
#endif /* USE_EXTERNALS */
return;
}
/*
* Process string buffer fields for DOWNLOADER or UPLOADER menus.
*/
PRIVATE void add_item_to_list ARGS2(
char *, buffer,
lynx_html_item_type **, list_ptr)
{
char *colon, *next_colon;
lynx_html_item_type *cur_item, *prev_item;
/*
* Make a linked list
*/
if (*list_ptr == NULL) {
/*
* First item.
*/
cur_item = (lynx_html_item_type *)calloc(sizeof(lynx_html_item_type),1);
if (cur_item == NULL)
outofmem(__FILE__, "read_cfg");
*list_ptr = cur_item;
atexit(free_item_list);
} else {
/*
* Find the last item.
*/
for (prev_item = *list_ptr;
prev_item->next != NULL;
prev_item = prev_item->next)
; /* null body */
cur_item = (lynx_html_item_type *)calloc(sizeof(lynx_html_item_type),1);
if (cur_item == NULL)
outofmem(__FILE__, "read_cfg");
else
prev_item->next = cur_item;
}
cur_item->next = NULL;
cur_item->name = NULL;
cur_item->command = NULL;
cur_item->always_enabled = FALSE;
/*
* Find first unescaped colon and process fields
*/
if ((colon = find_colon(buffer)) != NULL) {
/*
* Process name field
*/
cur_item->name = (char *)calloc((colon-buffer+1),sizeof(char));
if (cur_item->name == NULL)
outofmem(__FILE__, "read_cfg");
LYstrncpy(cur_item->name, buffer, (int)(colon-buffer));
remove_backslashes(cur_item->name);
/*
* Process TRUE/FALSE field
*/
if ((next_colon = find_colon(colon+1)) != NULL) {
cur_item->command = (char *)calloc(next_colon-colon, sizeof(char));
if (cur_item->command == NULL)
outofmem(__FILE__, "read_cfg");
LYstrncpy(cur_item->command, colon+1, (int)(next_colon-(colon+1)));
remove_backslashes(cur_item->command);
cur_item->always_enabled = is_true(next_colon+1);
}
}
}
/*
* Function for freeing the PRINTER menus list. - FM
*/
PRIVATE void free_printer_item_list NOARGS
{
lynx_printer_item_type *cur = printers;
lynx_printer_item_type *next;
while (cur) {
next = cur->next;
FREE(cur->name);
FREE(cur->command);
FREE(cur);
cur = next;
}
printers = NULL;
return;
}
/*
* Process string buffer fields for PRINTER menus.
*/
PRIVATE void add_printer_to_list ARGS2(
char *, buffer,
lynx_printer_item_type **, list_ptr)
{
char *colon, *next_colon;
lynx_printer_item_type *cur_item, *prev_item;
/*
* Make a linked list.
*/
if (*list_ptr == NULL) {
/*
* First item.
*/
cur_item = (lynx_printer_item_type *)calloc(sizeof(lynx_printer_item_type),1);
if (cur_item == NULL)
outofmem(__FILE__, "read_cfg");
*list_ptr = cur_item;
atexit(free_printer_item_list);
} else {
/*
* Find the last item.
*/
for (prev_item = *list_ptr;
prev_item->next != NULL;
prev_item = prev_item->next)
; /* null body */
cur_item = (lynx_printer_item_type *)calloc(sizeof(lynx_printer_item_type),1);
if (cur_item == NULL)
outofmem(__FILE__, "read_cfg");
else
prev_item->next = cur_item;
}
cur_item->next = NULL;
cur_item->name = NULL;
cur_item->command = NULL;
cur_item->always_enabled = FALSE;
/*
* Find first unescaped colon and process fields.
*/
if ((colon = find_colon(buffer)) != NULL) {
/*
* Process name field.
*/
cur_item->name = (char *)calloc((colon-buffer+1), sizeof(char));
if (cur_item->name == NULL)
outofmem(__FILE__, "read_cfg");
LYstrncpy(cur_item->name, buffer, (int)(colon-buffer));
remove_backslashes(cur_item->name);
/*
* Process TRUE/FALSE field.
*/
if ((next_colon = find_colon(colon+1)) != NULL) {
cur_item->command = (char *)calloc(next_colon-colon, sizeof(char));
if (cur_item->command == NULL)
outofmem(__FILE__, "read_cfg");
LYstrncpy(cur_item->command, colon+1, (int)(next_colon-(colon+1)));
remove_backslashes(cur_item->command);
cur_item->always_enabled = is_true(next_colon+1);
}
/*
* Process pagelen field.
*/
if ((next_colon = find_colon(next_colon+1)) != NULL) {
cur_item->pagelen = atoi(next_colon+1);
} else {
/* default to 66 lines */
cur_item->pagelen = 66;
}
}
}
#if defined(USE_COLOR_STYLE) || defined(USE_COLOR_TABLE)
#ifdef USE_SLANG
#define COLOR_WHITE 7
#define COLOR_BLACK 0
#endif
int default_fg = COLOR_WHITE;
int default_bg = COLOR_BLACK;
static CONST char *Color_Strings[16] =
{
"black",
"red",
"green",
"brown",
"blue",
"magenta",
"cyan",
"lightgray",
"gray",
"brightred",
"brightgreen",
"yellow",
"brightblue",
"brightmagenta",
"brightcyan",
"white"
};
#ifdef DOSPATH
/*
* PDCurses (and possibly some other implementations) use a non-ANSI set of
* codes for colors.
*/
PRIVATE int ColorCode ARGS1(
int, color)
{
static int map[] = {
0, 4, 2, 6, 1, 5, 3, 7,
8, 12, 10, 14, 9, 13, 11, 15 };
return map[color];
}
#else
#define ColorCode(color) (color)
#endif
/*
* Validator for COLOR fields.
*/
PUBLIC int check_color ARGS2(
char *, color,
int, the_default)
{
int i;
if (!strcasecomp(color, "default"))
return the_default;
if (!strcasecomp(color, "nocolor"))
return NO_COLOR;
for (i = 0; i < 16; i++) {
if (!strcasecomp(color, Color_Strings[i]))
return ColorCode(i);
}
return ERR_COLOR;
}
#endif /* USE_COLOR_STYLE || USE_COLOR_TABLE */
#if defined(USE_COLOR_TABLE)
/*
* Exit routine for failed COLOR parsing.
*/
PRIVATE void exit_with_color_syntax ARGS1(
char *, error_line)
{
unsigned int i;
fprintf (stderr, "\
Syntax Error parsing COLOR in configuration file:\n\
The line must be of the form:\n\
COLOR:INTEGER:FOREGROUND:BACKGROUND\n\
\n\
Here FOREGROUND and BACKGROUND must be one of:\n\
The special strings 'nocolor' or 'default', or\n"
);
for (i = 0; i < 16; i += 4) {
fprintf(stderr, "%16s %16s %16s %16s\n",
Color_Strings[i], Color_Strings[i + 1],
Color_Strings[i + 2], Color_Strings[i + 3]);
}
fprintf (stderr, "Offending line:\n%s\n",error_line);
#ifndef NOSIGHUP
(void) signal(SIGHUP, SIG_DFL);
#endif /* NOSIGHUP */
(void) signal(SIGTERM, SIG_DFL);
#ifndef VMS
(void) signal(SIGINT, SIG_DFL);
#endif /* !VMS */
#ifdef SIGTSTP
if (no_suspend)
(void) signal(SIGTSTP,SIG_DFL);
#endif /* SIGTSTP */
exit(-1);
}
/*
* Process string buffer fields for COLOR setting.
*/
PRIVATE void parse_color ARGS1(
char *, buffer)
{
int color;
char *fg, *bg;
char temp[501];
if (strlen(buffer) < sizeof(temp))
strcpy(temp, buffer);
else
strcpy(temp, "Color config line too long");
/*
* We are expecting a line of the form:
* INTEGER:FOREGROUND:BACKGROUND
*/
color = atoi(buffer);
if (NULL == (fg = find_colon(buffer)))
exit_with_color_syntax(temp);
*fg++ = '\0';
if (NULL == (bg = find_colon(fg)))
exit_with_color_syntax(temp);
*bg++ = '\0';
#if defined(USE_SLANG)
if ((check_color(fg, default_fg) < 0) ||
(check_color(bg, default_bg) < 0))
exit_with_color_syntax(temp);
SLtt_set_color(color, NULL, fg, bg);
#else
if (lynx_chg_color(color,
check_color(fg, default_fg),
check_color(bg, default_bg)) < 0)
exit_with_color_syntax(temp);
#endif
}
#endif /* USE_COLOR_TABLE */
#define MAX_LINE_BUFFER_LEN 501
typedef int (*ParseFunc) PARAMS((char *));
typedef union {
lynx_html_item_type ** add_value;
BOOLEAN * set_value;
int * int_value;
char ** str_value;
ParseFunc fun_value;
long def_value;
} ConfigUnion;
#undef PARSE_DEBUG
#ifdef PARSE_DEBUG
#define ParseData \
lynx_html_item_type** add_value; \
BOOLEAN *set_value; \
int *int_value; \
char **str_value; \
ParseFunc fun_value; \
long def_value
#define PARSE_ADD(n,t,v) {n,t, &v, 0, 0, 0, 0, 0}
#define PARSE_SET(n,t,v) {n,t, 0, &v, 0, 0, 0, 0}
#define PARSE_INT(n,t,v) {n,t, 0, 0, &v, 0, 0, 0}
#define PARSE_STR(n,t,v) {n,t, 0, 0, 0, &v, 0, 0}
#define PARSE_ENV(n,t,v) {n,t, 0, 0, 0, v, 0, 0}
#define PARSE_FUN(n,t,v) {n,t, 0, 0, 0, 0, v, 0}
#define PARSE_DEF(n,t,v) {n,t, 0, 0, 0, 0, 0, v}
#else
#define ParseData long value
#define PARSE_ADD(n,t,v) {n,t, (long)&(v)}
#define PARSE_SET(n,t,v) {n,t, (long)&(v)}
#define PARSE_INT(n,t,v) {n,t, (long)&(v)}
#define PARSE_STR(n,t,v) {n,t, (long)&(v)}
#define PARSE_ENV(n,t,v) {n,t, (long) (v)}
#define PARSE_FUN(n,t,v) {n,t, (long) (v)}
#define PARSE_DEF(n,t,v) {n,t, (long) (v)}
#endif
typedef struct
{
CONST char *name;
int type;
#define CONF_UNSPECIFIED 0
#define CONF_BOOL 1 /* BOOLEAN type */
#define CONF_FUN 2
#define CONF_INT 3
#define CONF_STR 4
#define CONF_ENV 5 /* from environment variable */
#define CONF_ENV2 6 /* from environment VARIABLE */
#define CONF_INCLUDE 7 /* include file-- handle special */
#define CONF_ADD_ITEM 8
#define CONF_ADD_TRUSTED 9
ParseData;
}
Config_Type;
static int assume_charset_fun ARGS1(
char *, value)
{
UCLYhndl_for_unspec = safeUCGetLYhndl_byMIME(value);
StrAllocCopy(UCAssume_MIMEcharset,
LYCharSet_UC[UCLYhndl_for_unspec].MIMEname);
/* this may be a memory for bogus typo -
StrAllocCopy(UCAssume_MIMEcharset, value);
LYLowerCase(UCAssume_MIMEcharset); */
return 0;
}
static int assume_local_charset_fun ARGS1(
char *, value)
{
UCLYhndl_HTFile_for_unspec = safeUCGetLYhndl_byMIME(value);
return 0;
}
static int assume_unrec_charset_fun ARGS1(
char *, value)
{
UCLYhndl_for_unrec = safeUCGetLYhndl_byMIME(value);
return 0;
}
static int character_set_fun ARGS1(
char *, value)
{
size_t i;
size_t len;
len = strlen (value);
for (i = 0; LYchar_set_names[i]; i++) { /* search by name, compatibility */
if (!strncmp(value, LYchar_set_names[i], len)) {
current_char_set = i;
return 0;
}
}
current_char_set = safeUCGetLYhndl_byMIME(value); /* by MIME */
return 0;
}
#ifdef USE_COLOR_TABLE
static int color_fun ARGS1(
char *, value)
{
parse_color (value);
return 0;
}
#endif
static int default_bookmark_file_fun ARGS1(
char *, value)
{
StrAllocCopy(bookmark_page, value);
StrAllocCopy(BookmarkPage, bookmark_page);
StrAllocCopy(MBM_A_subbookmark[0], bookmark_page);
StrAllocCopy(MBM_A_subdescript[0], MULTIBOOKMARKS_DEFAULT);
return 0;
}
static int default_cache_size_fun ARGS1(
char *, value)
{
HTCacheSize = atoi(value);
if (HTCacheSize < 2) HTCacheSize = 2;
return 0;
}
static int default_editor_fun ARGS1(
char *, value)
{
if (!system_editor) StrAllocCopy(editor, value);
return 0;
}
static int numbers_as_arrows_fun ARGS1(
char *, value)
{
if (is_true(value))
keypad_mode = NUMBERS_AS_ARROWS;
else
keypad_mode = LINKS_ARE_NUMBERED;
return 0;
}
static int default_user_mode_fun ARGS1(
char *, value)
{
if (!strncasecomp(value, "NOVICE", 6))
user_mode = NOVICE_MODE;
else if (!strncasecomp(value, "INTER", 5))
user_mode = INTERMEDIATE_MODE;
else if (!strncasecomp(value, "ADVANCE", 7))
user_mode = ADVANCED_MODE;
return 0;
}
#ifdef DIRED_SUPPORT
static int dired_menu_fun ARGS1(
char *, value)
{
add_menu_item(value);
return 0;
}
#endif
static int jumpfile_fun ARGS1(
char *, value)
{
char buffer [MAX_LINE_BUFFER_LEN];
sprintf (buffer, "JUMPFILE:%s", value);
if (!LYJumpInit(buffer))
CTRACE(tfp, "Failed to register %s\n", buffer);
return 0;
}
static int keymap_fun ARGS1(
char *, key)
{
char *func;
if ((func = strchr(key, ':')) != NULL) {
*func++ = '\0';
/* Allow comments on the ends of key remapping lines. - DT */
if (!remap(key, strtok(func, " \t\n#")))
fprintf(stderr,
"key remapping of %s to %s failed\n",key,func);
else if (!strcmp("TOGGLE_HELP", strtok(func, " \t\n#")))
LYUseNoviceLineTwo = FALSE;
}
return 0;
}
static int localhost_alias_fun ARGS1(
char *, value)
{
LYAddLocalhostAlias(value);
return 0;
}
#ifdef LYNXCGI_LINKS
static int lynxcgi_environment_fun ARGS1(
char *, value)
{
add_lynxcgi_environment(value);
return 0;
}
#endif
static int lynx_sig_file_fun ARGS1(
char *, value)
{
if (LYPathOffHomeOK(value, 256)) {
StrAllocCopy(LynxSigFile, value);
LYAddPathToHome(value, 256, LynxSigFile);
StrAllocCopy(LynxSigFile, value);
CTRACE(tfp, "LYNX_SIG_FILE set to '%s'\n", LynxSigFile);
} else {
CTRACE(tfp, "LYNX_SIG_FILE '%s' is bad. Ignoring.\n", LYNX_SIG_FILE);
}
return 0;
}
static int news_chunk_size_fun ARGS1(
char *, value)
{
HTNewsChunkSize = atoi(value);
/*
* If the new HTNewsChunkSize exceeds the maximum,
* increase HTNewsMaxChunk to this size. - FM
*/
if (HTNewsChunkSize > HTNewsMaxChunk)
HTNewsMaxChunk = HTNewsChunkSize;
return 0;
}
static int news_max_chunk_fun ARGS1(
char *, value)
{
HTNewsMaxChunk = atoi(value);
/*
* If HTNewsChunkSize exceeds the new maximum,
* reduce HTNewsChunkSize to this maximum. - FM
*/
if (HTNewsChunkSize > HTNewsMaxChunk)
HTNewsChunkSize = HTNewsMaxChunk;
return 0;
}
static int news_posting_fun ARGS1(
char *, value)
{
LYNewsPosting = is_true(value);
no_newspost = (LYNewsPosting == FALSE);
return 0;
}
static int printer_fun ARGS1(
char *, value)
{
add_printer_to_list(value, &printers);
return 0;
}
static int suffix_fun ARGS1(
char *, value)
{
char *mime_type;
if ((strlen (value) < 3)
|| (NULL == (mime_type = strchr (value, ':'))))
return 0;
*mime_type++ = '\0';
LYRemoveBlanks(mime_type);
LYLowerCase(mime_type);
if (strstr(mime_type, "tex") != NULL ||
strstr(mime_type, "postscript") != NULL ||
strstr(mime_type, "sh") != NULL ||
strstr(mime_type, "troff") != NULL ||
strstr(mime_type, "rtf") != NULL)
HTSetSuffix(value, mime_type, "8bit", 1.0);
else
HTSetSuffix(value, mime_type, "binary", 1.0);
return 0;
}
static int system_editor_fun ARGS1(
char *, value)
{
StrAllocCopy(editor, value);
system_editor = TRUE;
return 0;
}
static int viewer_fun ARGS1(
char *, value)
{
char *mime_type;
char *viewer;
char *environment;
mime_type = value;
if ((strlen (value) < 3)
|| (NULL == (viewer = strchr (mime_type, ':'))))
return 0;
*viewer++ = '\0';
LYRemoveBlanks(mime_type);
LYLowerCase(mime_type);
environment = strrchr(viewer, ':');
if ((environment != NULL) &&
(strlen(viewer) > 1) && *(environment-1) != '\\') {
*environment++ = '\0';
remove_backslashes(viewer);
/*
* If environment equals xwindows then only assign the presentation if
* there is a $DISPLAY variable.
*/
if (!strcasecomp(environment,"XWINDOWS")) {
if (LYgetXDisplay() != NULL)
HTSetPresentation(mime_type, viewer, 1.0, 3.0, 0.0, 0);
} else if (!strcasecomp(environment,"NON_XWINDOWS")) {
if (LYgetXDisplay() == NULL)
HTSetPresentation(mime_type, viewer, 1.0, 3.0, 0.0, 0);
} else {
HTSetPresentation(mime_type, viewer, 1.0, 3.0, 0.0, 0);
}
} else {
remove_backslashes(viewer);
HTSetPresentation(mime_type, viewer, 1.0, 3.0, 0.0, 0);
}
return 0;
}
/* This table should be sorted alphabetically */
static Config_Type Config_Table [] =
{
PARSE_SET("accept_all_cookies", CONF_BOOL, LYAcceptAllCookies),
PARSE_INT("alertsecs", CONF_INT, AlertSecs),
PARSE_SET("always_resubmit_posts", CONF_BOOL, LYresubmit_posts),
#ifdef EXEC_LINKS
PARSE_DEF("always_trusted_exec", CONF_ADD_TRUSTED, ALWAYS_EXEC_PATH),
#endif
PARSE_FUN("assume_charset", CONF_FUN, assume_charset_fun),
PARSE_FUN("assume_local_charset", CONF_FUN, assume_local_charset_fun),
PARSE_FUN("assume_unrec_charset", CONF_FUN, assume_unrec_charset_fun),
PARSE_SET("block_multi_bookmarks", CONF_BOOL, LYMBMBlocked),
PARSE_SET("bold_h1", CONF_BOOL, bold_H1),
PARSE_SET("bold_headers", CONF_BOOL, bold_headers),
PARSE_SET("bold_name_anchors", CONF_BOOL, bold_name_anchors),
PARSE_SET("case_sensitive_always_on", CONF_BOOL, case_sensitive),
PARSE_FUN("character_set", CONF_FUN, character_set_fun),
PARSE_SET("checkmail", CONF_BOOL, check_mail),
PARSE_SET("collapse_br_tags", CONF_BOOL, LYCollapseBRs),
#ifdef USE_COLOR_TABLE
PARSE_FUN("color", CONF_FUN, color_fun),
#endif
PARSE_STR("cookie_accept_domains", CONF_STR, LYCookieAcceptDomains),
#ifdef EXP_PERSISTENT_COOKIES
PARSE_STR("cookie_file", CONF_STR, LYCookieFile),
#endif /* EXP_PERSISTENT_COOKIES */
PARSE_STR("cookie_reject_domains", CONF_STR, LYCookieRejectDomains),
PARSE_ENV("cso_proxy", CONF_ENV, 0 ),
#ifdef VMS
PARSE_STR("CSWING_PATH", CONF_STR, LYCSwingPath),
#endif
PARSE_FUN("default_bookmark_file", CONF_FUN, default_bookmark_file_fun),
PARSE_FUN("default_cache_size", CONF_FUN, default_cache_size_fun),
PARSE_FUN("default_editor", CONF_FUN, default_editor_fun),
PARSE_STR("default_index_file", CONF_STR, indexfile),
PARSE_FUN("default_keypad_mode_is_numbers_as_arrows", CONF_FUN, numbers_as_arrows_fun),
PARSE_FUN("default_user_mode", CONF_FUN, default_user_mode_fun),
#if defined(VMS) && defined(VAXC) && !defined(__DECC)
PARSE_INT("default_virtual_memory_size", CONF_INT, HTVirtualMemorySize),
#endif
#ifdef DIRED_SUPPORT
PARSE_FUN("dired_menu", CONF_FUN, dired_menu_fun),
#endif
PARSE_ADD("downloader", CONF_ADD_ITEM, downloaders),
PARSE_SET("emacs_keys_always_on", CONF_BOOL, emacs_keys),
PARSE_SET("enable_scrollback", CONF_BOOL, enable_scrollback),
#ifdef USE_EXTERNALS
PARSE_ADD("external", CONF_ADD_ITEM, externals),
#endif
PARSE_ENV("finger_proxy", CONF_ENV, 0 ),
PARSE_SET("force_8bit_toupper", CONF_BOOL, UCForce8bitTOUPPER),
PARSE_SET("force_ssl_cookies_secure", CONF_BOOL, LYForceSSLCookiesSecure),
#ifndef EXP_FORMS_OPTIONS
PARSE_SET("forms_options", CONF_BOOL, LYUseFormsOptions),
#endif
PARSE_ENV("ftp_proxy", CONF_ENV, 0 ),
PARSE_STR("global_extension_map", CONF_STR, global_extension_map),
PARSE_STR("global_mailcap", CONF_STR, global_type_map),
PARSE_ENV("gopher_proxy", CONF_ENV, 0 ),
PARSE_SET("gotobuffer", CONF_BOOL, goto_buffer),
PARSE_STR("helpfile", CONF_STR, helpfile),
PARSE_SET("historical_comments", CONF_BOOL, historical_comments),
PARSE_ENV("http_proxy", CONF_ENV, 0 ),
PARSE_ENV("https_proxy", CONF_ENV, 0 ),
PARSE_FUN("include", CONF_INCLUDE, 0),
PARSE_INT("infosecs", CONF_INT, InfoSecs),
PARSE_STR("jump_prompt", CONF_STR, jumpprompt),
PARSE_SET("jumpbuffer", CONF_BOOL, jump_buffer),
PARSE_FUN("jumpfile", CONF_FUN, jumpfile_fun),
PARSE_FUN("keymap", CONF_FUN, keymap_fun),
PARSE_SET("list_news_numbers", CONF_BOOL, LYListNewsNumbers),
PARSE_SET("list_news_dates", CONF_BOOL, LYListNewsDates),
#ifndef VMS
PARSE_STR("list_format", CONF_STR, list_format),
#endif
PARSE_FUN("localhost_alias", CONF_FUN, localhost_alias_fun),
PARSE_STR("local_domain", CONF_STR, LYLocalDomain),
#if defined(EXEC_LINKS) || defined(EXEC_SCRIPTS)
PARSE_SET("local_execution_links_always_on", CONF_BOOL, local_exec),
PARSE_SET("local_execution_links_on_but_not_remote", CONF_BOOL, local_exec_on_local_files),
#endif
#ifdef LYNXCGI_LINKS
PARSE_FUN("lynxcgi_environment", CONF_FUN, lynxcgi_environment_fun),
#ifndef VMS
PARSE_STR("lynxcgi_document_root", CONF_STR, LYCgiDocumentRoot),
#endif
#endif
PARSE_STR("lynx_host_name", CONF_STR, LYHostName),
PARSE_FUN("lynx_sig_file", CONF_FUN, lynx_sig_file_fun),
PARSE_SET("mail_system_error_logging", CONF_BOOL, error_logging),
#ifdef VMS
PARSE_STR("mail_adrs", CONF_STR, mail_adrs),
#endif
PARSE_SET("make_links_for_all_images", CONF_BOOL, clickable_images),
PARSE_SET("make_pseudo_alts_for_inlines", CONF_BOOL, pseudo_inline_alts),
PARSE_INT("messagesecs", CONF_INT, MessageSecs),
PARSE_SET("minimal_comments", CONF_BOOL, minimal_comments),
PARSE_INT("multi_bookmark_support", CONF_BOOL, LYMultiBookmarks),
PARSE_SET("ncr_in_bookmarks", CONF_BOOL, UCSaveBookmarksInUnicode),
PARSE_FUN("news_chunk_size", CONF_FUN, news_chunk_size_fun),
PARSE_FUN("news_max_chunk", CONF_FUN, news_max_chunk_fun),
PARSE_FUN("news_posting", CONF_FUN, news_posting_fun),
PARSE_ENV("news_proxy", CONF_ENV, 0),
PARSE_ENV("newspost_proxy", CONF_ENV, 0),
PARSE_ENV("newsreply_proxy", CONF_ENV, 0),
PARSE_ENV("nntp_proxy", CONF_ENV, 0),
PARSE_ENV("nntpserver", CONF_ENV2, 0), /* actually NNTPSERVER */
PARSE_SET("no_dot_files", CONF_BOOL, no_dotfiles),
PARSE_SET("no_file_referer", CONF_BOOL, no_filereferer),
#ifndef VMS
PARSE_SET("no_forced_core_dump", CONF_BOOL, LYNoCore),
#endif
PARSE_SET("no_from_header", CONF_BOOL, LYNoFromHeader),
PARSE_SET("no_ismap_if_usemap", CONF_BOOL, LYNoISMAPifUSEMAP),
PARSE_ENV("no_proxy", CONF_ENV, 0 ),
PARSE_SET("no_referer_header", CONF_BOOL, LYNoRefererHeader),
#ifdef DISP_PARTIAL
PARSE_SET("partial", CONF_BOOL, display_partial),
#endif
PARSE_STR("personal_mailcap", CONF_STR, personal_type_map),
PARSE_STR("personal_extension_map", CONF_STR, personal_extension_map),
PARSE_STR("preferred_charset", CONF_STR, pref_charset),
PARSE_STR("preferred_language", CONF_STR, language),
PARSE_SET("prepend_base_to_source", CONF_BOOL, LYPrependBaseToSource),
PARSE_SET("prepend_charset_to_source", CONF_BOOL, LYPrependCharsetToSource),
PARSE_FUN("printer", CONF_FUN, printer_fun),
PARSE_SET("quit_default_yes", CONF_BOOL, LYQuitDefaultYes),
PARSE_STR("save_space", CONF_STR, lynx_save_space),
PARSE_SET("scan_for_buried_news_refs", CONF_BOOL, scan_for_buried_news_references),
PARSE_SET("seek_frag_area_in_cur", CONF_BOOL, LYSeekFragAREAinCur),
PARSE_SET("seek_frag_map_in_cur", CONF_BOOL, LYSeekFragMAPinCur),
PARSE_SET("set_cookies", CONF_BOOL, LYSetCookies),
PARSE_SET("show_cursor", CONF_BOOL, LYShowCursor),
PARSE_ENV("snews_proxy", CONF_ENV, 0 ),
PARSE_ENV("snewspost_proxy", CONF_ENV, 0 ),
PARSE_ENV("snewsreply_proxy", CONF_ENV, 0 ),
PARSE_SET("soft_dquotes", CONF_BOOL, soft_dquotes),
PARSE_STR("startfile", CONF_STR, startfile),
PARSE_SET("strip_dotdot_urls", CONF_BOOL, LYStripDotDotURLs),
PARSE_SET("substitute_underscores", CONF_BOOL, use_underscore),
PARSE_FUN("suffix", CONF_FUN, suffix_fun),
PARSE_FUN("system_editor", CONF_FUN, system_editor_fun),
PARSE_STR("system_mail", CONF_STR, system_mail),
PARSE_STR("system_mail_flags", CONF_STR, system_mail_flags),
#ifdef EXEC_LINKS
PARSE_DEF("trusted_exec", CONF_ADD_TRUSTED, EXEC_PATH),
#endif
#ifdef LYNXCGI_LINKS
PARSE_DEF("trusted_lynxcgi", CONF_ADD_TRUSTED, CGI_PATH),
#endif
PARSE_STR("url_domain_prefixes", CONF_STR, URLDomainPrefixes),
PARSE_STR("url_domain_suffixes", CONF_STR, URLDomainSuffixes),
#ifdef DIRED_SUPPORT
PARSE_ADD("uploader", CONF_ADD_ITEM, uploaders),
#endif
#ifdef VMS
PARSE_SET("use_fixed_records", CONF_BOOL, UseFixedRecords),
#endif
#if defined(NCURSES_MOUSE_VERSION) || defined(USE_SLANG_MOUSE)
PARSE_SET("use_mouse", CONF_BOOL, LYUseMouse),
#endif
PARSE_SET("use_select_popups", CONF_BOOL, LYSelectPopups),
PARSE_SET("verbose_images", CONF_BOOL, verbose_img),
PARSE_SET("vi_keys_always_on", CONF_BOOL, vi_keys),
PARSE_FUN("viewer", CONF_FUN, viewer_fun),
PARSE_ENV("wais_proxy", CONF_ENV, 0 ),
PARSE_STR("xloadimage_command", CONF_STR, XLoadImageCommand),
{0}
};
/*
* Free memory allocated in 'read_cfg()'
*/
PUBLIC void free_lynx_cfg NOARGS
{
Config_Type *tbl;
for (tbl = Config_Table; tbl->name != 0; tbl++) {
#ifdef PARSE_DEBUG
Config_Type *q = tbl;
#else
ConfigUnion *q = (ConfigUnion *)(&(tbl->value));
#endif
switch (tbl->type) {
case CONF_ENV:
if (q->str_value != 0) {
FREE(*(q->str_value));
free((char *)q->str_value);
}
break;
default:
break;
}
}
}
/*
* Process the configuration file (lynx.cfg).
*/
PUBLIC void read_cfg ARGS4(
char *, cfg_filename,
char *, parent_filename,
int, nesting_level,
FILE *, fp0)
{
FILE *fp;
char buffer[MAX_LINE_BUFFER_LEN];
CTRACE(tfp, "Loading cfg file '%s'.\n", cfg_filename);
/*
* Don't get hung up by an include file loop. Arbitrary max depth
* of 10. - BL
*/
if (nesting_level > 10) {
fprintf(stderr,
"More than %d nested lynx.cfg includes -- perhaps there is a loop?!?\n",
nesting_level - 1);
fprintf(stderr,"Last attempted include was '%s',\n", cfg_filename);
fprintf(stderr,"included from '%s'.\n", parent_filename);
exit(-1);
}
/*
* Locate and open the file.
*/
if (!cfg_filename || strlen(cfg_filename) == 0) {
CTRACE(tfp,"No filename following -cfg switch!\n");
return;
}
if ((fp = fopen(cfg_filename,"r")) == 0) {
CTRACE(tfp,"lynx.cfg file not found as %s\n",cfg_filename);
return;
}
have_read_cfg = TRUE;
/*
* Process each line in the file.
*/
while (fgets(buffer, sizeof(buffer), fp) != 0) {
char *name, *value;
char *cp;
Config_Type *tbl;
char ch;
#ifdef PARSE_DEBUG
Config_Type *q;
#else
ConfigUnion *q;
#endif
/* Most lines in the config file are comment lines. Weed them out
* now. Also, leading whitespace is ok, so trim it.
*/
name = LYSkipBlanks(buffer);
if (*name == '#')
continue;
LYTrimTrailing(name);
if (*name == 0) continue;
/* Significant lines are of the form KEYWORD:WHATEVER */
if ((value = strchr (name, ':')) == 0) {
/* fprintf (stderr, "Bad line-- no :\n"); */
continue;
}
/* skip past colon, but replace ':' with 0 to make name meaningful */
*value++ = 0;
/*
* Trim off any trailing comments.
*
* (Apparently, the original code considers a trailing comment
* valid only if preceded by a space character but is not followed
* by a colon. -- JED)
*/
if ((cp = strrchr (value, ':')) == 0)
cp = value;
if ((cp = strchr (cp, '#')) != 0) {
cp--;
if (isspace ((unsigned char) *cp))
*cp = 0;
}
tbl = Config_Table;
ch = TOUPPER(*name);
while (tbl->name != 0) {
char ch1 = tbl->name[0];
if ((ch == TOUPPER(ch1))
&& (0 == strcasecomp (name, tbl->name)))
break;
tbl++;
}
if (tbl->name == 0) {
/* Apparently, lynx ignores unknown keywords */
/* fprintf (stderr, "%s not found in config file */
continue;
}
#ifdef PARSE_DEBUG
q = tbl;
#else
q = (ConfigUnion *)(&(tbl->value));
#endif
switch ((fp0 != 0 && tbl->type != CONF_INCLUDE)
? CONF_UNSPECIFIED
: tbl->type) {
case CONF_BOOL:
if (q->set_value != 0)
*(q->set_value) = is_true (value);
break;
case CONF_FUN:
if (q->fun_value != 0)
(*(q->fun_value)) (value);
break;
case CONF_INT:
if (q->int_value != 0) {
int ival;
/* Apparently, if an integer value is not present, then the
* value is not changed. So, use the sscanf function to make
* this determination.
*/
if (1 == sscanf (value, "%d", &ival))
*(q->int_value) = ival;
}
break;
case CONF_STR:
if (q->str_value != 0)
StrAllocCopy(*(q->str_value), value);
break;
case CONF_ENV:
case CONF_ENV2:
if (tbl->type == CONF_ENV)
LYLowerCase(name);
else
LYUpperCase(name);
if (getenv (name) == 0) {
#ifdef VMS
Define_VMSLogical(tbl->name, value);
#else
char tmpbuf[MAX_LINE_BUFFER_LEN];
sprintf (tmpbuf, "%s=%s", name, value);
if ((q->str_value = (char **)calloc(1, sizeof(char **))) != 0) {
StrAllocCopy(*(q->str_value), tmpbuf);
putenv (*(q->str_value));
}
#endif
}
break;
case CONF_INCLUDE:
/* include another file */
if (fp0 != 0) {
fprintf(fp0, "%s:%s\n\n", name, value);
fprintf(fp0, " #<begin %s>\n", value);
}
read_cfg (value, cfg_filename, nesting_level + 1, fp0);
if (fp0 != 0)
fprintf(fp0, " #<end of %s>\n\n", value);
break;
case CONF_ADD_ITEM:
if (q->add_value != 0)
add_item_to_list (value, q->add_value);
break;
#if defined(EXEC_LINKS) || defined(LYNXCGI_LINKS)
case CONF_ADD_TRUSTED:
add_trusted (value, q->def_value);
break;
#endif
default:
if (fp0 != 0)
fprintf(fp0, "%s:%s\n", name, value);
break;
}
}
fclose (fp);
/*
* If any DOWNLOADER: commands have always_enabled set (:TRUE),
* make override_no_download TRUE, so that other restriction
* settings will not block presentation of a download menu
* with those always_enabled options still available. - FM
*/
if (downloaders != 0) {
lynx_html_item_type *cur_download;
cur_download = downloaders;
while (cur_download != 0) {
if (cur_download->always_enabled) {
override_no_download = TRUE;
break;
}
cur_download = cur_download->next;
}
}
/*
* If any COOKIE_{ACCEPT,REJECT}_DOMAINS have been defined,
* process them. These are comma delimited lists of
* domains, with leading '.'. - BJP
*/
if (LYCookieAcceptDomains != NULL) {
cookie_add_acceptlist(LYCookieAcceptDomains);
}
if (LYCookieRejectDomains != NULL) {
cookie_add_rejectlist(LYCookieRejectDomains);
}
}
/*
* lynx.cfg infopage, returns local url.
*/
PUBLIC char *lynx_cfg_infopage NOARGS
{
static char *local_url;
char tempfile[LY_MAXPATH];
char *temp = 0;
FILE *fp0;
if (local_url == 0) {
if ((fp0 = LYOpenTemp(tempfile, HTML_SUFFIX, "w")) == NULL) {
HTAlert(CANNOT_OPEN_TEMP);
return(0);
}
LYLocalFileToURL(&local_url, tempfile);
LYforce_no_cache = TRUE; /* don't cache this doc */
#ifdef HAVE_CONFIG_H
StrAllocCopy(temp, LYNX_CFG_FILE);
#else
StrAllocCopy(temp, helpfilepath);
StrAllocCat(temp, LYNXCFG_HELP); /* DJGPP/Win32/VMS should care of */
#endif /* HAVE_CONFIG_H */
BeginInternalPage (fp0, LYNXCFG_TITLE, NULL);
fprintf(fp0, "<pre>\n");
fprintf(fp0, "<em>This is read from your lynx.cfg file,\n");
fprintf(fp0, "please \"read\" distribution's <a href=\"%s\">lynx.cfg",
temp);
fprintf(fp0, "</a> for more comments.</em>\n\n");
fprintf(fp0, " #<em>Your primary configuration %s</em>\n",
lynx_cfg_file);
/*
* Process the configuration file.
*/
read_cfg(lynx_cfg_file, "main program", 1, fp0);
FREE(temp);
fprintf(fp0, "</pre>\n");
EndInternalPage(fp0);
LYCloseTempFP(fp0);
}
return(local_url);
}