forked from pool/dictd
3f55ce16c6
Clean up SPEC file, make the package building again. OBS-URL: https://build.opensuse.org/request/show/642025 OBS-URL: https://build.opensuse.org/package/show/Education/dictd?expand=0&rev=4
10217 lines
271 KiB
Diff
10217 lines
271 KiB
Diff
--- a/dictd.c
|
|
+++ b/dictd.c
|
|
@@ -31,14 +31,14 @@
|
|
#include "plugin.h"
|
|
#endif
|
|
|
|
-#include <grp.h> /* initgroups */
|
|
-#include <pwd.h> /* getpwuid */
|
|
-#include <locale.h> /* setlocale */
|
|
+#include <grp.h> /* initgroups */
|
|
+#include <pwd.h> /* getpwuid */
|
|
+#include <locale.h> /* setlocale */
|
|
#include <ctype.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
|
|
-#define MAXPROCTITLE 2048 /* Maximum amount of proc title we'll use. */
|
|
+#define MAXPROCTITLE 2048 /* Maximum amount of proc title we'll use. */
|
|
#undef MIN
|
|
#define MIN(a,b) ((a)<(b)?(a):(b))
|
|
#ifndef UID_NOBODY
|
|
@@ -55,278 +55,283 @@
|
|
|
|
|
|
|
|
-extern int yy_flex_debug;
|
|
+extern int yy_flex_debug;
|
|
|
|
-static int _dict_daemon;
|
|
-static int _dict_reaps;
|
|
+static int _dict_daemon;
|
|
+static int _dict_reaps;
|
|
|
|
-static char *_dict_argvstart;
|
|
-static int _dict_argvlen;
|
|
+static char *_dict_argvstart;
|
|
+static int _dict_argvlen;
|
|
|
|
- int _dict_forks;
|
|
+int _dict_forks;
|
|
|
|
|
|
|
|
-int default_strategy_set; /* 1 if set by command line option */
|
|
+int default_strategy_set; /* 1 if set by command line option */
|
|
|
|
|
|
-int logOptions = 0;
|
|
+int logOptions = 0;
|
|
|
|
-const char *logFile = NULL;
|
|
-int logFile_set; /* 1 if set by command line option */
|
|
+const char *logFile = NULL;
|
|
+int logFile_set; /* 1 if set by command line option */
|
|
|
|
-const char *pidFile = "/var/run/dictd.pid";
|
|
-int pidFile_set; /* 1 if set by command line option */
|
|
+const char *pidFile = "/var/run/dictd.pid";
|
|
+int pidFile_set; /* 1 if set by command line option */
|
|
|
|
-const char *daemon_service = DICT_DEFAULT_SERVICE;
|
|
-int daemon_service_set; /* 1 if set by command line option */
|
|
+const char *daemon_service = DICT_DEFAULT_SERVICE;
|
|
+int daemon_service_set; /* 1 if set by command line option */
|
|
|
|
-int _dict_daemon_limit_childs = DICT_DAEMON_LIMIT_CHILDS;
|
|
-int _dict_daemon_limit_childs_set; /* 1 if set by command line option */
|
|
+int _dict_daemon_limit_childs = DICT_DAEMON_LIMIT_CHILDS;
|
|
+int _dict_daemon_limit_childs_set; /* 1 if set by command line option */
|
|
|
|
-int _dict_daemon_limit_matches = DICT_DAEMON_LIMIT_MATCHES;
|
|
-int _dict_daemon_limit_defs = DICT_DAEMON_LIMIT_DEFS;
|
|
-int _dict_daemon_limit_time = DICT_DAEMON_LIMIT_TIME;
|
|
-int _dict_daemon_limit_queries = DICT_DAEMON_LIMIT_QUERIES;
|
|
+int _dict_daemon_limit_matches = DICT_DAEMON_LIMIT_MATCHES;
|
|
+int _dict_daemon_limit_defs = DICT_DAEMON_LIMIT_DEFS;
|
|
+int _dict_daemon_limit_time = DICT_DAEMON_LIMIT_TIME;
|
|
+int _dict_daemon_limit_queries = DICT_DAEMON_LIMIT_QUERIES;
|
|
|
|
-int _dict_markTime = 0;
|
|
-int _dict_markTime_set; /* 1 if set by command line option */
|
|
+int _dict_markTime = 0;
|
|
+int _dict_markTime_set; /* 1 if set by command line option */
|
|
|
|
-const char *locale = NULL;
|
|
-int locale_set; /* 1 if set by command line option */
|
|
+const char *locale = NULL;
|
|
+int locale_set; /* 1 if set by command line option */
|
|
|
|
-int client_delay = DICT_DEFAULT_DELAY;
|
|
-int client_delay_set; /* 1 if set by command line option */
|
|
+int client_delay = DICT_DEFAULT_DELAY;
|
|
+int client_delay_set; /* 1 if set by command line option */
|
|
|
|
-int depth = DICT_QUEUE_DEPTH;
|
|
-int depth_set; /* 1 if set by command line option */
|
|
+int depth = DICT_QUEUE_DEPTH;
|
|
+int depth_set; /* 1 if set by command line option */
|
|
|
|
-int useSyslog = 0;
|
|
-int syslog_facility_set; /* 1 if set by command line option */
|
|
+int useSyslog = 0;
|
|
+int syslog_facility_set; /* 1 if set by command line option */
|
|
|
|
-const char *preprocessor = NULL;
|
|
+const char *preprocessor = NULL;
|
|
|
|
-const char *bind_to = NULL;
|
|
-int bind_to_set; /* 1 if set by command line option */
|
|
+const char *bind_to = NULL;
|
|
+int bind_to_set; /* 1 if set by command line option */
|
|
|
|
/* information about dict server, i.e.
|
|
text returned by SHOW SERVER command
|
|
*/
|
|
-const char *site_info = NULL;
|
|
-int site_info_no_banner = 0;
|
|
-int site_info_no_uptime = 0;
|
|
-int site_info_no_dblist = 0;
|
|
+const char *site_info = NULL;
|
|
+int site_info_no_banner = 0;
|
|
+int site_info_no_uptime = 0;
|
|
+int site_info_no_dblist = 0;
|
|
|
|
|
|
|
|
|
|
-int inetd = 0;
|
|
+int inetd = 0;
|
|
|
|
-int need_reload_config = 0;
|
|
+int need_reload_config = 0;
|
|
|
|
-int need_unload_databases = 0;
|
|
-int databases_unloaded = 0;
|
|
+int need_unload_databases = 0;
|
|
+int databases_unloaded = 0;
|
|
|
|
-static const char *configFile = DICT_CONFIG_PATH DICTD_CONFIG_NAME;
|
|
+static const char *configFile = DICT_CONFIG_PATH DICTD_CONFIG_NAME;
|
|
|
|
-static void dict_close_databases (dictConfig *c);
|
|
-static void sanity (const char *confFile);
|
|
-static void dict_init_databases (dictConfig *c);
|
|
-static void dict_config_print (FILE *stream, dictConfig *c);
|
|
-static void postprocess_filenames (dictConfig *dc);
|
|
+static void dict_close_databases(dictConfig * c);
|
|
+static void sanity(const char *confFile);
|
|
+static void dict_init_databases(dictConfig * c);
|
|
+static void dict_config_print(FILE * stream, dictConfig * c);
|
|
+static void postprocess_filenames(dictConfig * dc);
|
|
|
|
-void dict_initsetproctitle( int argc, char **argv, char **envp )
|
|
+void dict_initsetproctitle(int argc, char **argv, char **envp)
|
|
{
|
|
- int i;
|
|
+ int i;
|
|
|
|
- _dict_argvstart = argv[0];
|
|
-
|
|
- for (i = 0; envp[i]; i++) continue;
|
|
+ _dict_argvstart = argv[0];
|
|
|
|
- if (i)
|
|
- _dict_argvlen = envp[i-1] + strlen(envp[i-1]) - _dict_argvstart;
|
|
- else
|
|
- _dict_argvlen = argv[argc-1] + strlen(argv[argc-1]) - _dict_argvstart;
|
|
+ for (i = 0; envp[i]; i++)
|
|
+ continue;
|
|
|
|
- argv[1] = NULL;
|
|
+ if (i)
|
|
+ _dict_argvlen =
|
|
+ envp[i - 1] + strlen(envp[i - 1]) - _dict_argvstart;
|
|
+ else
|
|
+ _dict_argvlen =
|
|
+ argv[argc - 1] + strlen(argv[argc - 1]) - _dict_argvstart;
|
|
+
|
|
+ argv[1] = NULL;
|
|
}
|
|
|
|
-void dict_setproctitle( const char *format, ... )
|
|
+void dict_setproctitle(const char *format, ...)
|
|
{
|
|
- va_list ap;
|
|
- int len;
|
|
- char buf[MAXPROCTITLE];
|
|
+ va_list ap;
|
|
+ int len;
|
|
+ char buf[MAXPROCTITLE];
|
|
|
|
- va_start( ap, format );
|
|
- vsnprintf( buf, MAXPROCTITLE, format, ap );
|
|
- va_end( ap );
|
|
+ va_start(ap, format);
|
|
+ vsnprintf(buf, MAXPROCTITLE, format, ap);
|
|
+ va_end(ap);
|
|
|
|
- if ((len = strlen(buf)) > MAXPROCTITLE-1)
|
|
- err_fatal( __func__, "buffer overflow (%d)\n", len );
|
|
+ if ((len = strlen(buf)) > MAXPROCTITLE - 1)
|
|
+ err_fatal(__func__, "buffer overflow (%d)\n", len);
|
|
|
|
- buf[ MIN(_dict_argvlen,MAXPROCTITLE) - 1 ] = '\0';
|
|
- strcpy( _dict_argvstart, buf );
|
|
- memset( _dict_argvstart+len, 0, _dict_argvlen-len );
|
|
+ buf[MIN(_dict_argvlen, MAXPROCTITLE) - 1] = '\0';
|
|
+ strcpy(_dict_argvstart, buf);
|
|
+ memset(_dict_argvstart + len, 0, _dict_argvlen - len);
|
|
}
|
|
|
|
-const char *dict_format_time( double t )
|
|
+const char *dict_format_time(double t)
|
|
{
|
|
- static int current = 0;
|
|
- static char buf[10][128]; /* Rotate 10 buffers */
|
|
- static char *this;
|
|
- long int s, m, h, d;
|
|
+ static int current = 0;
|
|
+ static char buf[10][128]; /* Rotate 10 buffers */
|
|
+ static char *this;
|
|
+ long int s, m, h, d;
|
|
|
|
- this = buf[current];
|
|
- if (++current >= 10) current = 0;
|
|
+ this = buf[current];
|
|
+ if (++current >= 10)
|
|
+ current = 0;
|
|
|
|
- if (t < 600) {
|
|
- snprintf( this, sizeof (buf [0]), "%0.3f", t );
|
|
- } else {
|
|
- s = (long int)t;
|
|
- d = s / (3600*24);
|
|
- s -= d * 3600 * 24;
|
|
- h = s / 3600;
|
|
- s -= h * 3600;
|
|
- m = s / 60;
|
|
- s -= m * 60;
|
|
+ if (t < 600) {
|
|
+ snprintf(this, sizeof(buf[0]), "%0.3f", t);
|
|
+ } else {
|
|
+ s = (long int) t;
|
|
+ d = s / (3600 * 24);
|
|
+ s -= d * 3600 * 24;
|
|
+ h = s / 3600;
|
|
+ s -= h * 3600;
|
|
+ m = s / 60;
|
|
+ s -= m * 60;
|
|
|
|
- if (d)
|
|
- snprintf( this, sizeof (buf [0]), "%ld+%02ld:%02ld:%02ld", d, h, m, s );
|
|
- else if (h)
|
|
- snprintf( this, sizeof (buf [0]), "%02ld:%02ld:%02ld", h, m, s );
|
|
- else
|
|
- snprintf( this, sizeof (buf [0]), "%02ld:%02ld", m, s );
|
|
- }
|
|
+ if (d)
|
|
+ snprintf(this, sizeof(buf[0]), "%ld+%02ld:%02ld:%02ld", d, h,
|
|
+ m, s);
|
|
+ else if (h)
|
|
+ snprintf(this, sizeof(buf[0]), "%02ld:%02ld:%02ld", h, m, s);
|
|
+ else
|
|
+ snprintf(this, sizeof(buf[0]), "%02ld:%02ld", m, s);
|
|
+ }
|
|
|
|
- return this;
|
|
+ return this;
|
|
}
|
|
|
|
-static int waitpid__exit_status (int status)
|
|
+static int waitpid__exit_status(int status)
|
|
{
|
|
- if (WIFEXITED(status)){
|
|
- return WEXITSTATUS(status);
|
|
- }else if (WIFSIGNALED(status)){
|
|
- return 128 + WTERMSIG(status);
|
|
- }else{
|
|
- return -1;
|
|
- }
|
|
+ if (WIFEXITED(status)) {
|
|
+ return WEXITSTATUS(status);
|
|
+ } else if (WIFSIGNALED(status)) {
|
|
+ return 128 + WTERMSIG(status);
|
|
+ } else {
|
|
+ return -1;
|
|
+ }
|
|
}
|
|
|
|
-static void reaper( int dummy )
|
|
+static void reaper(int dummy)
|
|
{
|
|
#if 0
|
|
- union wait status;
|
|
+ union wait status;
|
|
#else
|
|
- int status;
|
|
+ int status;
|
|
#endif
|
|
- pid_t pid;
|
|
+ pid_t pid;
|
|
|
|
- while ((pid = wait3(&status, WNOHANG, NULL)) > 0) {
|
|
- ++_dict_reaps;
|
|
+ while ((pid = wait3(&status, WNOHANG, NULL)) > 0) {
|
|
+ ++_dict_reaps;
|
|
|
|
- if (flg_test(LOG_SERVER))
|
|
- log_info( ":I: Reaped %d%s, exit status %i\n",
|
|
- pid,
|
|
- _dict_daemon ? " IN CHILD": "",
|
|
- waitpid__exit_status (status));
|
|
- }
|
|
+ if (flg_test(LOG_SERVER))
|
|
+ log_info(":I: Reaped %d%s, exit status %i\n",
|
|
+ pid,
|
|
+ _dict_daemon ? " IN CHILD" : "",
|
|
+ waitpid__exit_status(status));
|
|
+ }
|
|
}
|
|
|
|
-static int start_daemon( void )
|
|
+static int start_daemon(void)
|
|
{
|
|
- pid_t pid;
|
|
-
|
|
- ++_dict_forks;
|
|
- switch ((pid = fork())) {
|
|
- case 0:
|
|
- ++_dict_daemon;
|
|
- break;
|
|
- case -1:
|
|
- log_info( ":E: Unable to fork daemon\n" );
|
|
- alarm(10); /* Can't use sleep() here */
|
|
- pause();
|
|
- break;
|
|
- default:
|
|
- if (flg_test(LOG_SERVER)) log_info( ":I: Forked %d\n", pid );
|
|
- break;
|
|
- }
|
|
- return pid;
|
|
+ pid_t pid;
|
|
+
|
|
+ ++_dict_forks;
|
|
+ switch ((pid = fork())) {
|
|
+ case 0:
|
|
+ ++_dict_daemon;
|
|
+ break;
|
|
+ case -1:
|
|
+ log_info(":E: Unable to fork daemon\n");
|
|
+ alarm(10); /* Can't use sleep() here */
|
|
+ pause();
|
|
+ break;
|
|
+ default:
|
|
+ if (flg_test(LOG_SERVER))
|
|
+ log_info(":I: Forked %d\n", pid);
|
|
+ break;
|
|
+ }
|
|
+ return pid;
|
|
}
|
|
|
|
-static const char * signal2name (int sig)
|
|
+static const char *signal2name(int sig)
|
|
{
|
|
- static char name [50];
|
|
+ static char name[50];
|
|
|
|
- switch (sig) {
|
|
- case SIGHUP:
|
|
- return "SIGHUP";
|
|
- case SIGINT:
|
|
- return "SIGINT";
|
|
- case SIGQUIT:
|
|
- return "SIGQUIT";
|
|
- case SIGILL:
|
|
- return "SIGILL";
|
|
- case SIGTRAP:
|
|
- return "SIGTRAP";
|
|
- case SIGTERM:
|
|
- return "SIGTERM";
|
|
- case SIGPIPE:
|
|
- return "SIGPIPE";
|
|
- case SIGALRM:
|
|
- return "SIGALRM";
|
|
- default:
|
|
- snprintf (name, sizeof (name), "Signal %d", sig);
|
|
- return name;
|
|
- }
|
|
+ switch (sig) {
|
|
+ case SIGHUP:
|
|
+ return "SIGHUP";
|
|
+ case SIGINT:
|
|
+ return "SIGINT";
|
|
+ case SIGQUIT:
|
|
+ return "SIGQUIT";
|
|
+ case SIGILL:
|
|
+ return "SIGILL";
|
|
+ case SIGTRAP:
|
|
+ return "SIGTRAP";
|
|
+ case SIGTERM:
|
|
+ return "SIGTERM";
|
|
+ case SIGPIPE:
|
|
+ return "SIGPIPE";
|
|
+ case SIGALRM:
|
|
+ return "SIGALRM";
|
|
+ default:
|
|
+ snprintf(name, sizeof(name), "Signal %d", sig);
|
|
+ return name;
|
|
+ }
|
|
}
|
|
|
|
-static void log_sig_info (int sig)
|
|
+static void log_sig_info(int sig)
|
|
{
|
|
- log_info (
|
|
- ":I: %s: c/f = %d/%d; %sr %su %ss\n",
|
|
- signal2name (sig),
|
|
- _dict_comparisons,
|
|
- _dict_forks,
|
|
- dict_format_time (tim_get_real ("dictd")),
|
|
- dict_format_time (tim_get_user ("dictd")),
|
|
- dict_format_time (tim_get_system ("dictd")));
|
|
+ log_info(":I: %s: c/f = %d/%d; %sr %su %ss\n",
|
|
+ signal2name(sig),
|
|
+ _dict_comparisons,
|
|
+ _dict_forks,
|
|
+ dict_format_time(tim_get_real("dictd")),
|
|
+ dict_format_time(tim_get_user("dictd")),
|
|
+ dict_format_time(tim_get_system("dictd")));
|
|
}
|
|
|
|
-static void unload_databases (void)
|
|
+static void unload_databases(void)
|
|
{
|
|
- dict_close_databases (DictConfig);
|
|
- DictConfig = NULL;
|
|
+ dict_close_databases(DictConfig);
|
|
+ DictConfig = NULL;
|
|
}
|
|
|
|
-static void reload_config (void)
|
|
+static void reload_config(void)
|
|
{
|
|
- dict_close_databases (DictConfig);
|
|
+ dict_close_databases(DictConfig);
|
|
|
|
- if (!access(configFile,R_OK)){
|
|
- prs_file_pp (preprocessor, configFile);
|
|
- postprocess_filenames (DictConfig);
|
|
- }
|
|
+ if (!access(configFile, R_OK)) {
|
|
+ prs_file_pp(preprocessor, configFile);
|
|
+ postprocess_filenames(DictConfig);
|
|
+ }
|
|
|
|
- sanity (configFile);
|
|
+ sanity(configFile);
|
|
|
|
- if (dbg_test (DBG_VERBOSE))
|
|
- dict_config_print( NULL, DictConfig );
|
|
+ if (dbg_test(DBG_VERBOSE))
|
|
+ dict_config_print(NULL, DictConfig);
|
|
|
|
- dict_init_databases (DictConfig);
|
|
+ dict_init_databases(DictConfig);
|
|
}
|
|
|
|
-static void xsigaddset (sigset_t *set, int signo)
|
|
+static void xsigaddset(sigset_t * set, int signo)
|
|
{
|
|
- if (sigaddset (set, signo)){
|
|
- log_error ("", "sigaddset(2) failed: %s\n", strerror (errno));
|
|
- }
|
|
+ if (sigaddset(set, signo)) {
|
|
+ log_error("", "sigaddset(2) failed: %s\n", strerror(errno));
|
|
+ }
|
|
}
|
|
|
|
-static void xsigprocmask (int how, const sigset_t *set, sigset_t *oset)
|
|
+static void xsigprocmask(int how, const sigset_t * set, sigset_t * oset)
|
|
{
|
|
- if (sigprocmask (how, set, oset)){
|
|
- log_error ("", "sigaddset(2) failed: %s\n", strerror (errno));
|
|
- }
|
|
+ if (sigprocmask(how, set, oset)) {
|
|
+ log_error("", "sigaddset(2) failed: %s\n", strerror(errno));
|
|
+ }
|
|
}
|
|
|
|
/*
|
|
@@ -353,910 +358,930 @@ static void unblock_signals (void)
|
|
}
|
|
*/
|
|
|
|
-static void handler( int sig )
|
|
+static void handler(int sig)
|
|
{
|
|
- const char *name = NULL;
|
|
- time_t t;
|
|
+ const char *name = NULL;
|
|
+ time_t t;
|
|
|
|
- name = signal2name (sig);
|
|
+ name = signal2name(sig);
|
|
|
|
- if (_dict_daemon) {
|
|
- daemon_terminate( sig, name );
|
|
- } else {
|
|
- tim_stop( "dictd" );
|
|
- switch (sig){
|
|
- case SIGALRM:
|
|
- if (_dict_markTime > 0){
|
|
- time(&t);
|
|
- log_info( ":t: %24.24s; %d/%d %sr %su %ss\n",
|
|
- ctime(&t),
|
|
- _dict_forks - _dict_reaps,
|
|
- _dict_forks,
|
|
- dict_format_time( tim_get_real( "dictd" ) ),
|
|
- dict_format_time( tim_get_user( "dictd" ) ),
|
|
- dict_format_time( tim_get_system( "dictd" ) ) );
|
|
- alarm(_dict_markTime);
|
|
- return;
|
|
- }
|
|
+ if (_dict_daemon) {
|
|
+ daemon_terminate(sig, name);
|
|
+ } else {
|
|
+ tim_stop("dictd");
|
|
+ switch (sig) {
|
|
+ case SIGALRM:
|
|
+ if (_dict_markTime > 0) {
|
|
+ time(&t);
|
|
+ log_info(":t: %24.24s; %d/%d %sr %su %ss\n",
|
|
+ ctime(&t),
|
|
+ _dict_forks - _dict_reaps,
|
|
+ _dict_forks,
|
|
+ dict_format_time(tim_get_real("dictd")),
|
|
+ dict_format_time(tim_get_user("dictd")),
|
|
+ dict_format_time(tim_get_system("dictd")));
|
|
+ alarm(_dict_markTime);
|
|
+ return;
|
|
+ }
|
|
|
|
- break;
|
|
- }
|
|
+ break;
|
|
+ }
|
|
|
|
- log_sig_info (sig);
|
|
- }
|
|
- if (!dbg_test(DBG_NOFORK) || sig != SIGALRM)
|
|
- exit(sig+128);
|
|
+ log_sig_info(sig);
|
|
+ }
|
|
+ if (!dbg_test(DBG_NOFORK) || sig != SIGALRM)
|
|
+ exit(sig + 128);
|
|
}
|
|
|
|
-static const char *postprocess_filename (const char *fn, const char *prefix)
|
|
+static const char *postprocess_filename(const char *fn, const char *prefix)
|
|
{
|
|
- char *new_fn;
|
|
+ char *new_fn;
|
|
|
|
- if (!fn)
|
|
- return NULL;
|
|
+ if (!fn)
|
|
+ return NULL;
|
|
|
|
- if (fn [0] != '/' && fn [0] != '.'){
|
|
- new_fn = xmalloc (2 + strlen (prefix) + strlen (fn));
|
|
- strcpy (new_fn, prefix);
|
|
- strcat (new_fn, fn);
|
|
+ if (fn[0] != '/' && fn[0] != '.') {
|
|
+ new_fn = xmalloc(2 + strlen(prefix) + strlen(fn));
|
|
+ strcpy(new_fn, prefix);
|
|
+ strcat(new_fn, fn);
|
|
|
|
- return new_fn;
|
|
- }else{
|
|
- return xstrdup (fn);
|
|
- }
|
|
+ return new_fn;
|
|
+ } else {
|
|
+ return xstrdup(fn);
|
|
+ }
|
|
}
|
|
|
|
-const char *postprocess_plugin_filename (const char *fn)
|
|
+const char *postprocess_plugin_filename(const char *fn)
|
|
{
|
|
- return postprocess_filename (fn, DICT_PLUGIN_PATH);
|
|
+ return postprocess_filename(fn, DICT_PLUGIN_PATH);
|
|
}
|
|
|
|
-const char *postprocess_dict_filename (const char *fn)
|
|
+const char *postprocess_dict_filename(const char *fn)
|
|
{
|
|
- return postprocess_filename (fn, DICT_DICTIONARY_PATH);
|
|
+ return postprocess_filename(fn, DICT_DICTIONARY_PATH);
|
|
}
|
|
|
|
-static void postprocess_filenames (dictConfig *dc)
|
|
+static void postprocess_filenames(dictConfig * dc)
|
|
{
|
|
- lst_Position p;
|
|
- dictDatabase *db;
|
|
+ lst_Position p;
|
|
+ dictDatabase *db;
|
|
|
|
- LST_ITERATE(dc -> dbl, p, db) {
|
|
- db -> dataFilename = postprocess_dict_filename (db -> dataFilename);
|
|
- db -> indexFilename = postprocess_dict_filename (db -> indexFilename);
|
|
- db -> indexsuffixFilename = postprocess_dict_filename (db -> indexsuffixFilename);
|
|
- db -> indexwordFilename = postprocess_dict_filename (db -> indexwordFilename);
|
|
- db -> pluginFilename = postprocess_plugin_filename (db -> pluginFilename);
|
|
- }
|
|
+ LST_ITERATE(dc->dbl, p, db) {
|
|
+ db->dataFilename = postprocess_dict_filename(db->dataFilename);
|
|
+ db->indexFilename = postprocess_dict_filename(db->indexFilename);
|
|
+ db->indexsuffixFilename =
|
|
+ postprocess_dict_filename(db->indexsuffixFilename);
|
|
+ db->indexwordFilename =
|
|
+ postprocess_dict_filename(db->indexwordFilename);
|
|
+ db->pluginFilename =
|
|
+ postprocess_plugin_filename(db->pluginFilename);
|
|
+ }
|
|
|
|
- site_info = postprocess_dict_filename (site_info);
|
|
+ site_info = postprocess_dict_filename(site_info);
|
|
}
|
|
|
|
-static void handler_sighup (int sig)
|
|
+static void handler_sighup(int sig)
|
|
{
|
|
- log_sig_info (sig);
|
|
- need_reload_config = 1;
|
|
+ log_sig_info(sig);
|
|
+ need_reload_config = 1;
|
|
}
|
|
|
|
-static void handler_sigusr1 (int sig)
|
|
+static void handler_sigusr1(int sig)
|
|
{
|
|
- log_sig_info (sig);
|
|
- need_unload_databases = 1;
|
|
+ log_sig_info(sig);
|
|
+ need_unload_databases = 1;
|
|
}
|
|
|
|
-static void setsig( int sig, void (*f)(int), int sa_flags )
|
|
+static void setsig(int sig, void (*f) (int), int sa_flags)
|
|
{
|
|
- struct sigaction sa;
|
|
+ struct sigaction sa;
|
|
|
|
- sa.sa_handler = f;
|
|
- sigemptyset(&sa.sa_mask);
|
|
- sa.sa_flags = sa_flags;
|
|
- sigaction(sig, &sa, NULL);
|
|
+ sa.sa_handler = f;
|
|
+ sigemptyset(&sa.sa_mask);
|
|
+ sa.sa_flags = sa_flags;
|
|
+ sigaction(sig, &sa, NULL);
|
|
}
|
|
|
|
struct access_print_struct {
|
|
- FILE *s;
|
|
- int offset;
|
|
+ FILE *s;
|
|
+ int offset;
|
|
};
|
|
|
|
-static int access_print( const void *datum, void *arg )
|
|
+static int access_print(const void *datum, void *arg)
|
|
{
|
|
- dictAccess *a = (dictAccess *)datum;
|
|
- struct access_print_struct *aps = (struct access_print_struct *)arg;
|
|
- FILE *s = aps->s;
|
|
- int offset = aps->offset;
|
|
- int i;
|
|
- const char *desc;
|
|
-
|
|
- for (i = 0; i < offset; i++) fputc( ' ', s );
|
|
- switch (a->type) {
|
|
- case DICT_DENY: desc = "deny"; break;
|
|
- case DICT_ALLOW: desc = "allow"; break;
|
|
- case DICT_AUTHONLY: desc = "authonly"; break;
|
|
- case DICT_USER: desc = "user"; break;
|
|
- case DICT_GROUP: desc = "group"; break; /* Not implemented. */
|
|
- default: desc = "unknown"; break;
|
|
- }
|
|
- fprintf( s, "%s %s\n", desc, a->spec );
|
|
- return 0;
|
|
-}
|
|
-
|
|
-static void acl_print( FILE *s, lst_List l, int offset)
|
|
-{
|
|
- struct access_print_struct aps;
|
|
- int i;
|
|
-
|
|
- aps.s = s;
|
|
- aps.offset = offset + 3;
|
|
-
|
|
- for (i = 0; i < offset; i++) fputc( ' ', s );
|
|
- fprintf( s, "access {\n" );
|
|
- lst_iterate_arg( l, access_print, &aps );
|
|
- for (i = 0; i < offset; i++) fputc( ' ', s );
|
|
- fprintf( s, "}\n" );
|
|
-}
|
|
-
|
|
-static int user_print( const void *key, const void *datum, void *arg )
|
|
-{
|
|
- const char *username = (const char *)key;
|
|
- const char *secret = (const char *)datum;
|
|
- FILE *s = (FILE *)arg;
|
|
-
|
|
- if (dbg_test(DBG_AUTH))
|
|
- fprintf( s, "user %s %s\n", username, secret );
|
|
- else
|
|
- fprintf( s, "user %s *\n", username );
|
|
- return 0;
|
|
-}
|
|
-
|
|
-static int config_print( const void *datum, void *arg )
|
|
-{
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- FILE *s = (FILE *)arg;
|
|
-
|
|
- fprintf( s, "database %s {\n", db->databaseName );
|
|
-
|
|
- if (db->dataFilename)
|
|
- fprintf( s, " data %s\n", db->dataFilename );
|
|
- if (db->indexFilename)
|
|
- fprintf( s, " index %s\n", db->indexFilename );
|
|
- if (db->indexsuffixFilename)
|
|
- fprintf( s, " index_suffix %s\n", db->indexsuffixFilename );
|
|
- if (db->indexwordFilename)
|
|
- fprintf( s, " index_word %s\n", db->indexwordFilename );
|
|
- if (db->filter)
|
|
- fprintf( s, " filter %s\n", db->filter );
|
|
- if (db->prefilter)
|
|
- fprintf( s, " prefilter %s\n", db->prefilter );
|
|
- if (db->postfilter)
|
|
- fprintf( s, " postfilter %s\n", db->postfilter );
|
|
- if (db->databaseShort)
|
|
- fprintf( s, " name %s\n", db->databaseShort );
|
|
- if (db->acl)
|
|
- acl_print( s, db->acl, 3 );
|
|
-
|
|
- fprintf( s, "}\n" );
|
|
-
|
|
- return 0;
|
|
-}
|
|
-
|
|
-static void dict_config_print( FILE *stream, dictConfig *c )
|
|
-{
|
|
- FILE *s = stream ? stream : stderr;
|
|
-
|
|
- if (c->acl) acl_print( s, c->acl, 0 );
|
|
- lst_iterate_arg( c->dbl, config_print, s );
|
|
- if (c->usl) hsh_iterate_arg( c->usl, user_print, s );
|
|
-}
|
|
-
|
|
-static const char *get_entry_info( dictDatabase *db, const char *entryName )
|
|
-{
|
|
- dictWord *dw;
|
|
- lst_List list = lst_create();
|
|
- char *pt, *buf;
|
|
- size_t len;
|
|
-
|
|
- if (
|
|
- 0 >= dict_search (
|
|
- list, entryName, db, DICT_STRAT_EXACT, 0,
|
|
- NULL, NULL, NULL ))
|
|
- {
|
|
+ dictAccess *a = (dictAccess *) datum;
|
|
+ struct access_print_struct *aps = (struct access_print_struct *) arg;
|
|
+ FILE *s = aps->s;
|
|
+ int offset = aps->offset;
|
|
+ int i;
|
|
+ const char *desc;
|
|
+
|
|
+ for (i = 0; i < offset; i++)
|
|
+ fputc(' ', s);
|
|
+ switch (a->type) {
|
|
+ case DICT_DENY:
|
|
+ desc = "deny";
|
|
+ break;
|
|
+ case DICT_ALLOW:
|
|
+ desc = "allow";
|
|
+ break;
|
|
+ case DICT_AUTHONLY:
|
|
+ desc = "authonly";
|
|
+ break;
|
|
+ case DICT_USER:
|
|
+ desc = "user";
|
|
+ break;
|
|
+ case DICT_GROUP:
|
|
+ desc = "group";
|
|
+ break; /* Not implemented. */
|
|
+ default:
|
|
+ desc = "unknown";
|
|
+ break;
|
|
+ }
|
|
+ fprintf(s, "%s %s\n", desc, a->spec);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void acl_print(FILE * s, lst_List l, int offset)
|
|
+{
|
|
+ struct access_print_struct aps;
|
|
+ int i;
|
|
+
|
|
+ aps.s = s;
|
|
+ aps.offset = offset + 3;
|
|
+
|
|
+ for (i = 0; i < offset; i++)
|
|
+ fputc(' ', s);
|
|
+ fprintf(s, "access {\n");
|
|
+ lst_iterate_arg(l, access_print, &aps);
|
|
+ for (i = 0; i < offset; i++)
|
|
+ fputc(' ', s);
|
|
+ fprintf(s, "}\n");
|
|
+}
|
|
+
|
|
+static int user_print(const void *key, const void *datum, void *arg)
|
|
+{
|
|
+ const char *username = (const char *) key;
|
|
+ const char *secret = (const char *) datum;
|
|
+ FILE *s = (FILE *) arg;
|
|
+
|
|
+ if (dbg_test(DBG_AUTH))
|
|
+ fprintf(s, "user %s %s\n", username, secret);
|
|
+ else
|
|
+ fprintf(s, "user %s *\n", username);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int config_print(const void *datum, void *arg)
|
|
+{
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ FILE *s = (FILE *) arg;
|
|
+
|
|
+ fprintf(s, "database %s {\n", db->databaseName);
|
|
+
|
|
+ if (db->dataFilename)
|
|
+ fprintf(s, " data %s\n", db->dataFilename);
|
|
+ if (db->indexFilename)
|
|
+ fprintf(s, " index %s\n", db->indexFilename);
|
|
+ if (db->indexsuffixFilename)
|
|
+ fprintf(s, " index_suffix %s\n", db->indexsuffixFilename);
|
|
+ if (db->indexwordFilename)
|
|
+ fprintf(s, " index_word %s\n", db->indexwordFilename);
|
|
+ if (db->filter)
|
|
+ fprintf(s, " filter %s\n", db->filter);
|
|
+ if (db->prefilter)
|
|
+ fprintf(s, " prefilter %s\n", db->prefilter);
|
|
+ if (db->postfilter)
|
|
+ fprintf(s, " postfilter %s\n", db->postfilter);
|
|
+ if (db->databaseShort)
|
|
+ fprintf(s, " name %s\n", db->databaseShort);
|
|
+ if (db->acl)
|
|
+ acl_print(s, db->acl, 3);
|
|
+
|
|
+ fprintf(s, "}\n");
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void dict_config_print(FILE * stream, dictConfig * c)
|
|
+{
|
|
+ FILE *s = stream ? stream : stderr;
|
|
+
|
|
+ if (c->acl)
|
|
+ acl_print(s, c->acl, 0);
|
|
+ lst_iterate_arg(c->dbl, config_print, s);
|
|
+ if (c->usl)
|
|
+ hsh_iterate_arg(c->usl, user_print, s);
|
|
+}
|
|
+
|
|
+static const char *get_entry_info(dictDatabase * db, const char *entryName)
|
|
+{
|
|
+ dictWord *dw;
|
|
+ lst_List list = lst_create();
|
|
+ char *pt, *buf;
|
|
+ size_t len;
|
|
+
|
|
+ if (0 >= dict_search(list, entryName, db, DICT_STRAT_EXACT, 0,
|
|
+ NULL, NULL, NULL)) {
|
|
#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
#endif
|
|
- lst_destroy( list );
|
|
- return NULL;
|
|
- }
|
|
+ lst_destroy(list);
|
|
+ return NULL;
|
|
+ }
|
|
|
|
- dw = lst_nth_get( list, 1 );
|
|
+ dw = lst_nth_get(list, 1);
|
|
|
|
- assert (dw -> database);
|
|
+ assert(dw->database);
|
|
|
|
- buf = pt = dict_data_obtain( dw -> database, dw );
|
|
+ buf = pt = dict_data_obtain(dw->database, dw);
|
|
|
|
- if (!strncmp (pt, "00database", 10) || !strncmp (pt, "00-database", 11)){
|
|
- while (*pt != '\n')
|
|
- ++pt;
|
|
+ if (!strncmp(pt, "00database", 10) || !strncmp(pt, "00-database", 11)) {
|
|
+ while (*pt != '\n')
|
|
+ ++pt;
|
|
|
|
- ++pt;
|
|
- }
|
|
+ ++pt;
|
|
+ }
|
|
|
|
- while (*pt == ' ' || *pt == '\t')
|
|
- ++pt;
|
|
+ while (*pt == ' ' || *pt == '\t')
|
|
+ ++pt;
|
|
|
|
- len = strlen(pt);
|
|
- if (pt [len - 1] == '\n')
|
|
- pt [len - 1] = '\0';
|
|
+ len = strlen(pt);
|
|
+ if (pt[len - 1] == '\n')
|
|
+ pt[len - 1] = '\0';
|
|
|
|
#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
#endif
|
|
- dict_destroy_list( list );
|
|
+ dict_destroy_list(list);
|
|
|
|
- pt = xstrdup (pt);
|
|
+ pt = xstrdup(pt);
|
|
|
|
- xfree (buf);
|
|
+ xfree(buf);
|
|
|
|
- return pt;
|
|
+ return pt;
|
|
}
|
|
|
|
-static dictDatabase *dbname2database (const char *dbname)
|
|
+static dictDatabase *dbname2database(const char *dbname)
|
|
{
|
|
- dictDatabase *db = NULL;
|
|
- lst_Position db_pos = lst_init_position (DictConfig->dbl);
|
|
+ dictDatabase *db = NULL;
|
|
+ lst_Position db_pos = lst_init_position(DictConfig->dbl);
|
|
|
|
- while (db_pos){
|
|
- db = lst_get_position (db_pos);
|
|
+ while (db_pos) {
|
|
+ db = lst_get_position(db_pos);
|
|
|
|
- if (!strcmp (db -> databaseName, dbname)){
|
|
- return db;
|
|
- }
|
|
+ if (!strcmp(db->databaseName, dbname)) {
|
|
+ return db;
|
|
+ }
|
|
|
|
- db_pos = lst_next_position (db_pos);
|
|
- }
|
|
+ db_pos = lst_next_position(db_pos);
|
|
+ }
|
|
|
|
- return NULL;
|
|
+ return NULL;
|
|
}
|
|
|
|
-static lst_List string2virtual_db_list (char *s)
|
|
+static lst_List string2virtual_db_list(char *s)
|
|
{
|
|
- int len, i;
|
|
- lst_List virtual_db_list;
|
|
- char *p;
|
|
+ int len, i;
|
|
+ lst_List virtual_db_list;
|
|
+ char *p;
|
|
|
|
- dictDatabase *db = NULL;
|
|
+ dictDatabase *db = NULL;
|
|
|
|
- p = s;
|
|
- len = strlen (s);
|
|
+ p = s;
|
|
+ len = strlen(s);
|
|
|
|
- virtual_db_list = lst_create ();
|
|
+ virtual_db_list = lst_create();
|
|
|
|
- for (i = 0; i <= len; ++i){
|
|
- if (s [i] == ',' || s [i] == '\n' || s [i] == '\0'){
|
|
- s [i] = '\0';
|
|
+ for (i = 0; i <= len; ++i) {
|
|
+ if (s[i] == ',' || s[i] == '\n' || s[i] == '\0') {
|
|
+ s[i] = '\0';
|
|
|
|
- if (*p){
|
|
- db = dbname2database (p);
|
|
+ if (*p) {
|
|
+ db = dbname2database(p);
|
|
|
|
- if (db){
|
|
- lst_append (virtual_db_list, db);
|
|
- }else{
|
|
- log_info( ":E: Unknown database '%s'\n", p );
|
|
- PRINTF(DBG_INIT, (":E: Unknown database '%s'\n", p));
|
|
- exit (2);
|
|
+ if (db) {
|
|
+ lst_append(virtual_db_list, db);
|
|
+ } else {
|
|
+ log_info(":E: Unknown database '%s'\n", p);
|
|
+ PRINTF(DBG_INIT, (":E: Unknown database '%s'\n", p));
|
|
+ exit(2);
|
|
+ }
|
|
}
|
|
- }
|
|
|
|
- p = s + i + 1;
|
|
- }
|
|
- }
|
|
-
|
|
- return virtual_db_list;
|
|
-}
|
|
-
|
|
-static int init_virtual_db_list (const void *datum)
|
|
-{
|
|
- lst_List list;
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- dictWord *dw;
|
|
- char *buf;
|
|
- int ret;
|
|
-
|
|
- if (db -> database_list){
|
|
- buf = xstrdup (db -> database_list);
|
|
- db -> virtual_db_list = string2virtual_db_list (buf);
|
|
- xfree (buf);
|
|
- }else{
|
|
- if (!db -> index)
|
|
- return 0;
|
|
-
|
|
- list = lst_create();
|
|
- ret = dict_search (
|
|
- list, DICT_FLAG_VIRTUAL, db, DICT_STRAT_EXACT, 0,
|
|
- NULL, NULL, NULL);
|
|
-
|
|
- switch (ret){
|
|
- case 1: case 2:
|
|
- dw = (dictWord *) lst_pop (list);
|
|
- buf = dict_data_obtain (db, dw);
|
|
- dict_destroy_datum (dw);
|
|
-
|
|
- db -> virtual_db_list = string2virtual_db_list (buf);
|
|
-
|
|
- xfree (buf);
|
|
- break;
|
|
- case 0:
|
|
- break;
|
|
- default:
|
|
- err_fatal (
|
|
- __func__,
|
|
- "index file contains more than one %s entry",
|
|
- DICT_FLAG_VIRTUAL);
|
|
- }
|
|
-
|
|
- dict_destroy_list (list);
|
|
- }
|
|
-
|
|
- return 0;
|
|
-}
|
|
-
|
|
-static int init_mime_db_list (const void *datum)
|
|
-{
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
-
|
|
- if (!db -> mime_db)
|
|
- return 0;
|
|
-
|
|
- /* MIME */
|
|
- if (db -> mime_mimeDbname){
|
|
- db -> mime_mimeDB = dbname2database (db -> mime_mimeDbname);
|
|
-
|
|
- if (!db -> mime_mimeDB){
|
|
- err_fatal (
|
|
- __func__,
|
|
- "Incorrect database name '%s'\n",
|
|
- db -> mime_mimeDbname);
|
|
- }
|
|
- }else{
|
|
- err_fatal (
|
|
- __func__,
|
|
- "MIME database '%s' has no mime_dbname keyword\n",
|
|
- db -> databaseName);
|
|
- }
|
|
-
|
|
- /* NO MIME */
|
|
- if (db -> mime_nomimeDbname){
|
|
- db -> mime_nomimeDB = dbname2database (db -> mime_nomimeDbname);
|
|
-
|
|
- if (!db -> mime_nomimeDB){
|
|
- err_fatal (
|
|
- __func__,
|
|
- "Incorrect database name '%s'\n",
|
|
- db -> mime_nomimeDbname);
|
|
- }
|
|
- }else{
|
|
- err_fatal (
|
|
- __func__,
|
|
- "MIME database '%s' has no nomime_dbname keyword\n",
|
|
- db -> databaseName);
|
|
- }
|
|
+ p = s + i + 1;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ return virtual_db_list;
|
|
+}
|
|
+
|
|
+static int init_virtual_db_list(const void *datum)
|
|
+{
|
|
+ lst_List list;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ dictWord *dw;
|
|
+ char *buf;
|
|
+ int ret;
|
|
+
|
|
+ if (db->database_list) {
|
|
+ buf = xstrdup(db->database_list);
|
|
+ db->virtual_db_list = string2virtual_db_list(buf);
|
|
+ xfree(buf);
|
|
+ } else {
|
|
+ if (!db->index)
|
|
+ return 0;
|
|
+
|
|
+ list = lst_create();
|
|
+ ret = dict_search(list, DICT_FLAG_VIRTUAL, db, DICT_STRAT_EXACT, 0,
|
|
+ NULL, NULL, NULL);
|
|
+
|
|
+ switch (ret) {
|
|
+ case 1:
|
|
+ case 2:
|
|
+ dw = (dictWord *) lst_pop(list);
|
|
+ buf = dict_data_obtain(db, dw);
|
|
+ dict_destroy_datum(dw);
|
|
+
|
|
+ db->virtual_db_list = string2virtual_db_list(buf);
|
|
+
|
|
+ xfree(buf);
|
|
+ break;
|
|
+ case 0:
|
|
+ break;
|
|
+ default:
|
|
+ err_fatal(__func__,
|
|
+ "index file contains more than one %s entry",
|
|
+ DICT_FLAG_VIRTUAL);
|
|
+ }
|
|
+
|
|
+ dict_destroy_list(list);
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int init_mime_db_list(const void *datum)
|
|
+{
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+
|
|
+ if (!db->mime_db)
|
|
+ return 0;
|
|
+
|
|
+ /* MIME */
|
|
+ if (db->mime_mimeDbname) {
|
|
+ db->mime_mimeDB = dbname2database(db->mime_mimeDbname);
|
|
+
|
|
+ if (!db->mime_mimeDB) {
|
|
+ err_fatal(__func__,
|
|
+ "Incorrect database name '%s'\n",
|
|
+ db->mime_mimeDbname);
|
|
+ }
|
|
+ } else {
|
|
+ err_fatal(__func__,
|
|
+ "MIME database '%s' has no mime_dbname keyword\n",
|
|
+ db->databaseName);
|
|
+ }
|
|
+
|
|
+ /* NO MIME */
|
|
+ if (db->mime_nomimeDbname) {
|
|
+ db->mime_nomimeDB = dbname2database(db->mime_nomimeDbname);
|
|
+
|
|
+ if (!db->mime_nomimeDB) {
|
|
+ err_fatal(__func__,
|
|
+ "Incorrect database name '%s'\n",
|
|
+ db->mime_nomimeDbname);
|
|
+ }
|
|
+ } else {
|
|
+ err_fatal(__func__,
|
|
+ "MIME database '%s' has no nomime_dbname keyword\n",
|
|
+ db->databaseName);
|
|
+ }
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int init_plugin( const void *datum )
|
|
+static int init_plugin(const void *datum)
|
|
{
|
|
#ifdef USE_PLUGIN
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- dict_plugin_init (db);
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ dict_plugin_init(db);
|
|
#endif
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-void dict_disable_strat (dictDatabase *db, const char* strategy)
|
|
+void dict_disable_strat(dictDatabase * db, const char *strategy)
|
|
{
|
|
- int strat = -1;
|
|
- int array_size = get_max_strategy_num () + 1;
|
|
+ int strat = -1;
|
|
+ int array_size = get_max_strategy_num() + 1;
|
|
|
|
- assert (db);
|
|
- assert (strategy);
|
|
+ assert(db);
|
|
+ assert(strategy);
|
|
|
|
- if (!db -> strategy_disabled){
|
|
- db -> strategy_disabled = xmalloc (array_size * sizeof (int));
|
|
- memset (db -> strategy_disabled, 0, array_size * sizeof (int));
|
|
- }
|
|
+ if (!db->strategy_disabled) {
|
|
+ db->strategy_disabled = xmalloc(array_size * sizeof(int));
|
|
+ memset(db->strategy_disabled, 0, array_size * sizeof(int));
|
|
+ }
|
|
|
|
- strat = lookup_strategy_ex (strategy);
|
|
- assert (strat >= 0);
|
|
+ strat = lookup_strategy_ex(strategy);
|
|
+ assert(strat >= 0);
|
|
|
|
- db -> strategy_disabled [strat] = 1;
|
|
+ db->strategy_disabled[strat] = 1;
|
|
}
|
|
|
|
-static void init_database_alphabet (dictDatabase *db)
|
|
+static void init_database_alphabet(dictDatabase * db)
|
|
{
|
|
- int ret;
|
|
- lst_List l;
|
|
- const dictWord *dw;
|
|
- char *data;
|
|
+ int ret;
|
|
+ lst_List l;
|
|
+ const dictWord *dw;
|
|
+ char *data;
|
|
|
|
- if (!db -> normal_db)
|
|
- return;
|
|
+ if (!db->normal_db)
|
|
+ return;
|
|
|
|
- l = lst_create ();
|
|
+ l = lst_create();
|
|
|
|
- ret = dict_search_database_ (l, DICT_FLAG_ALPHABET, db, DICT_STRAT_EXACT);
|
|
+ ret =
|
|
+ dict_search_database_(l, DICT_FLAG_ALPHABET, db, DICT_STRAT_EXACT);
|
|
|
|
- if (ret){
|
|
- dw = (const dictWord *) lst_top (l);
|
|
- data = dict_data_obtain (db, dw);
|
|
- db -> alphabet = data;
|
|
+ if (ret) {
|
|
+ dw = (const dictWord *) lst_top(l);
|
|
+ data = dict_data_obtain(db, dw);
|
|
+ db->alphabet = data;
|
|
|
|
- data = strchr (db -> alphabet, '\n');
|
|
- if (data)
|
|
- *data = 0;
|
|
- }
|
|
+ data = strchr(db->alphabet, '\n');
|
|
+ if (data)
|
|
+ *data = 0;
|
|
+ }
|
|
|
|
- dict_destroy_list (l);
|
|
+ dict_destroy_list(l);
|
|
}
|
|
|
|
-static void init_database_default_strategy (dictDatabase *db)
|
|
+static void init_database_default_strategy(dictDatabase * db)
|
|
{
|
|
- int ret;
|
|
- lst_List l;
|
|
- const dictWord *dw;
|
|
- char *data;
|
|
- int def_strat = -1;
|
|
- char *p;
|
|
+ int ret;
|
|
+ lst_List l;
|
|
+ const dictWord *dw;
|
|
+ char *data;
|
|
+ int def_strat = -1;
|
|
+ char *p;
|
|
|
|
- if (!db -> normal_db)
|
|
- return;
|
|
+ if (!db->normal_db)
|
|
+ return;
|
|
|
|
- if (db -> default_strategy > 0){
|
|
- /* already set by `default_strategy' directive*/
|
|
- return;
|
|
- }
|
|
+ if (db->default_strategy > 0) {
|
|
+ /* already set by `default_strategy' directive */
|
|
+ return;
|
|
+ }
|
|
|
|
- l = lst_create ();
|
|
+ l = lst_create();
|
|
|
|
- ret = dict_search_database_ (l, DICT_FLAG_DEFAULT_STRAT, db, DICT_STRAT_EXACT);
|
|
+ ret =
|
|
+ dict_search_database_(l, DICT_FLAG_DEFAULT_STRAT, db,
|
|
+ DICT_STRAT_EXACT);
|
|
|
|
- if (ret){
|
|
- dw = (const dictWord *) lst_top (l);
|
|
- data = dict_data_obtain (db, dw);
|
|
+ if (ret) {
|
|
+ dw = (const dictWord *) lst_top(l);
|
|
+ data = dict_data_obtain(db, dw);
|
|
|
|
- for (p=data; *p && isalpha ((unsigned char) *p); ++p){
|
|
- }
|
|
- *p = '\0';
|
|
+ for (p = data; *p && isalpha((unsigned char) *p); ++p) {
|
|
+ }
|
|
+ *p = '\0';
|
|
|
|
- def_strat = lookup_strategy (data);
|
|
- if (-1 == def_strat){
|
|
- PRINTF (DBG_INIT, (":I: `%s' is not supported by dictd\n", data));
|
|
- }else{
|
|
- db -> default_strategy = def_strat;
|
|
- }
|
|
+ def_strat = lookup_strategy(data);
|
|
+ if (-1 == def_strat) {
|
|
+ PRINTF(DBG_INIT,
|
|
+ (":I: `%s' is not supported by dictd\n", data));
|
|
+ } else {
|
|
+ db->default_strategy = def_strat;
|
|
+ }
|
|
|
|
- xfree (data);
|
|
- }
|
|
+ xfree(data);
|
|
+ }
|
|
|
|
- dict_destroy_list (l);
|
|
+ dict_destroy_list(l);
|
|
}
|
|
|
|
-static int init_database_mime_header (const void *datum)
|
|
+static int init_database_mime_header(const void *datum)
|
|
{
|
|
- dictDatabase *db = (dictDatabase *) datum;
|
|
- int ret;
|
|
- lst_List l;
|
|
- const dictWord *dw;
|
|
- char *data;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ int ret;
|
|
+ lst_List l;
|
|
+ const dictWord *dw;
|
|
+ char *data;
|
|
|
|
- if (!db -> normal_db)
|
|
- return 0;
|
|
+ if (!db->normal_db)
|
|
+ return 0;
|
|
|
|
- if (db -> mime_header){
|
|
- /* already set by `mime_header' directive*/
|
|
- return 0;
|
|
- }
|
|
+ if (db->mime_header) {
|
|
+ /* already set by `mime_header' directive */
|
|
+ return 0;
|
|
+ }
|
|
|
|
- l = lst_create ();
|
|
+ l = lst_create();
|
|
|
|
- ret = dict_search_database_ (l, DICT_FLAG_MIME_HEADER, db, DICT_STRAT_EXACT);
|
|
+ ret =
|
|
+ dict_search_database_(l, DICT_FLAG_MIME_HEADER, db,
|
|
+ DICT_STRAT_EXACT);
|
|
|
|
- if (ret){
|
|
- dw = (const dictWord *) lst_top (l);
|
|
- data = dict_data_obtain (db, dw);
|
|
+ if (ret) {
|
|
+ dw = (const dictWord *) lst_top(l);
|
|
+ data = dict_data_obtain(db, dw);
|
|
|
|
- db -> mime_header = xstrdup (data);
|
|
+ db->mime_header = xstrdup(data);
|
|
|
|
- xfree (data);
|
|
- }
|
|
+ xfree(data);
|
|
+ }
|
|
|
|
- dict_destroy_list (l);
|
|
+ dict_destroy_list(l);
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int init_database( const void *datum )
|
|
+static int init_database(const void *datum)
|
|
{
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- const char *strat_name = NULL;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ const char *strat_name = NULL;
|
|
|
|
- PRINTF (DBG_INIT, (":I: Initializing '%s'\n", db->databaseName));
|
|
+ PRINTF(DBG_INIT, (":I: Initializing '%s'\n", db->databaseName));
|
|
|
|
- if (db->indexFilename){
|
|
- PRINTF (DBG_INIT, (":I: Opening indices\n"));
|
|
- }
|
|
+ if (db->indexFilename) {
|
|
+ PRINTF(DBG_INIT, (":I: Opening indices\n"));
|
|
+ }
|
|
|
|
- db->index = dict_index_open( db->indexFilename, 1, NULL );
|
|
+ db->index = dict_index_open(db->indexFilename, 1, NULL);
|
|
|
|
- if (db->indexFilename){
|
|
- PRINTF (DBG_INIT, (":I: .index <ok>\n"));
|
|
- }
|
|
+ if (db->indexFilename) {
|
|
+ PRINTF(DBG_INIT, (":I: .index <ok>\n"));
|
|
+ }
|
|
|
|
- if (db->index){
|
|
- db->index_suffix = dict_index_open(
|
|
- db->indexsuffixFilename,
|
|
- 0, db->index);
|
|
+ if (db->index) {
|
|
+ db->index_suffix = dict_index_open(db->indexsuffixFilename,
|
|
+ 0, db->index);
|
|
|
|
- db->index_word = dict_index_open(
|
|
- db->indexwordFilename,
|
|
- 0, db->index);
|
|
- }
|
|
+ db->index_word = dict_index_open(db->indexwordFilename,
|
|
+ 0, db->index);
|
|
+ }
|
|
|
|
- if (db->index_suffix){
|
|
- PRINTF (DBG_INIT, (":I: .indexsuffix <ok>\n"));
|
|
- db->index_suffix->flag_8bit = db->index->flag_8bit;
|
|
- db->index_suffix->flag_utf8 = db->index->flag_utf8;
|
|
- db->index_suffix->flag_allchars = db->index->flag_allchars;
|
|
- }
|
|
- if (db->index_word){
|
|
- PRINTF (DBG_INIT, (":I: .indexword <ok>\n"));
|
|
- db->index_word->flag_utf8 = db->index->flag_utf8;
|
|
- db->index_word->flag_8bit = db->index->flag_8bit;
|
|
- db->index_word->flag_allchars = db->index->flag_allchars;
|
|
- }
|
|
+ if (db->index_suffix) {
|
|
+ PRINTF(DBG_INIT, (":I: .indexsuffix <ok>\n"));
|
|
+ db->index_suffix->flag_8bit = db->index->flag_8bit;
|
|
+ db->index_suffix->flag_utf8 = db->index->flag_utf8;
|
|
+ db->index_suffix->flag_allchars = db->index->flag_allchars;
|
|
+ }
|
|
+ if (db->index_word) {
|
|
+ PRINTF(DBG_INIT, (":I: .indexword <ok>\n"));
|
|
+ db->index_word->flag_utf8 = db->index->flag_utf8;
|
|
+ db->index_word->flag_8bit = db->index->flag_8bit;
|
|
+ db->index_word->flag_allchars = db->index->flag_allchars;
|
|
+ }
|
|
|
|
- if (db->dataFilename){
|
|
- PRINTF (DBG_INIT, (":I: Opening data\n"));
|
|
- }
|
|
+ if (db->dataFilename) {
|
|
+ PRINTF(DBG_INIT, (":I: Opening data\n"));
|
|
+ }
|
|
|
|
- db->data = dict_data_open( db->dataFilename, 0 );
|
|
+ db->data = dict_data_open(db->dataFilename, 0);
|
|
|
|
- init_database_alphabet (db);
|
|
- if (db -> alphabet){
|
|
- PRINTF (DBG_INIT, (":I: alphabet: %s\n", db -> alphabet));
|
|
- }else{
|
|
- PRINTF (DBG_INIT, (":I: alphabet: (NULL)\n"));
|
|
- }
|
|
+ init_database_alphabet(db);
|
|
+ if (db->alphabet) {
|
|
+ PRINTF(DBG_INIT, (":I: alphabet: %s\n", db->alphabet));
|
|
+ } else {
|
|
+ PRINTF(DBG_INIT, (":I: alphabet: (NULL)\n"));
|
|
+ }
|
|
|
|
- if (db -> default_strategy){
|
|
- strat_name = get_strategy (db -> default_strategy) -> name;
|
|
- PRINTF (DBG_INIT, (":I: default_strategy (from conf file): %s\n",
|
|
- strat_name));
|
|
- }else{
|
|
- init_database_default_strategy (db);
|
|
- if (db -> default_strategy){
|
|
- strat_name = get_strategy (db -> default_strategy) -> name;
|
|
- PRINTF (DBG_INIT, (":I: default_strategy (from db): %s\n", strat_name));
|
|
- }else{
|
|
- db -> default_strategy = default_strategy;
|
|
- }
|
|
- }
|
|
+ if (db->default_strategy) {
|
|
+ strat_name = get_strategy(db->default_strategy)->name;
|
|
+ PRINTF(DBG_INIT,
|
|
+ (":I: default_strategy (from conf file): %s\n",
|
|
+ strat_name));
|
|
+ } else {
|
|
+ init_database_default_strategy(db);
|
|
+ if (db->default_strategy) {
|
|
+ strat_name = get_strategy(db->default_strategy)->name;
|
|
+ PRINTF(DBG_INIT,
|
|
+ (":I: default_strategy (from db): %s\n",
|
|
+ strat_name));
|
|
+ } else {
|
|
+ db->default_strategy = default_strategy;
|
|
+ }
|
|
+ }
|
|
|
|
- if (db->dataFilename){
|
|
- PRINTF(DBG_INIT,
|
|
- (":I: '%s' initialized\n", db->databaseName));
|
|
- }
|
|
+ if (db->dataFilename) {
|
|
+ PRINTF(DBG_INIT, (":I: '%s' initialized\n", db->databaseName));
|
|
+ }
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int init_database_short (const void *datum)
|
|
+static int init_database_short(const void *datum)
|
|
{
|
|
- char *NL;
|
|
+ char *NL;
|
|
|
|
- dictDatabase *db = (dictDatabase *) datum;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
|
|
- if (!db->databaseShort){
|
|
- db->databaseShort = get_entry_info( db, DICT_SHORT_ENTRY_NAME );
|
|
- }else if (*db->databaseShort == '@'){
|
|
- db->databaseShort = get_entry_info( db, db->databaseShort + 1 );
|
|
- }else{
|
|
- db->databaseShort = xstrdup (db->databaseShort);
|
|
- }
|
|
+ if (!db->databaseShort) {
|
|
+ db->databaseShort = get_entry_info(db, DICT_SHORT_ENTRY_NAME);
|
|
+ } else if (*db->databaseShort == '@') {
|
|
+ db->databaseShort = get_entry_info(db, db->databaseShort + 1);
|
|
+ } else {
|
|
+ db->databaseShort = xstrdup(db->databaseShort);
|
|
+ }
|
|
|
|
- if (db->databaseShort){
|
|
- NL = strchr (db->databaseShort, '\n');
|
|
- if (NL)
|
|
- *NL = 0;
|
|
- }
|
|
+ if (db->databaseShort) {
|
|
+ NL = strchr(db->databaseShort, '\n');
|
|
+ if (NL)
|
|
+ *NL = 0;
|
|
+ }
|
|
|
|
- if (!db->databaseShort)
|
|
- db->databaseShort = xstrdup (db->databaseName);
|
|
+ if (!db->databaseShort)
|
|
+ db->databaseShort = xstrdup(db->databaseName);
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int close_plugin (const void *datum)
|
|
+static int close_plugin(const void *datum)
|
|
{
|
|
#ifdef USE_PLUGIN
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- dict_plugin_destroy (db);
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ dict_plugin_destroy(db);
|
|
#endif
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int close_database (const void *datum)
|
|
+static int close_database(const void *datum)
|
|
{
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
|
|
- dict_index_close (db->index);
|
|
- dict_index_close (db->index_suffix);
|
|
- dict_index_close (db->index_word);
|
|
+ dict_index_close(db->index);
|
|
+ dict_index_close(db->index_suffix);
|
|
+ dict_index_close(db->index_word);
|
|
|
|
- dict_data_close (db->data);
|
|
+ dict_data_close(db->data);
|
|
|
|
- if (db -> databaseShort)
|
|
- xfree ((void *) db -> databaseShort);
|
|
+ if (db->databaseShort)
|
|
+ xfree((void *) db->databaseShort);
|
|
|
|
- if (db -> indexFilename)
|
|
- xfree ((void *) db -> indexFilename);
|
|
- if (db -> dataFilename)
|
|
- xfree ((void *) db -> dataFilename);
|
|
- if (db -> indexwordFilename)
|
|
- xfree ((void *) db -> indexwordFilename);
|
|
- if (db -> indexsuffixFilename)
|
|
- xfree ((void *) db -> indexsuffixFilename);
|
|
- if (db -> pluginFilename)
|
|
- xfree ((void *) db -> pluginFilename);
|
|
- if (db -> strategy_disabled)
|
|
- xfree ((void *) db -> strategy_disabled);
|
|
- if (db -> alphabet)
|
|
- xfree ((void *) db -> alphabet);
|
|
- if (db -> mime_header)
|
|
- xfree ((void *) db -> mime_header);
|
|
+ if (db->indexFilename)
|
|
+ xfree((void *) db->indexFilename);
|
|
+ if (db->dataFilename)
|
|
+ xfree((void *) db->dataFilename);
|
|
+ if (db->indexwordFilename)
|
|
+ xfree((void *) db->indexwordFilename);
|
|
+ if (db->indexsuffixFilename)
|
|
+ xfree((void *) db->indexsuffixFilename);
|
|
+ if (db->pluginFilename)
|
|
+ xfree((void *) db->pluginFilename);
|
|
+ if (db->strategy_disabled)
|
|
+ xfree((void *) db->strategy_disabled);
|
|
+ if (db->alphabet)
|
|
+ xfree((void *) db->alphabet);
|
|
+ if (db->mime_header)
|
|
+ xfree((void *) db->mime_header);
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static int log_database_info( const void *datum )
|
|
+static int log_database_info(const void *datum)
|
|
{
|
|
- dictDatabase *db = (dictDatabase *)datum;
|
|
- const char *pt;
|
|
- unsigned long headwords = 0;
|
|
+ dictDatabase *db = (dictDatabase *) datum;
|
|
+ const char *pt;
|
|
+ unsigned long headwords = 0;
|
|
|
|
- if (db->index){
|
|
- for (pt = db->index->start; pt < db->index->end; pt++)
|
|
- if (*pt == '\n') ++headwords;
|
|
- db->index->headwords = headwords;
|
|
+ if (db->index) {
|
|
+ for (pt = db->index->start; pt < db->index->end; pt++)
|
|
+ if (*pt == '\n')
|
|
+ ++headwords;
|
|
+ db->index->headwords = headwords;
|
|
|
|
- log_info( ":I: %-12.12s %12lu %12lu %12lu %12lu\n",
|
|
- db->databaseName, headwords,
|
|
- db->index->size, db->data->size, db->data->length );
|
|
- }
|
|
+ log_info(":I: %-12.12s %12lu %12lu %12lu %12lu\n",
|
|
+ db->databaseName, headwords,
|
|
+ db->index->size, db->data->size, db->data->length);
|
|
+ }
|
|
|
|
- return 0;
|
|
+ return 0;
|
|
}
|
|
|
|
-static void dict_ltdl_init ()
|
|
+static void dict_ltdl_init()
|
|
{
|
|
#if defined(USE_PLUGIN) && !HAVE_DLFCN_H
|
|
- if (lt_dlinit ())
|
|
- err_fatal( __func__, "Can not initialize 'ltdl' library\n" );
|
|
+ if (lt_dlinit())
|
|
+ err_fatal(__func__, "Can not initialize 'ltdl' library\n");
|
|
#endif
|
|
}
|
|
|
|
-static void dict_ltdl_close ()
|
|
+static void dict_ltdl_close()
|
|
{
|
|
#if defined(USE_PLUGIN) && !HAVE_DLFCN_H
|
|
- if (lt_dlexit ())
|
|
- err_fatal( __func__, "Can not deinitialize 'ltdl' library\n" );
|
|
+ if (lt_dlexit())
|
|
+ err_fatal(__func__, "Can not deinitialize 'ltdl' library\n");
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
Makes dictionary_exit db invisible if it is the last visible one
|
|
*/
|
|
-static void make_dictexit_invisible (dictConfig *c)
|
|
+static void make_dictexit_invisible(dictConfig * c)
|
|
+{
|
|
+ lst_Position p;
|
|
+ dictDatabase *db;
|
|
+ dictDatabase *db_exit = NULL;
|
|
+
|
|
+ LST_ITERATE(c->dbl, p, db) {
|
|
+ if (!db->invisible) {
|
|
+ if (db_exit)
|
|
+ db_exit->invisible = 0;
|
|
+
|
|
+ db_exit = NULL;
|
|
+ }
|
|
+
|
|
+ if (db->exit_db) {
|
|
+ db_exit = db;
|
|
+ db_exit->invisible = 1;
|
|
+ }
|
|
+ }
|
|
+}
|
|
+
|
|
+static void dict_init_databases(dictConfig * c)
|
|
+{
|
|
+ make_dictexit_invisible(c);
|
|
+
|
|
+ lst_iterate(c->dbl, init_database);
|
|
+ lst_iterate(c->dbl, init_plugin);
|
|
+ lst_iterate(c->dbl, init_virtual_db_list);
|
|
+ lst_iterate(c->dbl, init_mime_db_list);
|
|
+ lst_iterate(c->dbl, init_database_short);
|
|
+ lst_iterate(c->dbl, init_database_mime_header);
|
|
+ lst_iterate(c->dbl, log_database_info);
|
|
+}
|
|
+
|
|
+static void dict_close_databases(dictConfig * c)
|
|
+{
|
|
+ dictDatabase *db;
|
|
+ dictAccess *acl;
|
|
+
|
|
+ if (!c)
|
|
+ return;
|
|
+
|
|
+ if (c->dbl) {
|
|
+ while (lst_length(c->dbl) > 0) {
|
|
+ db = (dictDatabase *) lst_pop(c->dbl);
|
|
+
|
|
+ if (db->virtual_db_list)
|
|
+ lst_destroy(db->virtual_db_list);
|
|
+
|
|
+ close_plugin(db);
|
|
+ close_database(db);
|
|
+ xfree(db);
|
|
+ }
|
|
+ lst_destroy(c->dbl);
|
|
+ }
|
|
+
|
|
+ if (c->acl) {
|
|
+ while (lst_length(c->acl) > 0) {
|
|
+ acl = (dictAccess *) lst_pop(c->acl);
|
|
+ xfree(acl);
|
|
+ }
|
|
+ lst_destroy(c->acl);
|
|
+ }
|
|
+
|
|
+ if (site_info)
|
|
+ xfree((void *) site_info);
|
|
+
|
|
+ xfree(c);
|
|
+}
|
|
+
|
|
+static const char *id_string(void)
|
|
+{
|
|
+ static char buffer[BUFFERSIZE];
|
|
+
|
|
+ snprintf(buffer, BUFFERSIZE, "%s", DICT_VERSION);
|
|
+
|
|
+ return buffer;
|
|
+}
|
|
+
|
|
+const char *dict_get_banner(int shortFlag)
|
|
+{
|
|
+ static char *shortBuffer = NULL;
|
|
+ static char *longBuffer = NULL;
|
|
+ struct utsname uts;
|
|
+
|
|
+ if (shortFlag && shortBuffer)
|
|
+ return shortBuffer;
|
|
+ if (!shortFlag && longBuffer)
|
|
+ return longBuffer;
|
|
+
|
|
+ uname(&uts);
|
|
+
|
|
+ shortBuffer = xmalloc(256);
|
|
+ snprintf(shortBuffer, 256, "%s %s", err_program_name(), id_string());
|
|
+
|
|
+ longBuffer = xmalloc(256);
|
|
+ snprintf(longBuffer, 256,
|
|
+ "%s %s/rf on %s %s", err_program_name(), id_string(),
|
|
+ uts.sysname, uts.release);
|
|
+
|
|
+ if (shortFlag)
|
|
+ return shortBuffer;
|
|
+
|
|
+ return longBuffer;
|
|
+}
|
|
+
|
|
+static void banner(void)
|
|
{
|
|
- lst_Position p;
|
|
- dictDatabase *db;
|
|
- dictDatabase *db_exit = NULL;
|
|
-
|
|
- LST_ITERATE(c -> dbl, p, db) {
|
|
- if (!db -> invisible){
|
|
- if (db_exit)
|
|
- db_exit -> invisible = 0;
|
|
-
|
|
- db_exit = NULL;
|
|
- }
|
|
-
|
|
- if (db -> exit_db){
|
|
- db_exit = db;
|
|
- db_exit -> invisible = 1;
|
|
- }
|
|
- }
|
|
-}
|
|
-
|
|
-static void dict_init_databases( dictConfig *c )
|
|
-{
|
|
- make_dictexit_invisible (c);
|
|
-
|
|
- lst_iterate( c->dbl, init_database );
|
|
- lst_iterate( c->dbl, init_plugin );
|
|
- lst_iterate( c->dbl, init_virtual_db_list );
|
|
- lst_iterate( c->dbl, init_mime_db_list );
|
|
- lst_iterate( c->dbl, init_database_short );
|
|
- lst_iterate( c->dbl, init_database_mime_header);
|
|
- lst_iterate( c->dbl, log_database_info );
|
|
-}
|
|
-
|
|
-static void dict_close_databases (dictConfig *c)
|
|
-{
|
|
- dictDatabase *db;
|
|
- dictAccess *acl;
|
|
-
|
|
- if (!c)
|
|
- return;
|
|
-
|
|
- if (c -> dbl){
|
|
- while (lst_length (c -> dbl) > 0){
|
|
- db = (dictDatabase *) lst_pop (c -> dbl);
|
|
-
|
|
- if (db -> virtual_db_list)
|
|
- lst_destroy (db -> virtual_db_list);
|
|
-
|
|
- close_plugin (db);
|
|
- close_database (db);
|
|
- xfree (db);
|
|
- }
|
|
- lst_destroy (c -> dbl);
|
|
- }
|
|
-
|
|
- if (c -> acl){
|
|
- while (lst_length (c -> acl) > 0){
|
|
- acl = (dictAccess *) lst_pop (c->acl);
|
|
- xfree (acl);
|
|
- }
|
|
- lst_destroy (c -> acl);
|
|
- }
|
|
-
|
|
- if (site_info)
|
|
- xfree ((void *) site_info);
|
|
-
|
|
- xfree (c);
|
|
-}
|
|
-
|
|
-static const char *id_string (void)
|
|
-{
|
|
- static char buffer [BUFFERSIZE];
|
|
-
|
|
- snprintf( buffer, BUFFERSIZE, "%s", DICT_VERSION );
|
|
-
|
|
- return buffer;
|
|
-}
|
|
-
|
|
-const char *dict_get_banner( int shortFlag )
|
|
-{
|
|
- static char *shortBuffer = NULL;
|
|
- static char *longBuffer = NULL;
|
|
- struct utsname uts;
|
|
-
|
|
- if (shortFlag && shortBuffer) return shortBuffer;
|
|
- if (!shortFlag && longBuffer) return longBuffer;
|
|
-
|
|
- uname( &uts );
|
|
-
|
|
- shortBuffer = xmalloc(256);
|
|
- snprintf(
|
|
- shortBuffer, 256,
|
|
- "%s %s", err_program_name(), id_string () );
|
|
-
|
|
- longBuffer = xmalloc(256);
|
|
- snprintf(
|
|
- longBuffer, 256,
|
|
- "%s %s/rf on %s %s", err_program_name(), id_string (),
|
|
- uts.sysname,
|
|
- uts.release );
|
|
-
|
|
- if (shortFlag)
|
|
- return shortBuffer;
|
|
-
|
|
- return longBuffer;
|
|
-}
|
|
-
|
|
-static void banner( void )
|
|
-{
|
|
- printf( "%s\n", dict_get_banner(0) );
|
|
- printf( "Copyright 1997-2002 Rickard E. Faith (faith@dict.org)\n" );
|
|
- printf( "Copyright 2002-2007 Aleksey Cheusov (vle@gmx.net)\n" );
|
|
- printf( "\n" );
|
|
-}
|
|
-
|
|
-static void license( void )
|
|
-{
|
|
- static const char *license_msg[] = {
|
|
- "This program is free software; you can redistribute it and/or modify it",
|
|
- "under the terms of the GNU General Public License as published by the",
|
|
- "Free Software Foundation; either version 1, or (at your option) any",
|
|
- "later version.",
|
|
- "",
|
|
- "This program is distributed in the hope that it will be useful, but",
|
|
- "WITHOUT ANY WARRANTY; without even the implied warranty of",
|
|
- "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU",
|
|
- "General Public License for more details.",
|
|
- "",
|
|
- "You should have received a copy of the GNU General Public License along",
|
|
- "with this program; if not, write to the Free Software Foundation, Inc.,",
|
|
- "675 Mass Ave, Cambridge, MA 02139, USA.",
|
|
- 0 };
|
|
- const char **p = license_msg;
|
|
-
|
|
- banner();
|
|
- while (*p) printf( " %s\n", *p++ );
|
|
-}
|
|
-
|
|
-static void help( void )
|
|
-{
|
|
- static const char *help_msg[] = {
|
|
- "Usage: dictd [options]",
|
|
- "Start the dictd daemon",
|
|
- "",
|
|
- "-h --help give this help",
|
|
- " --license display software license",
|
|
- "-v --verbose verbose mode",
|
|
- "-V --version display version number",
|
|
- "-p --port <port> port number",
|
|
- " --delay <seconds> client timeout in seconds",
|
|
- " --depth <length> TCP/IP queue depth",
|
|
- " --limit <children> maximum simultaneous children",
|
|
- "-c --config <file> configuration file",
|
|
- "-l --log <option> select logging option",
|
|
- "-s --syslog log via syslog(3)",
|
|
- "-L --logfile <file> log via specified file",
|
|
- "-m --mark <minutes> how often should a timestamp be logged",
|
|
- " --facility <fac> set syslog logging facility",
|
|
- "-d --debug <option> select debug option",
|
|
- "-i --inetd run from inetd",
|
|
- " --pid-file <path> PID filename",
|
|
- " --pp <prog> set preprocessor for configuration file",
|
|
- "-f --force force startup even if daemon running",
|
|
- " --locale <locale> specifies the locale used for searching.\n\
|
|
+ printf("%s\n", dict_get_banner(0));
|
|
+ printf("Copyright 1997-2002 Rickard E. Faith (faith@dict.org)\n");
|
|
+ printf("Copyright 2002-2007 Aleksey Cheusov (vle@gmx.net)\n");
|
|
+ printf("\n");
|
|
+}
|
|
+
|
|
+static void license(void)
|
|
+{
|
|
+ static const char *license_msg[] = {
|
|
+ "This program is free software; you can redistribute it and/or modify it",
|
|
+ "under the terms of the GNU General Public License as published by the",
|
|
+ "Free Software Foundation; either version 1, or (at your option) any",
|
|
+ "later version.",
|
|
+ "",
|
|
+ "This program is distributed in the hope that it will be useful, but",
|
|
+ "WITHOUT ANY WARRANTY; without even the implied warranty of",
|
|
+ "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU",
|
|
+ "General Public License for more details.",
|
|
+ "",
|
|
+ "You should have received a copy of the GNU General Public License along",
|
|
+ "with this program; if not, write to the Free Software Foundation, Inc.,",
|
|
+ "675 Mass Ave, Cambridge, MA 02139, USA.",
|
|
+ 0
|
|
+ };
|
|
+ const char **p = license_msg;
|
|
+
|
|
+ banner();
|
|
+ while (*p)
|
|
+ printf(" %s\n", *p++);
|
|
+}
|
|
+
|
|
+static void help(void)
|
|
+{
|
|
+ static const char *help_msg[] = {
|
|
+ "Usage: dictd [options]",
|
|
+ "Start the dictd daemon",
|
|
+ "",
|
|
+ "-h --help give this help",
|
|
+ " --license display software license",
|
|
+ "-v --verbose verbose mode",
|
|
+ "-V --version display version number",
|
|
+ "-p --port <port> port number",
|
|
+ " --delay <seconds> client timeout in seconds",
|
|
+ " --depth <length> TCP/IP queue depth",
|
|
+ " --limit <children> maximum simultaneous children",
|
|
+ "-c --config <file> configuration file",
|
|
+ "-l --log <option> select logging option",
|
|
+ "-s --syslog log via syslog(3)",
|
|
+ "-L --logfile <file> log via specified file",
|
|
+ "-m --mark <minutes> how often should a timestamp be logged",
|
|
+ " --facility <fac> set syslog logging facility",
|
|
+ "-d --debug <option> select debug option",
|
|
+ "-i --inetd run from inetd",
|
|
+ " --pid-file <path> PID filename",
|
|
+ " --pp <prog> set preprocessor for configuration file",
|
|
+ "-f --force force startup even if daemon running",
|
|
+ " --locale <locale> specifies the locale used for searching.\n\
|
|
if no locale is specified, the \"C\" locale is used.",
|
|
-" --default-strategy set the default search strategy for 'match' queries.\n\
|
|
+ " --default-strategy set the default search strategy for 'match' queries.\n\
|
|
the default is 'lev'.",
|
|
-" --without-strategy <strategies> disable strategies.\n\
|
|
+ " --without-strategy <strategies> disable strategies.\n\
|
|
<strategies> is a comma-separated list.",
|
|
-" --add-strategy <strat>:<descr> adds new strategy <strat>\n\
|
|
+ " --add-strategy <strat>:<descr> adds new strategy <strat>\n\
|
|
with a description <descr>.",
|
|
-" --listen-to bind a socket to the specified address",
|
|
-"\n------------------ options for debugging ---------------------------",
|
|
-" --fast-start don't create additional (internal) index.",
|
|
+ " --listen-to bind a socket to the specified address",
|
|
+ "\n------------------ options for debugging ---------------------------",
|
|
+ " --fast-start don't create additional (internal) index.",
|
|
#ifdef HAVE_MMAP
|
|
-" --without-mmap do not use mmap() function and load files\n\
|
|
+ " --without-mmap do not use mmap() function and load files\n\
|
|
into memory instead.",
|
|
#endif
|
|
-" --stdin2stdout copy stdin to stdout (addition to -i option).",
|
|
- 0 };
|
|
- const char **p = help_msg;
|
|
+ " --stdin2stdout copy stdin to stdout (addition to -i option).",
|
|
+ 0
|
|
+ };
|
|
+ const char **p = help_msg;
|
|
|
|
- banner();
|
|
- while (*p)
|
|
- printf( "%s\n", *p++ );
|
|
+ banner();
|
|
+ while (*p)
|
|
+ printf("%s\n", *p++);
|
|
}
|
|
|
|
-void set_minimal( void )
|
|
+void set_minimal(void)
|
|
{
|
|
- flg_set(flg_name(LOG_FOUND));
|
|
- flg_set(flg_name(LOG_NOTFOUND));
|
|
- flg_set(flg_name(LOG_STATS));
|
|
- flg_set(flg_name(LOG_CLIENT));
|
|
- flg_set(flg_name(LOG_AUTH));
|
|
- flg_set("-min");
|
|
+ flg_set(flg_name(LOG_FOUND));
|
|
+ flg_set(flg_name(LOG_NOTFOUND));
|
|
+ flg_set(flg_name(LOG_STATS));
|
|
+ flg_set(flg_name(LOG_CLIENT));
|
|
+ flg_set(flg_name(LOG_AUTH));
|
|
+ flg_set("-min");
|
|
}
|
|
|
|
-static void release_root_privileges( void )
|
|
+static void release_root_privileges(void)
|
|
/* At the spring 1999 Linux Expo in Raleigh, Rik Faith told me that he
|
|
* did not want dictd to be allowed to run as root for any reason.
|
|
* This patch irrevocably releases root privileges. -- Kirk Hilliard
|
|
@@ -1266,24 +1291,24 @@ static void release_root_privileges( voi
|
|
* -- Bob Hilliard
|
|
*/
|
|
{
|
|
- int unused __attribute__((unused));
|
|
- if (geteuid() == 0) {
|
|
- struct passwd *pwd;
|
|
-
|
|
- if ((pwd = getpwnam("dictd"))) {
|
|
- unused = setgid(pwd->pw_gid);
|
|
- initgroups("dictd",pwd->pw_gid);
|
|
- unused = setuid(pwd->pw_uid);
|
|
- } else if ((pwd = getpwnam("nobody"))) {
|
|
- unused = setgid(pwd->pw_gid);
|
|
- initgroups("nobody",pwd->pw_gid);
|
|
- unused = setuid(pwd->pw_uid);
|
|
- } else {
|
|
- unused = setgid(GID_NOGROUP);
|
|
- initgroups("nobody", GID_NOGROUP);
|
|
- unused = setuid(UID_NOBODY);
|
|
- }
|
|
- }
|
|
+ int unused __attribute__ ((unused));
|
|
+ if (geteuid() == 0) {
|
|
+ struct passwd *pwd;
|
|
+
|
|
+ if ((pwd = getpwnam("dictd"))) {
|
|
+ unused = setgid(pwd->pw_gid);
|
|
+ initgroups("dictd", pwd->pw_gid);
|
|
+ unused = setuid(pwd->pw_uid);
|
|
+ } else if ((pwd = getpwnam("nobody"))) {
|
|
+ unused = setgid(pwd->pw_gid);
|
|
+ initgroups("nobody", pwd->pw_gid);
|
|
+ unused = setuid(pwd->pw_uid);
|
|
+ } else {
|
|
+ unused = setgid(GID_NOGROUP);
|
|
+ initgroups("nobody", GID_NOGROUP);
|
|
+ unused = setuid(UID_NOBODY);
|
|
+ }
|
|
+ }
|
|
}
|
|
|
|
/* Perform sanity checks that are often problems for people trying to
|
|
@@ -1291,538 +1316,580 @@ static void release_root_privileges( voi
|
|
* console. */
|
|
static void sanity(const char *confFile)
|
|
{
|
|
- int fail = 0;
|
|
- int reading_error = 0;
|
|
- struct passwd *pw = NULL;
|
|
- struct group *gr = NULL;
|
|
-
|
|
- if (access(confFile,R_OK)) {
|
|
- log_info(":E: %s is not readable (config file)\n", confFile);
|
|
- ++fail;
|
|
- }
|
|
- if (DictConfig && !DictConfig->dbl) {
|
|
- log_info(":E: no databases have been defined\n");
|
|
- log_info(":E: check %s or use -c\n", confFile);
|
|
- ++fail;
|
|
- }
|
|
- if (DictConfig && DictConfig->dbl) {
|
|
- lst_Position p;
|
|
- dictDatabase *e;
|
|
- LST_ITERATE(DictConfig->dbl, p, e) {
|
|
- if (e->indexFilename && access(e->indexFilename, R_OK)) {
|
|
- log_info(":E: %s is not readable (index file)\n",
|
|
- e->indexFilename);
|
|
- ++fail;
|
|
- reading_error = 1;
|
|
- }
|
|
- if (e->indexsuffixFilename && access(e->indexsuffixFilename, R_OK)) {
|
|
- log_info(":E: %s is not readable (index_suffix file)\n",
|
|
- e->indexsuffixFilename);
|
|
- ++fail;
|
|
- reading_error = 1;
|
|
- }
|
|
- if (e->indexwordFilename && access(e->indexwordFilename, R_OK)) {
|
|
- log_info(":E: %s is not readable (index_word file)\n",
|
|
- e->indexwordFilename);
|
|
- ++fail;
|
|
- reading_error = 1;
|
|
- }
|
|
-
|
|
- if (e->dataFilename && access(e->dataFilename, R_OK)) {
|
|
- log_info(":E: %s is not readable (data file)\n",
|
|
- e->dataFilename);
|
|
- ++fail;
|
|
- reading_error = 1;
|
|
- }
|
|
- if (e->virtual_db && !e->database_list){
|
|
- log_info(
|
|
- ":E: database list is not specified for virtual dictionary '%s'\n",
|
|
- e->databaseName);
|
|
- ++fail;
|
|
- }
|
|
- if (e->normal_db && !e->dataFilename){
|
|
- log_info(
|
|
- ":E: data filename is not specified for dictionary '%s'\n",
|
|
- e->databaseName);
|
|
- ++fail;
|
|
- }
|
|
- if (e->normal_db && !e->indexFilename){
|
|
- log_info(
|
|
- ":E: index filename is not specified for dictionary '%s'\n",
|
|
- e->databaseName);
|
|
- ++fail;
|
|
- }
|
|
- if (e->plugin_db && !e->pluginFilename){
|
|
- log_info(
|
|
- ":E: plugin filename is not specified for dictionary '%s'\n",
|
|
- e->databaseName);
|
|
- ++fail;
|
|
- }
|
|
+ int fail = 0;
|
|
+ int reading_error = 0;
|
|
+ struct passwd *pw = NULL;
|
|
+ struct group *gr = NULL;
|
|
+
|
|
+ if (access(confFile, R_OK)) {
|
|
+ log_info(":E: %s is not readable (config file)\n", confFile);
|
|
+ ++fail;
|
|
+ }
|
|
+ if (DictConfig && !DictConfig->dbl) {
|
|
+ log_info(":E: no databases have been defined\n");
|
|
+ log_info(":E: check %s or use -c\n", confFile);
|
|
+ ++fail;
|
|
+ }
|
|
+ if (DictConfig && DictConfig->dbl) {
|
|
+ lst_Position p;
|
|
+ dictDatabase *e;
|
|
+ LST_ITERATE(DictConfig->dbl, p, e) {
|
|
+ if (e->indexFilename && access(e->indexFilename, R_OK)) {
|
|
+ log_info(":E: %s is not readable (index file)\n",
|
|
+ e->indexFilename);
|
|
+ ++fail;
|
|
+ reading_error = 1;
|
|
+ }
|
|
+ if (e->indexsuffixFilename
|
|
+ && access(e->indexsuffixFilename, R_OK)) {
|
|
+ log_info(":E: %s is not readable (index_suffix file)\n",
|
|
+ e->indexsuffixFilename);
|
|
+ ++fail;
|
|
+ reading_error = 1;
|
|
+ }
|
|
+ if (e->indexwordFilename && access(e->indexwordFilename, R_OK)) {
|
|
+ log_info(":E: %s is not readable (index_word file)\n",
|
|
+ e->indexwordFilename);
|
|
+ ++fail;
|
|
+ reading_error = 1;
|
|
+ }
|
|
+
|
|
+ if (e->dataFilename && access(e->dataFilename, R_OK)) {
|
|
+ log_info(":E: %s is not readable (data file)\n",
|
|
+ e->dataFilename);
|
|
+ ++fail;
|
|
+ reading_error = 1;
|
|
+ }
|
|
+ if (e->virtual_db && !e->database_list) {
|
|
+ log_info
|
|
+ (":E: database list is not specified for virtual dictionary '%s'\n",
|
|
+ e->databaseName);
|
|
+ ++fail;
|
|
+ }
|
|
+ if (e->normal_db && !e->dataFilename) {
|
|
+ log_info
|
|
+ (":E: data filename is not specified for dictionary '%s'\n",
|
|
+ e->databaseName);
|
|
+ ++fail;
|
|
+ }
|
|
+ if (e->normal_db && !e->indexFilename) {
|
|
+ log_info
|
|
+ (":E: index filename is not specified for dictionary '%s'\n",
|
|
+ e->databaseName);
|
|
+ ++fail;
|
|
+ }
|
|
+ if (e->plugin_db && !e->pluginFilename) {
|
|
+ log_info
|
|
+ (":E: plugin filename is not specified for dictionary '%s'\n",
|
|
+ e->databaseName);
|
|
+ ++fail;
|
|
+ }
|
|
#ifndef USE_PLUGIN
|
|
- if (e -> plugin_db){
|
|
- log_info (
|
|
- ":E: plugin support was disabled at compile time\n");
|
|
- ++fail;
|
|
- }
|
|
+ if (e->plugin_db) {
|
|
+ log_info
|
|
+ (":E: plugin support was disabled at compile time\n");
|
|
+ ++fail;
|
|
+ }
|
|
#endif
|
|
- }
|
|
- }
|
|
- if (fail) {
|
|
- if (reading_error){
|
|
- pw = getpwuid (geteuid ());
|
|
- gr = getgrgid (getegid ());
|
|
-
|
|
- log_info(":E: for security, this program will not run as root.\n");
|
|
- log_info(":E: if started as root, this program will change"
|
|
- " to \"dictd\" or \"nobody\".\n");
|
|
- log_info(":E: currently running as user %d/%s, group %d/%s\n",
|
|
- geteuid(), pw && pw->pw_name ? pw->pw_name : "?",
|
|
- getegid(), gr && gr->gr_name ? gr->gr_name : "?");
|
|
- log_info(":E: config and db files must be readable by that user\n");
|
|
- }
|
|
- err_fatal(__func__, ":E: terminating due to errors. See log file\n");
|
|
- }
|
|
-}
|
|
-
|
|
-static void set_locale_and_flags (const char *loc)
|
|
-{
|
|
- const char *charset = NULL;
|
|
- int ascii_mode;
|
|
-
|
|
- if (!setlocale(LC_COLLATE, loc) || !setlocale(LC_CTYPE, loc)){
|
|
- fprintf (stderr, "invalid locale '%s'\n", locale);
|
|
- exit (2);
|
|
- }
|
|
+ }
|
|
+ }
|
|
+ if (fail) {
|
|
+ if (reading_error) {
|
|
+ pw = getpwuid(geteuid());
|
|
+ gr = getgrgid(getegid());
|
|
+
|
|
+ log_info
|
|
+ (":E: for security, this program will not run as root.\n");
|
|
+ log_info(":E: if started as root, this program will change"
|
|
+ " to \"dictd\" or \"nobody\".\n");
|
|
+ log_info(":E: currently running as user %d/%s, group %d/%s\n",
|
|
+ geteuid(), pw
|
|
+ && pw->pw_name ? pw->pw_name : "?", getegid(), gr
|
|
+ && gr->gr_name ? gr->gr_name : "?");
|
|
+ log_info
|
|
+ (":E: config and db files must be readable by that user\n");
|
|
+ }
|
|
+ err_fatal(__func__,
|
|
+ ":E: terminating due to errors. See log file\n");
|
|
+ }
|
|
+}
|
|
+
|
|
+static void set_locale_and_flags(const char *loc)
|
|
+{
|
|
+ const char *charset = NULL;
|
|
+ int ascii_mode;
|
|
+
|
|
+ if (!setlocale(LC_COLLATE, loc) || !setlocale(LC_CTYPE, loc)) {
|
|
+ fprintf(stderr, "invalid locale '%s'\n", locale);
|
|
+ exit(2);
|
|
+ }
|
|
|
|
- charset = nl_langinfo (CODESET);
|
|
+ charset = nl_langinfo(CODESET);
|
|
|
|
- utf8_mode = !strcmp (charset, "UTF-8") || !strcmp (charset, "utf-8");
|
|
+ utf8_mode = !strcmp(charset, "UTF-8") || !strcmp(charset, "utf-8");
|
|
|
|
#if !HAVE_UTF8
|
|
- if (utf8_mode){
|
|
- err_fatal (
|
|
- __func__,
|
|
- "utf-8 support was disabled at compile time\n");
|
|
- }
|
|
+ if (utf8_mode) {
|
|
+ err_fatal(__func__,
|
|
+ "utf-8 support was disabled at compile time\n");
|
|
+ }
|
|
#endif
|
|
|
|
- ascii_mode =
|
|
- !strcmp (charset, "ANSI_X3.4-1968") ||
|
|
- !strcmp (charset, "US-ASCII") ||
|
|
- (locale [0] == 'C' && locale [1] == 0);
|
|
+ ascii_mode =
|
|
+ !strcmp(charset, "ANSI_X3.4-1968") ||
|
|
+ !strcmp(charset, "US-ASCII") ||
|
|
+ (locale[0] == 'C' && locale[1] == 0);
|
|
|
|
- bit8_mode = !ascii_mode && !utf8_mode;
|
|
+ bit8_mode = !ascii_mode && !utf8_mode;
|
|
}
|
|
|
|
-static void set_umask (void)
|
|
+static void set_umask(void)
|
|
{
|
|
#if defined(__OPENNT) || defined(__INTERIX)
|
|
- umask(002); /* set safe umask */
|
|
+ umask(002); /* set safe umask */
|
|
#else
|
|
- umask(022); /* set safe umask */
|
|
+ umask(022); /* set safe umask */
|
|
#endif
|
|
}
|
|
|
|
-static void init (const char *fn)
|
|
+static void init(const char *fn)
|
|
{
|
|
- maa_init (fn);
|
|
- dict_ltdl_init ();
|
|
- dict_init_strategies ();
|
|
+ maa_init(fn);
|
|
+ dict_ltdl_init();
|
|
+ dict_init_strategies();
|
|
}
|
|
|
|
-static void destroy ()
|
|
+static void destroy()
|
|
{
|
|
- maa_shutdown ();
|
|
+ maa_shutdown();
|
|
|
|
- dict_ltdl_close ();
|
|
- dict_destroy_strategies ();
|
|
+ dict_ltdl_close();
|
|
+ dict_destroy_strategies();
|
|
}
|
|
|
|
FILE *pid_fd = NULL;
|
|
|
|
-static void pid_file_create ()
|
|
+static void pid_file_create()
|
|
{
|
|
- pid_fd = fopen (pidFile, "w");
|
|
+ pid_fd = fopen(pidFile, "w");
|
|
|
|
- if (!pid_fd){
|
|
- log_info(":E: cannot open pid file '%s'\n:E: err msg: %s\n",
|
|
- pidFile, strerror (errno));
|
|
- err_fatal(__func__,
|
|
- ":E: terminating due to errors. See log file\n");
|
|
- }
|
|
-}
|
|
-
|
|
-static void pid_file_write ()
|
|
-{
|
|
- if (-1 == fprintf (pid_fd, "%lu\n", (unsigned long) getpid ()) ||
|
|
- fclose (pid_fd))
|
|
- {
|
|
- log_info(":E: cannot write to pid file '%s'\n:E: err msg: %s\n",
|
|
- pidFile, strerror (errno));
|
|
- err_fatal(__func__,
|
|
- ":E: terminating due to errors. See log file\n");
|
|
- }
|
|
-}
|
|
-
|
|
-static void reopen_012 (void)
|
|
-{
|
|
- int fd = open ("/dev/null", O_RDWR);
|
|
- int unused __attribute__((unused));
|
|
- if (fd == -1)
|
|
- err_fatal_errno (__func__, ":E: can't open /dev/null");
|
|
-
|
|
- close (0);
|
|
- close (1);
|
|
- close (2);
|
|
-
|
|
- unused = dup (fd);
|
|
- unused = dup (fd);
|
|
- unused = dup (fd);
|
|
-}
|
|
-
|
|
-int main (int argc, char **argv, char **envp)
|
|
-{
|
|
- int childSocket;
|
|
- int masterSocket;
|
|
- struct sockaddr_in csin;
|
|
- int c;
|
|
- time_t startTime;
|
|
- socklen_t alen = sizeof (csin);
|
|
- int detach = 1;
|
|
- int forceStartup = 0;
|
|
- int i;
|
|
-
|
|
- int errno_accept = 0;
|
|
- int unused __attribute__((unused));
|
|
-
|
|
- const char * default_strategy_arg = "???";
|
|
-
|
|
- char * new_strategy;
|
|
- char * new_strategy_descr;
|
|
-
|
|
- struct option longopts[] = {
|
|
- { "verbose", 0, 0, 'v' },
|
|
- { "version", 0, 0, 'V' },
|
|
- { "debug", 1, 0, 'd' },
|
|
- { "port", 1, 0, 'p' },
|
|
- { "config", 1, 0, 'c' },
|
|
- { "help", 0, 0, 'h' },
|
|
- { "license", 0, 0, 500 },
|
|
- { "log", 1, 0, 'l' },
|
|
- { "logfile", 1, 0, 'L' },
|
|
- { "syslog", 0, 0, 's' },
|
|
- { "mark", 1, 0, 'm' },
|
|
- { "delay", 1, 0, 502 },
|
|
- { "depth", 1, 0, 503 },
|
|
- { "limit", 1, 0, 504 },
|
|
- { "facility", 1, 0, 505 },
|
|
- { "force", 1, 0, 'f' },
|
|
- { "inetd", 0, 0, 'i' },
|
|
- { "locale", 1, 0, 506 },
|
|
+ if (!pid_fd) {
|
|
+ log_info(":E: cannot open pid file '%s'\n:E: err msg: %s\n",
|
|
+ pidFile, strerror(errno));
|
|
+ err_fatal(__func__,
|
|
+ ":E: terminating due to errors. See log file\n");
|
|
+ }
|
|
+}
|
|
+
|
|
+static void pid_file_write()
|
|
+{
|
|
+ if (-1 == fprintf(pid_fd, "%lu\n", (unsigned long) getpid()) ||
|
|
+ fclose(pid_fd)) {
|
|
+ log_info(":E: cannot write to pid file '%s'\n:E: err msg: %s\n",
|
|
+ pidFile, strerror(errno));
|
|
+ err_fatal(__func__,
|
|
+ ":E: terminating due to errors. See log file\n");
|
|
+ }
|
|
+}
|
|
+
|
|
+static void reopen_012(void)
|
|
+{
|
|
+ int fd = open("/dev/null", O_RDWR);
|
|
+ int unused __attribute__ ((unused));
|
|
+ if (fd == -1)
|
|
+ err_fatal_errno(__func__, ":E: can't open /dev/null");
|
|
+
|
|
+ close(0);
|
|
+ close(1);
|
|
+ close(2);
|
|
+
|
|
+ unused = dup(fd);
|
|
+ unused = dup(fd);
|
|
+ unused = dup(fd);
|
|
+}
|
|
+
|
|
+int main(int argc, char **argv, char **envp)
|
|
+{
|
|
+ int childSocket;
|
|
+ int masterSocket;
|
|
+ struct sockaddr_storage csin;
|
|
+ int c;
|
|
+ time_t startTime;
|
|
+ socklen_t alen = sizeof(csin);
|
|
+ int detach = 1;
|
|
+ int forceStartup = 0;
|
|
+ int i;
|
|
+
|
|
+ int errno_accept = 0;
|
|
+ int unused __attribute__ ((unused));
|
|
+
|
|
+ const char *default_strategy_arg = "???";
|
|
+
|
|
+ char *new_strategy;
|
|
+ char *new_strategy_descr;
|
|
+
|
|
+ struct option longopts[] = {
|
|
+ {"verbose", 0, 0, 'v'},
|
|
+ {"version", 0, 0, 'V'},
|
|
+ {"debug", 1, 0, 'd'},
|
|
+ {"port", 1, 0, 'p'},
|
|
+ {"config", 1, 0, 'c'},
|
|
+ {"help", 0, 0, 'h'},
|
|
+ {"license", 0, 0, 500},
|
|
+ {"log", 1, 0, 'l'},
|
|
+ {"logfile", 1, 0, 'L'},
|
|
+ {"syslog", 0, 0, 's'},
|
|
+ {"mark", 1, 0, 'm'},
|
|
+ {"delay", 1, 0, 502},
|
|
+ {"depth", 1, 0, 503},
|
|
+ {"limit", 1, 0, 504},
|
|
+ {"facility", 1, 0, 505},
|
|
+ {"force", 1, 0, 'f'},
|
|
+ {"inetd", 0, 0, 'i'},
|
|
+ {"locale", 1, 0, 506},
|
|
#ifdef HAVE_MMAP
|
|
- { "no-mmap", 0, 0, 508 },
|
|
- { "without-mmap", 0, 0, 508 },
|
|
+ {"no-mmap", 0, 0, 508},
|
|
+ {"without-mmap", 0, 0, 508},
|
|
#endif
|
|
- { "default-strategy", 1, 0, 511 },
|
|
- { "without-strategy", 1, 0, 513 },
|
|
- { "add-strategy", 1, 0, 516 },
|
|
- { "fast-start", 0, 0, 517 },
|
|
- { "pp", 1, 0, 518 },
|
|
- { "listen-to", 1, 0, 519 },
|
|
- { "pid-file", 1, 0, 521 },
|
|
- { "stdin2stdout", 0, 0, 522 },
|
|
- { 0, 0, 0, 0 }
|
|
- };
|
|
-
|
|
- set_umask ();
|
|
- init (argv[0]);
|
|
-
|
|
- flg_register( LOG_SERVER, "server" );
|
|
- flg_register( LOG_CONNECT, "connect" );
|
|
- flg_register( LOG_STATS, "stats" );
|
|
- flg_register( LOG_COMMAND, "command" );
|
|
- flg_register( LOG_FOUND, "found" );
|
|
- flg_register( LOG_NOTFOUND, "notfound" );
|
|
- flg_register( LOG_CLIENT, "client" );
|
|
- flg_register( LOG_HOST, "host" );
|
|
- flg_register( LOG_TIMESTAMP, "timestamp" );
|
|
- flg_register( LOG_MIN, "min" );
|
|
- flg_register( LOG_AUTH, "auth" );
|
|
-
|
|
- dbg_register( DBG_VERBOSE, "verbose" );
|
|
- dbg_register( DBG_UNZIP, "unzip" );
|
|
- dbg_register( DBG_SCAN, "scan" );
|
|
- dbg_register( DBG_PARSE, "parse" );
|
|
- dbg_register( DBG_SEARCH, "search" );
|
|
- dbg_register( DBG_INIT, "init" );
|
|
- dbg_register( DBG_PORT, "port" );
|
|
- dbg_register( DBG_LEV, "lev" );
|
|
- dbg_register( DBG_AUTH, "auth" );
|
|
- dbg_register( DBG_NODETACH, "nodetach" );
|
|
- dbg_register( DBG_NOFORK, "nofork" );
|
|
- dbg_register( DBG_ALT, "alt" );
|
|
-
|
|
- log_stream ("dictd", stderr);
|
|
-
|
|
- while ((c = getopt_long( argc, argv,
|
|
- "vVd:p:c:hL:t:l:sm:fi", longopts, NULL )) != EOF)
|
|
- switch (c) {
|
|
- /* Remember to copy optarg since we're
|
|
- going to destroy argv soon... */
|
|
- case 'v': dbg_set( "verbose" ); break;
|
|
- case 'V': banner(); exit(1); break;
|
|
- case 'd': dbg_set( optarg ); break;
|
|
- case 'p':
|
|
- daemon_service = str_copy(optarg);
|
|
- daemon_service_set = 1;
|
|
- break;
|
|
- case 'c': configFile = str_copy(optarg); break;
|
|
- case 'L':
|
|
- logFile = str_copy(optarg);
|
|
- logFile_set = 1;
|
|
- break;
|
|
- case 's': ++useSyslog; break;
|
|
- case 'm':
|
|
- _dict_markTime = 60*atoi(optarg);
|
|
- _dict_markTime_set = 1;
|
|
- break;
|
|
- case 'f': ++forceStartup; break;
|
|
- case 'i':
|
|
- inetd = 1;
|
|
- optStart_mode = 0;
|
|
- break;
|
|
- case 'l':
|
|
- ++logOptions;
|
|
- flg_set( optarg );
|
|
- if (flg_test(LOG_MIN)) set_minimal();
|
|
- break;
|
|
- case 500: license(); exit(1); break;
|
|
- case 502:
|
|
- client_delay = atoi(optarg);
|
|
- client_delay_set = 1;
|
|
- _dict_daemon_limit_time = 0;
|
|
- break;
|
|
- case 503:
|
|
- depth = atoi(optarg);
|
|
- depth_set = 1;
|
|
- break;
|
|
- case 504:
|
|
- _dict_daemon_limit_childs = atoi(optarg);
|
|
- _dict_daemon_limit_childs_set = 1;
|
|
- break;
|
|
- case 505:
|
|
- ++useSyslog;
|
|
- log_set_facility (optarg);
|
|
- syslog_facility_set = 1;
|
|
- break;
|
|
- case 506:
|
|
- locale = str_copy (optarg);
|
|
- locale_set = 1;
|
|
- break;
|
|
- case 508: mmap_mode = 0; break;
|
|
- case 511:
|
|
- default_strategy_arg = str_copy (optarg);
|
|
- default_strategy = lookup_strategy_ex (default_strategy_arg);
|
|
- default_strategy_set = 1;
|
|
- break;
|
|
- case 513:
|
|
- dict_disable_strategies (optarg);
|
|
- break;
|
|
- case 516:
|
|
- new_strategy = optarg;
|
|
- new_strategy_descr = strchr (new_strategy, ':');
|
|
- if (!new_strategy_descr){
|
|
- fprintf (stderr, "missing ':' symbol in --add-strategy option\n");
|
|
- exit (1);
|
|
- }
|
|
-
|
|
- *new_strategy_descr++ = 0;
|
|
-
|
|
- dict_add_strategy (new_strategy, new_strategy_descr);
|
|
-
|
|
- break;
|
|
- case 517: optStart_mode = 0; break;
|
|
- case 518: preprocessor = str_copy (optarg); break;
|
|
- case 519:
|
|
- bind_to = str_copy (optarg);
|
|
- bind_to_set = 1;
|
|
- break;
|
|
- case 521:
|
|
- pidFile = str_copy(optarg);
|
|
- pidFile_set = 1;
|
|
- break;
|
|
- case 522:
|
|
- stdin2stdout_mode = 1;
|
|
- break;
|
|
- case 'h':
|
|
- default: help(); exit(0); break;
|
|
- }
|
|
-
|
|
- if (inetd)
|
|
- detach = 0;
|
|
-
|
|
- if (-1 == default_strategy){
|
|
- fprintf (stderr, "%s is not a valid search strategy\n",
|
|
- default_strategy_arg);
|
|
-
|
|
- fprintf (stderr, "available ones are:\n");
|
|
-
|
|
- for (i = 0; i < get_strategy_count (); ++i){
|
|
- fprintf (
|
|
- stderr, " %15s : %s\n",
|
|
- get_strategies () [i] -> name, get_strategies () [i] -> description);
|
|
- }
|
|
- exit (1);
|
|
- }
|
|
-
|
|
- if (dbg_test(DBG_NOFORK)) dbg_set_flag( DBG_NODETACH);
|
|
- if (dbg_test(DBG_NODETACH)) detach = 0;
|
|
- if (dbg_test(DBG_PARSE)) prs_set_debug(1);
|
|
- if (dbg_test(DBG_SCAN)) yy_flex_debug = 1;
|
|
- else yy_flex_debug = 0;
|
|
- if (flg_test(LOG_TIMESTAMP)) log_option( LOG_OPTION_FULL );
|
|
- else log_option( LOG_OPTION_NO_FULL );
|
|
-
|
|
- if (!access(configFile,R_OK)) {
|
|
- prs_file_pp (preprocessor, configFile );
|
|
- postprocess_filenames (DictConfig);
|
|
- }
|
|
-
|
|
- if (detach) pid_file_create (); /* before leaving root priviledges */
|
|
-
|
|
- release_root_privileges();
|
|
-
|
|
- if (logFile) log_file ("dictd", logFile);
|
|
- log_stream ("dictd", NULL);
|
|
- if (useSyslog) log_syslog ("dictd");
|
|
- if (! inetd && ! detach) log_stream ("dictd", stderr);
|
|
- if ((logFile || useSyslog || !detach) && !logOptions)
|
|
- set_minimal();
|
|
-
|
|
- if (detach){
|
|
- /* become a daemon */
|
|
- unused = daemon (0, 1);
|
|
- reopen_012 ();
|
|
-
|
|
- /* after fork from daemon(3) */
|
|
- pid_file_write ();
|
|
- }
|
|
-
|
|
- time(&startTime);
|
|
- log_info(":I: %d starting %s %24.24s\n",
|
|
- getpid(), dict_get_banner(0), ctime(&startTime));
|
|
-
|
|
- tim_start( "dictd" );
|
|
- alarm(_dict_markTime);
|
|
-
|
|
- if (locale)
|
|
- set_locale_and_flags (locale);
|
|
-
|
|
- sanity(configFile);
|
|
-
|
|
- setsig(SIGCHLD, reaper, SA_RESTART);
|
|
- setsig(SIGHUP, handler_sighup, 0);
|
|
- setsig(SIGUSR1, handler_sigusr1, 0);
|
|
- if (!dbg_test(DBG_NOFORK))
|
|
- setsig(SIGINT, handler, 0);
|
|
- setsig(SIGQUIT, handler, 0);
|
|
- setsig(SIGILL, handler, 0);
|
|
- setsig(SIGTRAP, handler, 0);
|
|
- setsig(SIGTERM, handler, 0);
|
|
- setsig(SIGPIPE, handler, 0);
|
|
- setsig(SIGALRM, handler, SA_RESTART);
|
|
-
|
|
- fflush(stdout);
|
|
- fflush(stderr);
|
|
-
|
|
- if (locale)
|
|
- log_info(":I: using locale \"%s\"\n", locale);
|
|
-
|
|
- if (dbg_test(DBG_VERBOSE))
|
|
- dict_config_print( NULL, DictConfig );
|
|
-
|
|
- dict_init_databases( DictConfig );
|
|
-
|
|
- dict_initsetproctitle(argc, argv, envp);
|
|
-
|
|
- if (inetd) {
|
|
- dict_inetd(0);
|
|
- exit(0);
|
|
- }
|
|
-
|
|
- masterSocket = net_open_tcp( bind_to, daemon_service, depth );
|
|
-
|
|
-
|
|
- for (;;) {
|
|
- dict_setproctitle( "%s: %d/%d",
|
|
- dict_get_banner(1),
|
|
- _dict_forks - _dict_reaps,
|
|
- _dict_forks );
|
|
+ {"default-strategy", 1, 0, 511},
|
|
+ {"without-strategy", 1, 0, 513},
|
|
+ {"add-strategy", 1, 0, 516},
|
|
+ {"fast-start", 0, 0, 517},
|
|
+ {"pp", 1, 0, 518},
|
|
+ {"listen-to", 1, 0, 519},
|
|
+ {"pid-file", 1, 0, 521},
|
|
+ {"stdin2stdout", 0, 0, 522},
|
|
+ {0, 0, 0, 0}
|
|
+ };
|
|
+
|
|
+ set_umask();
|
|
+ init(argv[0]);
|
|
+
|
|
+ flg_register(LOG_SERVER, "server");
|
|
+ flg_register(LOG_CONNECT, "connect");
|
|
+ flg_register(LOG_STATS, "stats");
|
|
+ flg_register(LOG_COMMAND, "command");
|
|
+ flg_register(LOG_FOUND, "found");
|
|
+ flg_register(LOG_NOTFOUND, "notfound");
|
|
+ flg_register(LOG_CLIENT, "client");
|
|
+ flg_register(LOG_HOST, "host");
|
|
+ flg_register(LOG_TIMESTAMP, "timestamp");
|
|
+ flg_register(LOG_MIN, "min");
|
|
+ flg_register(LOG_AUTH, "auth");
|
|
+
|
|
+ dbg_register(DBG_VERBOSE, "verbose");
|
|
+ dbg_register(DBG_UNZIP, "unzip");
|
|
+ dbg_register(DBG_SCAN, "scan");
|
|
+ dbg_register(DBG_PARSE, "parse");
|
|
+ dbg_register(DBG_SEARCH, "search");
|
|
+ dbg_register(DBG_INIT, "init");
|
|
+ dbg_register(DBG_PORT, "port");
|
|
+ dbg_register(DBG_LEV, "lev");
|
|
+ dbg_register(DBG_AUTH, "auth");
|
|
+ dbg_register(DBG_NODETACH, "nodetach");
|
|
+ dbg_register(DBG_NOFORK, "nofork");
|
|
+ dbg_register(DBG_ALT, "alt");
|
|
+
|
|
+ log_stream("dictd", stderr);
|
|
+
|
|
+ while ((c = getopt_long(argc, argv,
|
|
+ "vVd:p:c:hL:t:l:sm:fi", longopts,
|
|
+ NULL)) != EOF)
|
|
+ switch (c) {
|
|
+ /* Remember to copy optarg since we're
|
|
+ going to destroy argv soon... */
|
|
+ case 'v':
|
|
+ dbg_set("verbose");
|
|
+ break;
|
|
+ case 'V':
|
|
+ banner();
|
|
+ exit(1);
|
|
+ break;
|
|
+ case 'd':
|
|
+ dbg_set(optarg);
|
|
+ break;
|
|
+ case 'p':
|
|
+ daemon_service = str_copy(optarg);
|
|
+ daemon_service_set = 1;
|
|
+ break;
|
|
+ case 'c':
|
|
+ configFile = str_copy(optarg);
|
|
+ break;
|
|
+ case 'L':
|
|
+ logFile = str_copy(optarg);
|
|
+ logFile_set = 1;
|
|
+ break;
|
|
+ case 's':
|
|
+ ++useSyslog;
|
|
+ break;
|
|
+ case 'm':
|
|
+ _dict_markTime = 60 * atoi(optarg);
|
|
+ _dict_markTime_set = 1;
|
|
+ break;
|
|
+ case 'f':
|
|
+ ++forceStartup;
|
|
+ break;
|
|
+ case 'i':
|
|
+ inetd = 1;
|
|
+ optStart_mode = 0;
|
|
+ break;
|
|
+ case 'l':
|
|
+ ++logOptions;
|
|
+ flg_set(optarg);
|
|
+ if (flg_test(LOG_MIN))
|
|
+ set_minimal();
|
|
+ break;
|
|
+ case 500:
|
|
+ license();
|
|
+ exit(1);
|
|
+ break;
|
|
+ case 502:
|
|
+ client_delay = atoi(optarg);
|
|
+ client_delay_set = 1;
|
|
+ _dict_daemon_limit_time = 0;
|
|
+ break;
|
|
+ case 503:
|
|
+ depth = atoi(optarg);
|
|
+ depth_set = 1;
|
|
+ break;
|
|
+ case 504:
|
|
+ _dict_daemon_limit_childs = atoi(optarg);
|
|
+ _dict_daemon_limit_childs_set = 1;
|
|
+ break;
|
|
+ case 505:
|
|
+ ++useSyslog;
|
|
+ log_set_facility(optarg);
|
|
+ syslog_facility_set = 1;
|
|
+ break;
|
|
+ case 506:
|
|
+ locale = str_copy(optarg);
|
|
+ locale_set = 1;
|
|
+ break;
|
|
+ case 508:
|
|
+ mmap_mode = 0;
|
|
+ break;
|
|
+ case 511:
|
|
+ default_strategy_arg = str_copy(optarg);
|
|
+ default_strategy = lookup_strategy_ex(default_strategy_arg);
|
|
+ default_strategy_set = 1;
|
|
+ break;
|
|
+ case 513:
|
|
+ dict_disable_strategies(optarg);
|
|
+ break;
|
|
+ case 516:
|
|
+ new_strategy = optarg;
|
|
+ new_strategy_descr = strchr(new_strategy, ':');
|
|
+ if (!new_strategy_descr) {
|
|
+ fprintf(stderr,
|
|
+ "missing ':' symbol in --add-strategy option\n");
|
|
+ exit(1);
|
|
+ }
|
|
+
|
|
+ *new_strategy_descr++ = 0;
|
|
|
|
- if (flg_test(LOG_SERVER))
|
|
- log_info( ":I: %d accepting on %s\n", getpid(), daemon_service );
|
|
+ dict_add_strategy(new_strategy, new_strategy_descr);
|
|
+
|
|
+ break;
|
|
+ case 517:
|
|
+ optStart_mode = 0;
|
|
+ break;
|
|
+ case 518:
|
|
+ preprocessor = str_copy(optarg);
|
|
+ break;
|
|
+ case 519:
|
|
+ bind_to = str_copy(optarg);
|
|
+ bind_to_set = 1;
|
|
+ break;
|
|
+ case 521:
|
|
+ pidFile = str_copy(optarg);
|
|
+ pidFile_set = 1;
|
|
+ break;
|
|
+ case 522:
|
|
+ stdin2stdout_mode = 1;
|
|
+ break;
|
|
+ case 'h':
|
|
+ default:
|
|
+ help();
|
|
+ exit(0);
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ if (inetd)
|
|
+ detach = 0;
|
|
+
|
|
+ if (-1 == default_strategy) {
|
|
+ fprintf(stderr, "%s is not a valid search strategy\n",
|
|
+ default_strategy_arg);
|
|
+
|
|
+ fprintf(stderr, "available ones are:\n");
|
|
+
|
|
+ for (i = 0; i < get_strategy_count(); ++i) {
|
|
+ fprintf(stderr, " %15s : %s\n",
|
|
+ get_strategies()[i]->name,
|
|
+ get_strategies()[i]->description);
|
|
+ }
|
|
+ exit(1);
|
|
+ }
|
|
+
|
|
+ if (dbg_test(DBG_NOFORK))
|
|
+ dbg_set_flag(DBG_NODETACH);
|
|
+ if (dbg_test(DBG_NODETACH))
|
|
+ detach = 0;
|
|
+ if (dbg_test(DBG_PARSE))
|
|
+ prs_set_debug(1);
|
|
+ if (dbg_test(DBG_SCAN))
|
|
+ yy_flex_debug = 1;
|
|
+ else
|
|
+ yy_flex_debug = 0;
|
|
+ if (flg_test(LOG_TIMESTAMP))
|
|
+ log_option(LOG_OPTION_FULL);
|
|
+ else
|
|
+ log_option(LOG_OPTION_NO_FULL);
|
|
+
|
|
+ if (!access(configFile, R_OK)) {
|
|
+ prs_file_pp(preprocessor, configFile);
|
|
+ postprocess_filenames(DictConfig);
|
|
+ }
|
|
+
|
|
+ if (detach)
|
|
+ pid_file_create(); /* before leaving root priviledges */
|
|
+
|
|
+ release_root_privileges();
|
|
+
|
|
+ if (logFile)
|
|
+ log_file("dictd", logFile);
|
|
+ log_stream("dictd", NULL);
|
|
+ if (useSyslog)
|
|
+ log_syslog("dictd");
|
|
+ if (!inetd && !detach)
|
|
+ log_stream("dictd", stderr);
|
|
+ if ((logFile || useSyslog || !detach) && !logOptions)
|
|
+ set_minimal();
|
|
+
|
|
+ if (detach) {
|
|
+ /* become a daemon */
|
|
+ unused = daemon(0, 1);
|
|
+ reopen_012();
|
|
+
|
|
+ /* after fork from daemon(3) */
|
|
+ pid_file_write();
|
|
+ }
|
|
+
|
|
+ time(&startTime);
|
|
+ log_info(":I: %d starting %s %24.24s\n",
|
|
+ getpid(), dict_get_banner(0), ctime(&startTime));
|
|
+
|
|
+ tim_start("dictd");
|
|
+ alarm(_dict_markTime);
|
|
+
|
|
+ if (locale)
|
|
+ set_locale_and_flags(locale);
|
|
+
|
|
+ sanity(configFile);
|
|
+
|
|
+ setsig(SIGCHLD, reaper, SA_RESTART);
|
|
+ setsig(SIGHUP, handler_sighup, 0);
|
|
+ setsig(SIGUSR1, handler_sigusr1, 0);
|
|
+ if (!dbg_test(DBG_NOFORK))
|
|
+ setsig(SIGINT, handler, 0);
|
|
+ setsig(SIGQUIT, handler, 0);
|
|
+ setsig(SIGILL, handler, 0);
|
|
+ setsig(SIGTRAP, handler, 0);
|
|
+ setsig(SIGTERM, handler, 0);
|
|
+ setsig(SIGPIPE, handler, 0);
|
|
+ setsig(SIGALRM, handler, SA_RESTART);
|
|
+
|
|
+ fflush(stdout);
|
|
+ fflush(stderr);
|
|
+
|
|
+ if (locale)
|
|
+ log_info(":I: using locale \"%s\"\n", locale);
|
|
+
|
|
+ if (dbg_test(DBG_VERBOSE))
|
|
+ dict_config_print(NULL, DictConfig);
|
|
+
|
|
+ dict_init_databases(DictConfig);
|
|
+
|
|
+ dict_initsetproctitle(argc, argv, envp);
|
|
+
|
|
+ if (inetd) {
|
|
+ dict_inetd(0);
|
|
+ exit(0);
|
|
+ }
|
|
+
|
|
+ masterSocket = net_open_tcp(bind_to, daemon_service, depth);
|
|
+
|
|
+
|
|
+ for (;;) {
|
|
+ dict_setproctitle("%s: %d/%d",
|
|
+ dict_get_banner(1),
|
|
+ _dict_forks - _dict_reaps, _dict_forks);
|
|
+
|
|
+ if (flg_test(LOG_SERVER))
|
|
+ log_info(":I: %d accepting on %s\n", getpid(), daemon_service);
|
|
|
|
/*unblock_signals ();*/
|
|
- childSocket = accept (masterSocket,
|
|
- (struct sockaddr *)&csin, &alen);
|
|
- errno_accept = errno;
|
|
+ childSocket = accept(masterSocket,
|
|
+ (struct sockaddr *) &csin, &alen);
|
|
+ errno_accept = errno;
|
|
/*block_signals ();*/
|
|
|
|
- if (childSocket < 0){
|
|
- switch (errno_accept){
|
|
- case EINTR:
|
|
- if (need_reload_config){
|
|
- reload_config ();
|
|
- need_reload_config = 0;
|
|
- databases_unloaded = 0;
|
|
+ if (childSocket < 0) {
|
|
+ switch (errno_accept) {
|
|
+ case EINTR:
|
|
+ if (need_reload_config) {
|
|
+ reload_config();
|
|
+ need_reload_config = 0;
|
|
+ databases_unloaded = 0;
|
|
+ }
|
|
+
|
|
+ if (need_unload_databases) {
|
|
+ unload_databases();
|
|
+ need_unload_databases = 0;
|
|
+ databases_unloaded = 1;
|
|
+ }
|
|
+ continue;
|
|
+ case ECONNABORTED:
|
|
+ case ECONNRESET:
|
|
+ case ETIMEDOUT:
|
|
+ case EHOSTUNREACH:
|
|
+ case ENETUNREACH:
|
|
+ continue;
|
|
+ default:
|
|
+ log_info(":E: can't accept: errno = %d: %s\n",
|
|
+ errno_accept, strerror(errno_accept));
|
|
+ err_fatal_errno(__func__, ":E: can't accept");
|
|
}
|
|
+ }
|
|
|
|
- if (need_unload_databases){
|
|
- unload_databases ();
|
|
- need_unload_databases = 0;
|
|
- databases_unloaded = 1;
|
|
- }
|
|
- continue;
|
|
- case ECONNABORTED:
|
|
- case ECONNRESET:
|
|
- case ETIMEDOUT:
|
|
- case EHOSTUNREACH:
|
|
- case ENETUNREACH:
|
|
- continue;
|
|
- default:
|
|
- log_info (":E: can't accept: errno = %d: %s\n",
|
|
- errno_accept, strerror (errno_accept));
|
|
- err_fatal_errno (__func__, ":E: can't accept");
|
|
- }
|
|
- }
|
|
-
|
|
- if (_dict_daemon || dbg_test(DBG_NOFORK)) {
|
|
- dict_daemon(childSocket,&csin,0);
|
|
- } else {
|
|
- if (_dict_forks - _dict_reaps < _dict_daemon_limit_childs) {
|
|
- if (!start_daemon()) { /* child */
|
|
- int databases_loaded = (DictConfig != NULL);
|
|
-
|
|
- alarm(0);
|
|
- if (_dict_daemon_limit_time){
|
|
- setsig (SIGALRM, handler, 0);
|
|
- alarm(_dict_daemon_limit_time);
|
|
- }
|
|
-
|
|
- dict_daemon (childSocket, &csin, databases_loaded ? 0 : 2);
|
|
-
|
|
- exit(0);
|
|
- } else { /* parent */
|
|
- close(childSocket);
|
|
+ if (_dict_daemon || dbg_test(DBG_NOFORK)) {
|
|
+ dict_daemon(childSocket, &csin, 0);
|
|
+ } else {
|
|
+ if (_dict_forks - _dict_reaps < _dict_daemon_limit_childs) {
|
|
+ if (!start_daemon()) { /* child */
|
|
+ int databases_loaded = (DictConfig != NULL);
|
|
+
|
|
+ alarm(0);
|
|
+ if (_dict_daemon_limit_time) {
|
|
+ setsig(SIGALRM, handler, 0);
|
|
+ alarm(_dict_daemon_limit_time);
|
|
+ }
|
|
+
|
|
+ dict_daemon(childSocket, &csin,
|
|
+ databases_loaded ? 0 : 2);
|
|
+
|
|
+ exit(0);
|
|
+ } else { /* parent */
|
|
+ close(childSocket);
|
|
+ }
|
|
+ } else {
|
|
+ dict_daemon(childSocket, &csin, 1);
|
|
}
|
|
- } else {
|
|
- dict_daemon(childSocket, &csin, 1);
|
|
- }
|
|
- }
|
|
- }
|
|
+ }
|
|
+ }
|
|
|
|
- dict_close_databases (DictConfig);
|
|
+ dict_close_databases(DictConfig);
|
|
|
|
- destroy ();
|
|
- return 0;
|
|
+ destroy();
|
|
+ return 0;
|
|
}
|
|
--- a/dict.c
|
|
+++ b/dict.c
|
|
@@ -23,14 +23,14 @@
|
|
#include "md5.h"
|
|
#include <stdarg.h>
|
|
|
|
-extern int yy_flex_debug;
|
|
- lst_List dict_Servers;
|
|
- FILE *dict_output;
|
|
- FILE *dict_error;
|
|
- int formatted;
|
|
- int flush;
|
|
+extern int yy_flex_debug;
|
|
+lst_List dict_Servers;
|
|
+FILE *dict_output;
|
|
+FILE *dict_error;
|
|
+int formatted;
|
|
+int flush;
|
|
|
|
-const char *host_connected = NULL;
|
|
+const char *host_connected = NULL;
|
|
const char *service_connected = NULL;
|
|
|
|
#define BUFFERSIZE 2048
|
|
@@ -56,33 +56,33 @@ const char *service_connected = NULL;
|
|
#define CMD_OPTION_MIME 15
|
|
|
|
struct cmd {
|
|
- int command;
|
|
- int sent;
|
|
- int flag;
|
|
- const char *host;
|
|
- const char *service;
|
|
- const char *database;
|
|
- const char *strategy;
|
|
- const char *word;
|
|
- const char *client;
|
|
- const char *user;
|
|
- const char *key;
|
|
- const char *comment;
|
|
+ int command;
|
|
+ int sent;
|
|
+ int flag;
|
|
+ const char *host;
|
|
+ const char *service;
|
|
+ const char *database;
|
|
+ const char *strategy;
|
|
+ const char *word;
|
|
+ const char *client;
|
|
+ const char *user;
|
|
+ const char *key;
|
|
+ const char *comment;
|
|
};
|
|
|
|
-lst_List cmd_list;
|
|
+lst_List cmd_list;
|
|
unsigned long client_defines;
|
|
unsigned long client_bytes;
|
|
unsigned long client_pipesize = PIPESIZE;
|
|
-char *client_text = NULL;
|
|
+char *client_text = NULL;
|
|
|
|
int option_mime = 0;
|
|
|
|
int ex_status = 0;
|
|
-static void set_ex_status (int status)
|
|
+static void set_ex_status(int status)
|
|
{
|
|
- if (!ex_status)
|
|
- ex_status = status;
|
|
+ if (!ex_status)
|
|
+ ex_status = status;
|
|
}
|
|
|
|
#define EXST_NO_MATCH 20
|
|
@@ -104,1364 +104,1516 @@ static void set_ex_status (int status)
|
|
#define EXST_CONNECTION_FAILED 41
|
|
|
|
struct def {
|
|
- lst_List data;
|
|
- const char *word;
|
|
- const char *db;
|
|
- const char *dbname;
|
|
+ lst_List data;
|
|
+ const char *word;
|
|
+ const char *db;
|
|
+ const char *dbname;
|
|
};
|
|
|
|
struct reply {
|
|
- int s;
|
|
- const char *host;
|
|
- const char *service;
|
|
- const char *user;
|
|
- const char *key;
|
|
- const char *msgid;
|
|
- const char *word;
|
|
- lst_List data;
|
|
- int retcode;
|
|
- int count; /* definitions found */
|
|
- int matches; /* matches found */
|
|
- int match; /* doing match found */
|
|
- int listed; /* Databases or strategies listed */
|
|
- struct def *defs;
|
|
+ int s;
|
|
+ const char *host;
|
|
+ const char *service;
|
|
+ const char *user;
|
|
+ const char *key;
|
|
+ const char *msgid;
|
|
+ const char *word;
|
|
+ lst_List data;
|
|
+ int retcode;
|
|
+ int count; /* definitions found */
|
|
+ int matches; /* matches found */
|
|
+ int match; /* doing match found */
|
|
+ int listed; /* Databases or strategies listed */
|
|
+ struct def *defs;
|
|
} cmd_reply;
|
|
|
|
-static const char *cpy( const char *s )
|
|
+static const char *cpy(const char *s)
|
|
{
|
|
- if (!s || !*s) return NULL;
|
|
- return str_copy(s);
|
|
+ if (!s || !*s)
|
|
+ return NULL;
|
|
+ return str_copy(s);
|
|
}
|
|
|
|
-static void client_crlf( char *d, const char *s )
|
|
+static void client_crlf(char *d, const char *s)
|
|
{
|
|
- int flag = 0;
|
|
-
|
|
- while (*s) {
|
|
- if (*s == '\n') {
|
|
- *d++ = '\r';
|
|
- *d++ = '\n';
|
|
- ++s;
|
|
- ++flag;
|
|
- } else {
|
|
- *d++ = *s++;
|
|
- flag = 0;
|
|
- }
|
|
- }
|
|
- if (!flag) {
|
|
- *d++ = '\r';
|
|
- *d++ = '\n';
|
|
- }
|
|
- *d = '\0';
|
|
-}
|
|
-
|
|
-static void unexpected_status_code (
|
|
- int expected_status, int act_status,
|
|
- const char *message)
|
|
-{
|
|
- switch (act_status){
|
|
- case CODE_TEMPORARILY_UNAVAILABLE:
|
|
- fprintf (stderr,
|
|
- "%s\n",
|
|
- (message ? message : "Server temporarily unavailable"));
|
|
- exit (EXST_TEMPORARILY_UNAVAILABLE);
|
|
-
|
|
- case CODE_SHUTTING_DOWN:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Server shutting down"));
|
|
- exit (EXST_SHUTTING_DOWN);
|
|
-
|
|
- case CODE_SYNTAX_ERROR:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Command not recognized"));
|
|
- exit (EXST_SYNTAX_ERROR);
|
|
-
|
|
- case CODE_ILLEGAL_PARAM:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Illegal parameters"));
|
|
- exit (EXST_ILLEGAL_PARAM);
|
|
-
|
|
- case CODE_COMMAND_NOT_IMPLEMENTED:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Command not implemented"));
|
|
- exit (EXST_COMMAND_NOT_IMPLEMENTED);
|
|
-
|
|
- case CODE_PARAM_NOT_IMPLEMENTED:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Parameter not implemented"));
|
|
- exit (EXST_PARAM_NOT_IMPLEMENTED);
|
|
-
|
|
- case CODE_ACCESS_DENIED:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Access denied"));
|
|
- exit (EXST_ACCESS_DENIED);
|
|
-
|
|
- case CODE_AUTH_DENIED:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Authentication denied"));
|
|
- exit (EXST_AUTH_DENIED);
|
|
-
|
|
- case CODE_INVALID_DB:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Invalid database"));
|
|
- exit (EXST_INVALID_DB);
|
|
-
|
|
- case CODE_INVALID_STRATEGY:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "Invalid strategy"));
|
|
- exit (EXST_INVALID_STRATEGY);
|
|
-
|
|
- case CODE_NO_MATCH:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "No matches found"));
|
|
- exit (EXST_NO_MATCH);
|
|
-
|
|
- case CODE_NO_DATABASES:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "No databases"));
|
|
- exit (EXST_NO_DATABASES);
|
|
-
|
|
- case CODE_NO_STRATEGIES:
|
|
- fprintf (stderr,
|
|
- "%s\n", (message ? message : "No strategies"));
|
|
- exit (EXST_NO_STRATEGIES);
|
|
-
|
|
- default:
|
|
- fprintf (stderr,
|
|
- "Unexpected status code %d (%s), wanted %d\n",
|
|
- act_status,
|
|
- (message ? message : "no message"),
|
|
- expected_status);
|
|
-
|
|
- exit (EXST_UNEXPECTED);
|
|
- }
|
|
-}
|
|
-
|
|
-static void client_open_pager( void )
|
|
-{
|
|
- dict_output = stdout;
|
|
- dict_error = stderr;
|
|
-}
|
|
-
|
|
-static void client_close_pager( void )
|
|
-{
|
|
- fflush (dict_output);
|
|
-}
|
|
-
|
|
-static lst_List client_read_text( int s )
|
|
-{
|
|
- lst_List l = lst_create();
|
|
- char line[BUFFERSIZE];
|
|
- int len;
|
|
-
|
|
- while ((len = net_read(s, line, BUFFERSIZE - 1)) >= 0) {
|
|
- line [len] = 0;
|
|
-
|
|
- client_bytes += len;
|
|
- PRINTF(DBG_RAW,("* Text: %s\n",line));
|
|
- if (line[0] == '.' && line[1] == '\0') break;
|
|
- if (len >= 2 && line[0] == '.' && line[1] == '.')
|
|
- lst_append( l, xstrdup(line + 1) );
|
|
- else
|
|
- lst_append( l, xstrdup(line) );
|
|
- }
|
|
- if (len < 0) {
|
|
- client_close_pager();
|
|
- err_fatal_errno( __func__, "Error reading from socket\n" );
|
|
- }
|
|
- return l;
|
|
-}
|
|
-
|
|
-static void client_print_text( lst_List l, int print_host_port )
|
|
-{
|
|
- lst_Position p;
|
|
- const char *e;
|
|
-
|
|
- if (!l) return;
|
|
-
|
|
- if (formatted && print_host_port){
|
|
- fprintf (dict_output, "%s\t%s\n", host_connected, service_connected);
|
|
- }
|
|
-
|
|
- LST_ITERATE(l,p,e) fprintf( dict_output, " %s\n", e );
|
|
-}
|
|
-
|
|
-static void client_print_definitions (const struct def *r)
|
|
-{
|
|
- if (formatted){
|
|
- fprintf (dict_output, "%s\t%s\t", host_connected, service_connected);
|
|
-
|
|
- if (r -> dbname && r -> db)
|
|
- {
|
|
- fprintf( dict_output, "%s\t%s\n", r -> db, r -> dbname);
|
|
- } else if (r -> dbname) {
|
|
- fprintf( dict_output, "%s\n", r -> dbname );
|
|
- } else if (r -> db) {
|
|
- fprintf( dict_output, "%s\n", r -> db );
|
|
- } else {
|
|
- fprintf( dict_output, "unknown\n" );
|
|
- }
|
|
- }else{
|
|
- fprintf (dict_output, "\nFrom ");
|
|
- if (r -> dbname && r -> db)
|
|
- {
|
|
- fprintf( dict_output, "%s [%s]",
|
|
- r -> dbname,
|
|
- r -> db);
|
|
- } else if (r -> dbname) {
|
|
- fprintf( dict_output, "%s", r -> dbname );
|
|
- } else if (r -> db) {
|
|
- fprintf( dict_output, "%s", r -> db );
|
|
- } else {
|
|
- fprintf( dict_output, "unknown" );
|
|
- }
|
|
-
|
|
- fprintf( dict_output, ":\n\n" );
|
|
- }
|
|
-
|
|
- client_print_text( r -> data, 0 );
|
|
-}
|
|
-
|
|
-static void client_print_matches( lst_List l, int flag, const char *word )
|
|
-{
|
|
- lst_Position p;
|
|
- const char *e;
|
|
- arg_List a;
|
|
- const char *prev = NULL;
|
|
- const char *last;
|
|
- const char *db;
|
|
- static int first = 1;
|
|
- int pos = 0;
|
|
- int len;
|
|
- int count;
|
|
- int empty_line_found = 0;
|
|
-
|
|
- const char *arg0 = NULL;
|
|
- const char *arg1 = NULL;
|
|
-
|
|
- count = 0;
|
|
- if (l) {
|
|
- last = NULL;
|
|
- LST_ITERATE(l,p,e) {
|
|
- if (last && !strcmp(last,e)) continue;
|
|
- ++count;
|
|
- last = e;
|
|
- }
|
|
- } else {
|
|
- if (flag)
|
|
- fprintf( dict_error, "No matches found for \"%s\"\n", word );
|
|
- set_ex_status (EXST_NO_MATCH);
|
|
- return;
|
|
- }
|
|
-
|
|
- last = NULL;
|
|
- LST_ITERATE(l,p,e) {
|
|
- /* skip MIME header */
|
|
- if (option_mime && !empty_line_found){
|
|
- empty_line_found = (e [0] == 0 ||
|
|
- (e [0] == '\r' && e [1] == 0));
|
|
- continue;
|
|
- }
|
|
-
|
|
- /* */
|
|
- if (last && !strcmp(last,e)) continue;
|
|
- last = e;
|
|
- a = arg_argify( e, 0 );
|
|
- if (arg_count(a) != 2)
|
|
- err_internal( __func__,
|
|
- "MATCH command didn't return 2 args: \"%s\"\n", e );
|
|
-
|
|
- arg0 = arg_get (a,0);
|
|
- arg1 = arg_get (a,1);
|
|
-
|
|
- if (formatted){
|
|
- fprintf (dict_output, "%s\t%s\t%s\t%s\n",
|
|
- host_connected, service_connected, arg0, arg1);
|
|
- }else{
|
|
- if ((db = str_find(arg0)) != prev) {
|
|
- if (!first) fprintf( dict_output, "\n" );
|
|
- first = 0;
|
|
- fprintf( dict_output, "%s:", db );
|
|
- prev = db;
|
|
- pos = 6 + strlen(db);
|
|
- }
|
|
- len = strlen(arg1);
|
|
- if (pos + len + 4 > 70) {
|
|
- fprintf( dict_output, "\n" );
|
|
- pos = 0;
|
|
- }
|
|
- if (strchr (arg1,' ')) {
|
|
- fprintf( dict_output, " \"%s\"", arg1 );
|
|
- pos += len + 4;
|
|
- } else {
|
|
- fprintf( dict_output, " %s", arg1 );
|
|
- pos += len + 2;
|
|
- }
|
|
- }
|
|
-
|
|
- arg_destroy(a);
|
|
- }
|
|
- fprintf( dict_output, "\n" );
|
|
-}
|
|
-
|
|
-static void client_print_listed( lst_List l )
|
|
-{
|
|
- lst_Position p;
|
|
- const char *e;
|
|
- arg_List a;
|
|
- int colWidth = 10; /* minimum size of first column */
|
|
- int colMax = 16; /* maximum size of unragged first column */
|
|
- char format[32];
|
|
- int empty_line_found = 0;
|
|
- int len;
|
|
-
|
|
- if (!l) return;
|
|
- LST_ITERATE(l,p,e) {
|
|
- /* skip MIME header */
|
|
- if (option_mime && !empty_line_found){
|
|
- empty_line_found = (e [0] == 0 ||
|
|
- (e [0] == '\r' && e [1] == 0));
|
|
- continue;
|
|
- }
|
|
-
|
|
- /* */
|
|
- a = arg_argify( e, 0 );
|
|
- if (arg_count(a) != 2)
|
|
- err_internal( __func__,
|
|
- "SHOW command didn't return 2 args: \"%s\"\n", e );
|
|
-
|
|
- len = strlen (arg_get (a,0));
|
|
-
|
|
- if (len > colWidth && len <= colMax)
|
|
- colWidth = len;
|
|
-
|
|
- arg_destroy(a);
|
|
- }
|
|
-
|
|
- snprintf( format, sizeof (format), " %%-%ds %%s\n", colWidth );
|
|
-
|
|
- empty_line_found = 0;
|
|
-
|
|
- LST_ITERATE(l,p,e) {
|
|
- /* skip MIME header */
|
|
- if (option_mime && !empty_line_found){
|
|
- empty_line_found = (e [0] == 0 ||
|
|
- (e [0] == '\r' && e [1] == 0));
|
|
- continue;
|
|
- }
|
|
-
|
|
- /* */
|
|
- a = arg_argify( e, 0 );
|
|
-
|
|
- if (formatted){
|
|
- assert (host_connected);
|
|
- assert (service_connected);
|
|
-
|
|
- fprintf (dict_output, "%s\t%s\t%s\t%s\n",
|
|
- host_connected, service_connected, arg_get (a,0), arg_get (a,1));
|
|
- }else{
|
|
- fprintf (dict_output, format, arg_get (a,0), arg_get (a,1));
|
|
- }
|
|
-
|
|
- arg_destroy(a);
|
|
- }
|
|
-}
|
|
-
|
|
-static void client_free_text( lst_List l )
|
|
-{
|
|
- lst_Position p;
|
|
- char *e;
|
|
-
|
|
- if (!l) return;
|
|
- LST_ITERATE(l,p,e) {
|
|
- if (e) xfree(e);
|
|
- }
|
|
- lst_destroy(l);
|
|
-}
|
|
-
|
|
-static int client_read_status( int s,
|
|
- const char **message,
|
|
- int *count,
|
|
- const char **word,
|
|
- const char **db,
|
|
- const char **dbname,
|
|
- const char **msgid )
|
|
-{
|
|
- static char buf[BUFFERSIZE];
|
|
- arg_List cmdline;
|
|
- int argc;
|
|
- char **argv;
|
|
- int status;
|
|
- char *start, *end, *p;
|
|
- int len;
|
|
-
|
|
- if ((len = net_read( s, buf, BUFFERSIZE )) < 0) {
|
|
- client_close_pager();
|
|
- err_fatal_errno( __func__, "Error reading from socket\n" );
|
|
- }
|
|
- client_bytes += len;
|
|
- PRINTF(DBG_RAW,("* Read: %s\n",buf));
|
|
-
|
|
- if ((status = atoi(buf)) < 100) status = 600;
|
|
- PRINTF(DBG_RAW,("* Status = %d\n",status));
|
|
-
|
|
- if (message && (p = strchr(buf, ' '))) *message = p + 1;
|
|
-
|
|
- if (count) *count = 0;
|
|
- if (word) *word = NULL;
|
|
- if (db) *db = NULL;
|
|
- if (dbname) *dbname = NULL;
|
|
- if (msgid) *msgid = NULL;
|
|
-
|
|
- switch (status) {
|
|
- case CODE_HELLO:
|
|
- if ((start = strrchr(buf, '<')) && (end = strrchr(buf,'>'))) {
|
|
- end[1] = '\0';
|
|
- *msgid = str_copy( start );
|
|
- PRINTF(DBG_VERBOSE,("Msgid is \"%s\"\n",*msgid));
|
|
- }
|
|
- break;
|
|
- case CODE_DATABASE_LIST:
|
|
- case CODE_STRATEGY_LIST:
|
|
- case CODE_DEFINITIONS_FOUND:
|
|
- case CODE_MATCHES_FOUND:
|
|
- cmdline = arg_argify(buf,0);
|
|
- arg_get_vector( cmdline, &argc, &argv );
|
|
- if (argc > 1 && count) *count = atoi(argv[1]);
|
|
- arg_destroy(cmdline);
|
|
- break;
|
|
- case CODE_DEFINITION_FOLLOWS:
|
|
- cmdline = arg_argify(buf,0);
|
|
- arg_get_vector( cmdline, &argc, &argv );
|
|
- if (argc > 1 && word) *word = str_find(argv[1]);
|
|
- if (argc > 2 && db) *db = str_find(argv[2]);
|
|
- if (argc > 3 && dbname) *dbname = str_find(argv[3]);
|
|
- arg_destroy(cmdline);
|
|
- break;
|
|
- default:
|
|
- break;
|
|
- }
|
|
-
|
|
- return status;
|
|
-}
|
|
-
|
|
-static struct cmd *make_command( int command, ... )
|
|
-{
|
|
- va_list ap;
|
|
- struct cmd *c = xmalloc( sizeof( struct cmd ) );
|
|
-
|
|
- memset( c, 0, sizeof( struct cmd ) );
|
|
- c->command = command;
|
|
-
|
|
- va_start( ap, command );
|
|
- switch (command) {
|
|
- case CMD_PRINT:
|
|
- c->comment = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_DEFPRINT:
|
|
- c->database = va_arg( ap, const char * );
|
|
- c->word = va_arg( ap, const char * );
|
|
- c->flag = va_arg( ap, int );
|
|
- break;
|
|
- case CMD_CONNECT:
|
|
- c->host = va_arg( ap, const char * );
|
|
- c->service = va_arg( ap, const char * );
|
|
- c->user = va_arg( ap, const char * );
|
|
- c->key = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_CLIENT:
|
|
- c->client = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_AUTH:
|
|
- break;
|
|
- case CMD_INFO:
|
|
- c->database = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_SERVER:
|
|
- break;
|
|
- case CMD_DBS:
|
|
- break;
|
|
- case CMD_STRATS:
|
|
- break;
|
|
- case CMD_HELP:
|
|
- break;
|
|
- case CMD_MATCH:
|
|
- c->database = va_arg( ap, const char * );
|
|
- c->strategy = va_arg( ap, const char * );
|
|
- c->word = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_DEFINE:
|
|
- c->database = va_arg( ap, const char * );
|
|
- c->word = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_SPELL:
|
|
- c->database = va_arg( ap, const char * );
|
|
- c->word = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_WIND:
|
|
- c->database = va_arg( ap, const char * );
|
|
- c->strategy = va_arg( ap, const char * );
|
|
- c->word = va_arg( ap, const char * );
|
|
- break;
|
|
- case CMD_CLOSE:
|
|
- break;
|
|
- case CMD_OPTION_MIME:
|
|
- break;
|
|
- default:
|
|
- err_internal( __func__, "Illegal command %d\n", command );
|
|
- }
|
|
- va_end( ap );
|
|
-
|
|
- return c;
|
|
-}
|
|
-
|
|
-static void append_command( struct cmd *c )
|
|
-{
|
|
- if (!cmd_list) cmd_list = lst_create();
|
|
- lst_append( cmd_list, c );
|
|
-}
|
|
-
|
|
-
|
|
-static void prepend_command( struct cmd *c )
|
|
-{
|
|
- if (!cmd_list) cmd_list = lst_create();
|
|
- lst_push( cmd_list, c );
|
|
-}
|
|
-
|
|
-
|
|
-static void request( void )
|
|
-{
|
|
- char b[BUFFERSIZE];
|
|
- char *buffer = alloca( client_pipesize );
|
|
- char *p = buffer;
|
|
- lst_Position pos;
|
|
- struct cmd *c = NULL;
|
|
- unsigned char digest[16];
|
|
- char hex[33];
|
|
- struct MD5Context ctx;
|
|
- int i;
|
|
- unsigned len;
|
|
- int total = 0;
|
|
- int count = 0;
|
|
-
|
|
- *p = '\0';
|
|
- c = lst_top(cmd_list);
|
|
- if (c->command == CMD_CONNECT) {
|
|
- cmd_reply.user = c->user;
|
|
- cmd_reply.key = c->key;
|
|
- }
|
|
-
|
|
- LST_ITERATE(cmd_list,pos,c) {
|
|
- b[0] = '\0';
|
|
- len = 0;
|
|
- PRINTF(DBG_PIPE,("* Looking at request %d\n",c->command));
|
|
- if (c->sent) {
|
|
- PRINTF(DBG_PIPE,("* Skipping\n"));
|
|
- return; /* FIXME! Keep sending deeper things? */
|
|
- }
|
|
- ++count;
|
|
- switch( c->command) {
|
|
- case CMD_PRINT: break;
|
|
- case CMD_DEFPRINT: break;
|
|
- case CMD_CONNECT: break;
|
|
- case CMD_AUTH:
|
|
- if (!cmd_reply.key || !cmd_reply.user) break;
|
|
- if (!cmd_reply.msgid) goto end;
|
|
- MD5Init(&ctx);
|
|
- MD5Update(&ctx, (const unsigned char *) cmd_reply.msgid, strlen(cmd_reply.msgid));
|
|
- MD5Update(&ctx, (const unsigned char *) cmd_reply.key, strlen(cmd_reply.key));
|
|
- MD5Final(digest, &ctx );
|
|
- for (i = 0; i < 16; i++)
|
|
- sprintf( hex+2*i, "%02x", digest[i] );
|
|
- hex[32] = '\0';
|
|
- snprintf( b, BUFFERSIZE, "auth %s %s\n", cmd_reply.user, hex );
|
|
- break;
|
|
- case CMD_CLIENT:
|
|
- if (client_text)
|
|
- snprintf( b, BUFFERSIZE, "client \"%s: %s\"\n",
|
|
- c->client, client_text );
|
|
- else
|
|
- snprintf( b, BUFFERSIZE, "client \"%s\"\n", c->client );
|
|
- break;
|
|
- case CMD_INFO: snprintf( b, BUFFERSIZE, "show info %s\n",
|
|
- c->database ); break;
|
|
- case CMD_SERVER: snprintf( b, BUFFERSIZE, "show server\n" ); break;
|
|
- case CMD_DBS: snprintf( b, BUFFERSIZE, "show db\n" ); break;
|
|
- case CMD_STRATS: snprintf( b, BUFFERSIZE, "show strat\n" ); break;
|
|
- case CMD_HELP: snprintf( b, BUFFERSIZE, "help\n" ); break;
|
|
- case CMD_MATCH:
|
|
- cmd_reply.word = c->word;
|
|
- snprintf( b, BUFFERSIZE,
|
|
- "match %s %s \"%s\"\n",
|
|
- c->database, c->strategy, c->word ); break;
|
|
- case CMD_DEFINE:
|
|
- cmd_reply.word = c->word;
|
|
- snprintf( b, BUFFERSIZE, "define %s \"%s\"\n",
|
|
- c->database, c->word ); break;
|
|
- case CMD_SPELL: goto end;
|
|
- case CMD_WIND: goto end;
|
|
- case CMD_CLOSE: snprintf( b, BUFFERSIZE, "quit\n" ); break;
|
|
- case CMD_OPTION_MIME: snprintf( b, BUFFERSIZE, "option mime\n" ); break;
|
|
- default:
|
|
- err_internal( __func__, "Unknown command %d\n", c->command );
|
|
- }
|
|
- len = strlen(b);
|
|
- if (total + len + 3 > client_pipesize) {
|
|
- if (count == 1 && p == buffer && total == 0) {
|
|
- /* The buffer is too small, but we have to
|
|
- send something... */
|
|
- PRINTF(DBG_PIPE,("* Reallocating buffer to %d bytes\n",len+1));
|
|
- p = buffer = alloca( len + 1 );
|
|
- } else {
|
|
- break;
|
|
- }
|
|
- }
|
|
- strcpy( p, b );
|
|
- p += len;
|
|
- total += len;
|
|
- ++c->sent;
|
|
- if (dbg_test(DBG_SERIAL)) break; /* Don't pipeline. */
|
|
- }
|
|
+ int flag = 0;
|
|
|
|
-end: /* Ready to send buffer, but are we
|
|
+ while (*s) {
|
|
+ if (*s == '\n') {
|
|
+ *d++ = '\r';
|
|
+ *d++ = '\n';
|
|
+ ++s;
|
|
+ ++flag;
|
|
+ } else {
|
|
+ *d++ = *s++;
|
|
+ flag = 0;
|
|
+ }
|
|
+ }
|
|
+ if (!flag) {
|
|
+ *d++ = '\r';
|
|
+ *d++ = '\n';
|
|
+ }
|
|
+ *d = '\0';
|
|
+}
|
|
+
|
|
+static void unexpected_status_code(int expected_status, int act_status,
|
|
+ const char *message)
|
|
+{
|
|
+ switch (act_status) {
|
|
+ case CODE_TEMPORARILY_UNAVAILABLE:
|
|
+ fprintf(stderr,
|
|
+ "%s\n",
|
|
+ (message ? message : "Server temporarily unavailable"));
|
|
+ exit(EXST_TEMPORARILY_UNAVAILABLE);
|
|
+
|
|
+ case CODE_SHUTTING_DOWN:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Server shutting down"));
|
|
+ exit(EXST_SHUTTING_DOWN);
|
|
+
|
|
+ case CODE_SYNTAX_ERROR:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Command not recognized"));
|
|
+ exit(EXST_SYNTAX_ERROR);
|
|
+
|
|
+ case CODE_ILLEGAL_PARAM:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Illegal parameters"));
|
|
+ exit(EXST_ILLEGAL_PARAM);
|
|
+
|
|
+ case CODE_COMMAND_NOT_IMPLEMENTED:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Command not implemented"));
|
|
+ exit(EXST_COMMAND_NOT_IMPLEMENTED);
|
|
+
|
|
+ case CODE_PARAM_NOT_IMPLEMENTED:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Parameter not implemented"));
|
|
+ exit(EXST_PARAM_NOT_IMPLEMENTED);
|
|
+
|
|
+ case CODE_ACCESS_DENIED:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "Access denied"));
|
|
+ exit(EXST_ACCESS_DENIED);
|
|
+
|
|
+ case CODE_AUTH_DENIED:
|
|
+ fprintf(stderr,
|
|
+ "%s\n", (message ? message : "Authentication denied"));
|
|
+ exit(EXST_AUTH_DENIED);
|
|
+
|
|
+ case CODE_INVALID_DB:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "Invalid database"));
|
|
+ exit(EXST_INVALID_DB);
|
|
+
|
|
+ case CODE_INVALID_STRATEGY:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "Invalid strategy"));
|
|
+ exit(EXST_INVALID_STRATEGY);
|
|
+
|
|
+ case CODE_NO_MATCH:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "No matches found"));
|
|
+ exit(EXST_NO_MATCH);
|
|
+
|
|
+ case CODE_NO_DATABASES:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "No databases"));
|
|
+ exit(EXST_NO_DATABASES);
|
|
+
|
|
+ case CODE_NO_STRATEGIES:
|
|
+ fprintf(stderr, "%s\n", (message ? message : "No strategies"));
|
|
+ exit(EXST_NO_STRATEGIES);
|
|
+
|
|
+ default:
|
|
+ fprintf(stderr,
|
|
+ "Unexpected status code %d (%s), wanted %d\n",
|
|
+ act_status,
|
|
+ (message ? message : "no message"), expected_status);
|
|
+
|
|
+ exit(EXST_UNEXPECTED);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void client_open_pager(void)
|
|
+{
|
|
+ dict_output = stdout;
|
|
+ dict_error = stderr;
|
|
+}
|
|
+
|
|
+static void client_close_pager(void)
|
|
+{
|
|
+ fflush(dict_output);
|
|
+}
|
|
+
|
|
+static lst_List client_read_text(int s)
|
|
+{
|
|
+ lst_List l = lst_create();
|
|
+ char line[BUFFERSIZE];
|
|
+ int len;
|
|
+
|
|
+ while ((len = net_read(s, line, BUFFERSIZE - 1)) >= 0) {
|
|
+ line[len] = 0;
|
|
+
|
|
+ client_bytes += len;
|
|
+ PRINTF(DBG_RAW, ("* Text: %s\n", line));
|
|
+ if (line[0] == '.' && line[1] == '\0')
|
|
+ break;
|
|
+ if (len >= 2 && line[0] == '.' && line[1] == '.')
|
|
+ lst_append(l, xstrdup(line + 1));
|
|
+ else
|
|
+ lst_append(l, xstrdup(line));
|
|
+ }
|
|
+ if (len < 0) {
|
|
+ client_close_pager();
|
|
+ err_fatal_errno(__func__, "Error reading from socket\n");
|
|
+ }
|
|
+ return l;
|
|
+}
|
|
+
|
|
+static void client_print_text(lst_List l, int print_host_port)
|
|
+{
|
|
+ lst_Position p;
|
|
+ const char *e;
|
|
+
|
|
+ if (!l)
|
|
+ return;
|
|
+
|
|
+ if (formatted && print_host_port) {
|
|
+ fprintf(dict_output, "%s\t%s\n", host_connected,
|
|
+ service_connected);
|
|
+ }
|
|
+
|
|
+ LST_ITERATE(l, p, e) fprintf(dict_output, " %s\n", e);
|
|
+}
|
|
+
|
|
+static void client_print_definitions(const struct def *r)
|
|
+{
|
|
+ if (formatted) {
|
|
+ fprintf(dict_output, "%s\t%s\t", host_connected,
|
|
+ service_connected);
|
|
+
|
|
+ if (r->dbname && r->db) {
|
|
+ fprintf(dict_output, "%s\t%s\n", r->db, r->dbname);
|
|
+ } else if (r->dbname) {
|
|
+ fprintf(dict_output, "%s\n", r->dbname);
|
|
+ } else if (r->db) {
|
|
+ fprintf(dict_output, "%s\n", r->db);
|
|
+ } else {
|
|
+ fprintf(dict_output, "unknown\n");
|
|
+ }
|
|
+ } else {
|
|
+ fprintf(dict_output, "\nFrom ");
|
|
+ if (r->dbname && r->db) {
|
|
+ fprintf(dict_output, "%s [%s]", r->dbname, r->db);
|
|
+ } else if (r->dbname) {
|
|
+ fprintf(dict_output, "%s", r->dbname);
|
|
+ } else if (r->db) {
|
|
+ fprintf(dict_output, "%s", r->db);
|
|
+ } else {
|
|
+ fprintf(dict_output, "unknown");
|
|
+ }
|
|
+
|
|
+ fprintf(dict_output, ":\n\n");
|
|
+ }
|
|
+
|
|
+ client_print_text(r->data, 0);
|
|
+}
|
|
+
|
|
+static void client_print_matches(lst_List l, int flag, const char *word)
|
|
+{
|
|
+ lst_Position p;
|
|
+ const char *e;
|
|
+ arg_List a;
|
|
+ const char *prev = NULL;
|
|
+ const char *last;
|
|
+ const char *db;
|
|
+ static int first = 1;
|
|
+ int pos = 0;
|
|
+ int len;
|
|
+ int count;
|
|
+ int empty_line_found = 0;
|
|
+
|
|
+ const char *arg0 = NULL;
|
|
+ const char *arg1 = NULL;
|
|
+
|
|
+ count = 0;
|
|
+ if (l) {
|
|
+ last = NULL;
|
|
+ LST_ITERATE(l, p, e) {
|
|
+ if (last && !strcmp(last, e))
|
|
+ continue;
|
|
+ ++count;
|
|
+ last = e;
|
|
+ }
|
|
+ } else {
|
|
+ if (flag)
|
|
+ fprintf(dict_error, "No matches found for \"%s\"\n", word);
|
|
+ set_ex_status(EXST_NO_MATCH);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ last = NULL;
|
|
+ LST_ITERATE(l, p, e) {
|
|
+ /* skip MIME header */
|
|
+ if (option_mime && !empty_line_found) {
|
|
+ empty_line_found = (e[0] == 0 || (e[0] == '\r' && e[1] == 0));
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ /* */
|
|
+ if (last && !strcmp(last, e))
|
|
+ continue;
|
|
+ last = e;
|
|
+ a = arg_argify(e, 0);
|
|
+ if (arg_count(a) != 2)
|
|
+ err_internal(__func__,
|
|
+ "MATCH command didn't return 2 args: \"%s\"\n",
|
|
+ e);
|
|
+
|
|
+ arg0 = arg_get(a, 0);
|
|
+ arg1 = arg_get(a, 1);
|
|
+
|
|
+ if (formatted) {
|
|
+ fprintf(dict_output, "%s\t%s\t%s\t%s\n",
|
|
+ host_connected, service_connected, arg0, arg1);
|
|
+ } else {
|
|
+ if ((db = str_find(arg0)) != prev) {
|
|
+ if (!first)
|
|
+ fprintf(dict_output, "\n");
|
|
+ first = 0;
|
|
+ fprintf(dict_output, "%s:", db);
|
|
+ prev = db;
|
|
+ pos = 6 + strlen(db);
|
|
+ }
|
|
+ len = strlen(arg1);
|
|
+ if (pos + len + 4 > 70) {
|
|
+ fprintf(dict_output, "\n");
|
|
+ pos = 0;
|
|
+ }
|
|
+ if (strchr(arg1, ' ')) {
|
|
+ fprintf(dict_output, " \"%s\"", arg1);
|
|
+ pos += len + 4;
|
|
+ } else {
|
|
+ fprintf(dict_output, " %s", arg1);
|
|
+ pos += len + 2;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ arg_destroy(a);
|
|
+ }
|
|
+ fprintf(dict_output, "\n");
|
|
+}
|
|
+
|
|
+static void client_print_listed(lst_List l)
|
|
+{
|
|
+ lst_Position p;
|
|
+ const char *e;
|
|
+ arg_List a;
|
|
+ int colWidth = 10; /* minimum size of first column */
|
|
+ int colMax = 16; /* maximum size of unragged first column */
|
|
+ char format[32];
|
|
+ int empty_line_found = 0;
|
|
+ int len;
|
|
+
|
|
+ if (!l)
|
|
+ return;
|
|
+ LST_ITERATE(l, p, e) {
|
|
+ /* skip MIME header */
|
|
+ if (option_mime && !empty_line_found) {
|
|
+ empty_line_found = (e[0] == 0 || (e[0] == '\r' && e[1] == 0));
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ /* */
|
|
+ a = arg_argify(e, 0);
|
|
+ if (arg_count(a) != 2)
|
|
+ err_internal(__func__,
|
|
+ "SHOW command didn't return 2 args: \"%s\"\n", e);
|
|
+
|
|
+ len = strlen(arg_get(a, 0));
|
|
+
|
|
+ if (len > colWidth && len <= colMax)
|
|
+ colWidth = len;
|
|
+
|
|
+ arg_destroy(a);
|
|
+ }
|
|
+
|
|
+ snprintf(format, sizeof(format), " %%-%ds %%s\n", colWidth);
|
|
+
|
|
+ empty_line_found = 0;
|
|
+
|
|
+ LST_ITERATE(l, p, e) {
|
|
+ /* skip MIME header */
|
|
+ if (option_mime && !empty_line_found) {
|
|
+ empty_line_found = (e[0] == 0 || (e[0] == '\r' && e[1] == 0));
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ /* */
|
|
+ a = arg_argify(e, 0);
|
|
+
|
|
+ if (formatted) {
|
|
+ assert(host_connected);
|
|
+ assert(service_connected);
|
|
+
|
|
+ fprintf(dict_output, "%s\t%s\t%s\t%s\n",
|
|
+ host_connected, service_connected, arg_get(a, 0),
|
|
+ arg_get(a, 1));
|
|
+ } else {
|
|
+ fprintf(dict_output, format, arg_get(a, 0), arg_get(a, 1));
|
|
+ }
|
|
+
|
|
+ arg_destroy(a);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void client_free_text(lst_List l)
|
|
+{
|
|
+ lst_Position p;
|
|
+ char *e;
|
|
+
|
|
+ if (!l)
|
|
+ return;
|
|
+ LST_ITERATE(l, p, e) {
|
|
+ if (e)
|
|
+ xfree(e);
|
|
+ }
|
|
+ lst_destroy(l);
|
|
+}
|
|
+
|
|
+static int client_read_status(int s,
|
|
+ const char **message,
|
|
+ int *count,
|
|
+ const char **word,
|
|
+ const char **db,
|
|
+ const char **dbname, const char **msgid)
|
|
+{
|
|
+ static char buf[BUFFERSIZE];
|
|
+ arg_List cmdline;
|
|
+ int argc;
|
|
+ char **argv;
|
|
+ int status;
|
|
+ char *start, *end, *p;
|
|
+ int len;
|
|
+
|
|
+ if ((len = net_read(s, buf, BUFFERSIZE)) < 0) {
|
|
+ client_close_pager();
|
|
+ err_fatal_errno(__func__, "Error reading from socket\n");
|
|
+ }
|
|
+ client_bytes += len;
|
|
+ PRINTF(DBG_RAW, ("* Read: %s\n", buf));
|
|
+
|
|
+ if ((status = atoi(buf)) < 100)
|
|
+ status = 600;
|
|
+ PRINTF(DBG_RAW, ("* Status = %d\n", status));
|
|
+
|
|
+ if (message && (p = strchr(buf, ' ')))
|
|
+ *message = p + 1;
|
|
+
|
|
+ if (count)
|
|
+ *count = 0;
|
|
+ if (word)
|
|
+ *word = NULL;
|
|
+ if (db)
|
|
+ *db = NULL;
|
|
+ if (dbname)
|
|
+ *dbname = NULL;
|
|
+ if (msgid)
|
|
+ *msgid = NULL;
|
|
+
|
|
+ switch (status) {
|
|
+ case CODE_HELLO:
|
|
+ if ((start = strrchr(buf, '<')) && (end = strrchr(buf, '>'))) {
|
|
+ end[1] = '\0';
|
|
+ *msgid = str_copy(start);
|
|
+ PRINTF(DBG_VERBOSE, ("Msgid is \"%s\"\n", *msgid));
|
|
+ }
|
|
+ break;
|
|
+ case CODE_DATABASE_LIST:
|
|
+ case CODE_STRATEGY_LIST:
|
|
+ case CODE_DEFINITIONS_FOUND:
|
|
+ case CODE_MATCHES_FOUND:
|
|
+ cmdline = arg_argify(buf, 0);
|
|
+ arg_get_vector(cmdline, &argc, &argv);
|
|
+ if (argc > 1 && count)
|
|
+ *count = atoi(argv[1]);
|
|
+ arg_destroy(cmdline);
|
|
+ break;
|
|
+ case CODE_DEFINITION_FOLLOWS:
|
|
+ cmdline = arg_argify(buf, 0);
|
|
+ arg_get_vector(cmdline, &argc, &argv);
|
|
+ if (argc > 1 && word)
|
|
+ *word = str_find(argv[1]);
|
|
+ if (argc > 2 && db)
|
|
+ *db = str_find(argv[2]);
|
|
+ if (argc > 3 && dbname)
|
|
+ *dbname = str_find(argv[3]);
|
|
+ arg_destroy(cmdline);
|
|
+ break;
|
|
+ default:
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return status;
|
|
+}
|
|
+
|
|
+static struct cmd *make_command(int command, ...)
|
|
+{
|
|
+ va_list ap;
|
|
+ struct cmd *c = xmalloc(sizeof(struct cmd));
|
|
+
|
|
+ memset(c, 0, sizeof(struct cmd));
|
|
+ c->command = command;
|
|
+
|
|
+ va_start(ap, command);
|
|
+ switch (command) {
|
|
+ case CMD_PRINT:
|
|
+ c->comment = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_DEFPRINT:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ c->word = va_arg(ap, const char *);
|
|
+ c->flag = va_arg(ap, int);
|
|
+ break;
|
|
+ case CMD_CONNECT:
|
|
+ c->host = va_arg(ap, const char *);
|
|
+ c->service = va_arg(ap, const char *);
|
|
+ c->user = va_arg(ap, const char *);
|
|
+ c->key = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_CLIENT:
|
|
+ c->client = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_AUTH:
|
|
+ break;
|
|
+ case CMD_INFO:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_SERVER:
|
|
+ break;
|
|
+ case CMD_DBS:
|
|
+ break;
|
|
+ case CMD_STRATS:
|
|
+ break;
|
|
+ case CMD_HELP:
|
|
+ break;
|
|
+ case CMD_MATCH:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ c->strategy = va_arg(ap, const char *);
|
|
+ c->word = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_DEFINE:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ c->word = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_SPELL:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ c->word = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_WIND:
|
|
+ c->database = va_arg(ap, const char *);
|
|
+ c->strategy = va_arg(ap, const char *);
|
|
+ c->word = va_arg(ap, const char *);
|
|
+ break;
|
|
+ case CMD_CLOSE:
|
|
+ break;
|
|
+ case CMD_OPTION_MIME:
|
|
+ break;
|
|
+ default:
|
|
+ err_internal(__func__, "Illegal command %d\n", command);
|
|
+ }
|
|
+ va_end(ap);
|
|
+
|
|
+ return c;
|
|
+}
|
|
+
|
|
+static void append_command(struct cmd *c)
|
|
+{
|
|
+ if (!cmd_list)
|
|
+ cmd_list = lst_create();
|
|
+ lst_append(cmd_list, c);
|
|
+}
|
|
+
|
|
+
|
|
+static void prepend_command(struct cmd *c)
|
|
+{
|
|
+ if (!cmd_list)
|
|
+ cmd_list = lst_create();
|
|
+ lst_push(cmd_list, c);
|
|
+}
|
|
+
|
|
+
|
|
+static void request(void)
|
|
+{
|
|
+ char b[BUFFERSIZE];
|
|
+ char *buffer = alloca(client_pipesize);
|
|
+ char *p = buffer;
|
|
+ lst_Position pos;
|
|
+ struct cmd *c = NULL;
|
|
+ unsigned char digest[16];
|
|
+ char hex[33];
|
|
+ struct MD5Context ctx;
|
|
+ int i;
|
|
+ unsigned len;
|
|
+ int total = 0;
|
|
+ int count = 0;
|
|
+
|
|
+ *p = '\0';
|
|
+ c = lst_top(cmd_list);
|
|
+ if (c->command == CMD_CONNECT) {
|
|
+ cmd_reply.user = c->user;
|
|
+ cmd_reply.key = c->key;
|
|
+ }
|
|
+
|
|
+ LST_ITERATE(cmd_list, pos, c) {
|
|
+ b[0] = '\0';
|
|
+ len = 0;
|
|
+ PRINTF(DBG_PIPE, ("* Looking at request %d\n", c->command));
|
|
+ if (c->sent) {
|
|
+ PRINTF(DBG_PIPE, ("* Skipping\n"));
|
|
+ return; /* FIXME! Keep sending deeper things? */
|
|
+ }
|
|
+ ++count;
|
|
+ switch (c->command) {
|
|
+ case CMD_PRINT:
|
|
+ break;
|
|
+ case CMD_DEFPRINT:
|
|
+ break;
|
|
+ case CMD_CONNECT:
|
|
+ break;
|
|
+ case CMD_AUTH:
|
|
+ if (!cmd_reply.key || !cmd_reply.user)
|
|
+ break;
|
|
+ if (!cmd_reply.msgid)
|
|
+ goto end;
|
|
+ MD5Init(&ctx);
|
|
+ MD5Update(&ctx, (const unsigned char *) cmd_reply.msgid,
|
|
+ strlen(cmd_reply.msgid));
|
|
+ MD5Update(&ctx, (const unsigned char *) cmd_reply.key,
|
|
+ strlen(cmd_reply.key));
|
|
+ MD5Final(digest, &ctx);
|
|
+ for (i = 0; i < 16; i++)
|
|
+ sprintf(hex + 2 * i, "%02x", digest[i]);
|
|
+ hex[32] = '\0';
|
|
+ snprintf(b, BUFFERSIZE, "auth %s %s\n", cmd_reply.user, hex);
|
|
+ break;
|
|
+ case CMD_CLIENT:
|
|
+ if (client_text)
|
|
+ snprintf(b, BUFFERSIZE, "client \"%s: %s\"\n",
|
|
+ c->client, client_text);
|
|
+ else
|
|
+ snprintf(b, BUFFERSIZE, "client \"%s\"\n", c->client);
|
|
+ break;
|
|
+ case CMD_INFO:
|
|
+ snprintf(b, BUFFERSIZE, "show info %s\n", c->database);
|
|
+ break;
|
|
+ case CMD_SERVER:
|
|
+ snprintf(b, BUFFERSIZE, "show server\n");
|
|
+ break;
|
|
+ case CMD_DBS:
|
|
+ snprintf(b, BUFFERSIZE, "show db\n");
|
|
+ break;
|
|
+ case CMD_STRATS:
|
|
+ snprintf(b, BUFFERSIZE, "show strat\n");
|
|
+ break;
|
|
+ case CMD_HELP:
|
|
+ snprintf(b, BUFFERSIZE, "help\n");
|
|
+ break;
|
|
+ case CMD_MATCH:
|
|
+ cmd_reply.word = c->word;
|
|
+ snprintf(b, BUFFERSIZE,
|
|
+ "match %s %s \"%s\"\n",
|
|
+ c->database, c->strategy, c->word);
|
|
+ break;
|
|
+ case CMD_DEFINE:
|
|
+ cmd_reply.word = c->word;
|
|
+ snprintf(b, BUFFERSIZE, "define %s \"%s\"\n",
|
|
+ c->database, c->word);
|
|
+ break;
|
|
+ case CMD_SPELL:
|
|
+ goto end;
|
|
+ case CMD_WIND:
|
|
+ goto end;
|
|
+ case CMD_CLOSE:
|
|
+ snprintf(b, BUFFERSIZE, "quit\n");
|
|
+ break;
|
|
+ case CMD_OPTION_MIME:
|
|
+ snprintf(b, BUFFERSIZE, "option mime\n");
|
|
+ break;
|
|
+ default:
|
|
+ err_internal(__func__, "Unknown command %d\n", c->command);
|
|
+ }
|
|
+ len = strlen(b);
|
|
+ if (total + len + 3 > client_pipesize) {
|
|
+ if (count == 1 && p == buffer && total == 0) {
|
|
+ /* The buffer is too small, but we have to
|
|
+ send something... */
|
|
+ PRINTF(DBG_PIPE,
|
|
+ ("* Reallocating buffer to %d bytes\n", len + 1));
|
|
+ p = buffer = alloca(len + 1);
|
|
+ } else {
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+ strcpy(p, b);
|
|
+ p += len;
|
|
+ total += len;
|
|
+ ++c->sent;
|
|
+ if (dbg_test(DBG_SERIAL))
|
|
+ break; /* Don't pipeline. */
|
|
+ }
|
|
+
|
|
+ end: /* Ready to send buffer, but are we
|
|
connected? */
|
|
- if (!cmd_reply.s) {
|
|
- c = lst_top(cmd_list);
|
|
- if (c->command != CMD_CONNECT) {
|
|
- err_internal( __func__, "Not connected, but no CMD_CONNECT\n" );
|
|
- }
|
|
- if ((cmd_reply.s = net_connect_tcp( c->host,
|
|
- c->service
|
|
- ? c->service
|
|
- : DICT_DEFAULT_SERVICE )) < 0) {
|
|
- const char *message;
|
|
-
|
|
- switch (cmd_reply.s) {
|
|
- case NET_NOHOST: message = "Can't get host entry for"; break;
|
|
- case NET_NOSERVICE: message = "Can't get service entry for"; break;
|
|
- case NET_NOPROTOCOL: message = "Can't get protocol entry for"; break;
|
|
- case NET_NOCONNECT: message = "Can't connect to"; break;
|
|
- default: message = "Unknown error for"; break;
|
|
- }
|
|
- PRINTF(DBG_VERBOSE,("%s %s.%s\n",
|
|
- message,
|
|
- c->host,
|
|
- c->service ? c->service : DICT_DEFAULT_SERVICE));
|
|
- if (lst_length(cmd_list) > 1) {
|
|
- c = lst_nth_get(cmd_list,2);
|
|
- if (c->command == CMD_CONNECT) {
|
|
- /* undo pipelining */
|
|
- cmd_reply.s = 0;
|
|
- if (!dbg_test(DBG_SERIAL)) {
|
|
- LST_ITERATE(cmd_list,pos,c) c->sent = 0;
|
|
- }
|
|
- return;
|
|
- }
|
|
- }
|
|
- client_close_pager();
|
|
- fprintf (stderr, "Cannot connect to any servers%s\n",\
|
|
- dbg_test(DBG_VERBOSE) ? "" : " (use -v to see why)" );
|
|
- exit (EXST_CONNECTION_FAILED);
|
|
- }
|
|
- cmd_reply.host = c->host;
|
|
- cmd_reply.service = c->service ? c->service : DICT_DEFAULT_SERVICE;
|
|
- cmd_reply.user = c->user;
|
|
- cmd_reply.key = c->key;
|
|
- }
|
|
- if ((len = strlen(buffer))) {
|
|
- char *pt;
|
|
-
|
|
- PRINTF(DBG_PIPE,("* Sending %d commands (%d bytes)\n",count,len));
|
|
- PRINTF(DBG_RAW,("* Send/%d: %s",c->command,buffer));
|
|
- pt = alloca(2*len);
|
|
- client_crlf(pt,buffer);
|
|
- net_write( cmd_reply.s, pt, strlen(pt) );
|
|
- } else {
|
|
- PRINTF(DBG_PIPE,("* Sending nothing\n"));
|
|
- PRINTF(DBG_RAW,("* Send/%d\n",c->command));
|
|
- }
|
|
-}
|
|
-
|
|
-static void process( void )
|
|
-{
|
|
- struct cmd *c;
|
|
- int expected;
|
|
- const char *message = NULL;
|
|
- int i;
|
|
- int *listed;
|
|
- FILE *old;
|
|
-
|
|
- while ((c = lst_top( cmd_list ))) {
|
|
- request(); /* Send requests */
|
|
- lst_pop( cmd_list );
|
|
- expected = CODE_OK;
|
|
- switch (c->command) {
|
|
- case CMD_PRINT:
|
|
- if (!formatted){
|
|
- if (c->comment) fprintf( dict_output, "%s", c->comment );
|
|
- }
|
|
-
|
|
- if (cmd_reply.match)
|
|
- client_print_matches( cmd_reply.data, 1, cmd_reply.word );
|
|
- else if (cmd_reply.listed)
|
|
- client_print_listed( cmd_reply.data );
|
|
- else
|
|
- client_print_text( cmd_reply.data, 1 );
|
|
-
|
|
- if (flush){
|
|
- fflush (dict_output);
|
|
- }
|
|
-
|
|
- client_free_text( cmd_reply.data );
|
|
- cmd_reply.data = NULL;
|
|
- cmd_reply.matches = cmd_reply.match = cmd_reply.listed = 0;
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_DEFPRINT:
|
|
- if (cmd_reply.count) {
|
|
- if (c->flag) {
|
|
- fprintf( dict_output, "%d definition%s found",
|
|
- cmd_reply.count,
|
|
- cmd_reply.count == 1 ? "" : "s" );
|
|
- fprintf( dict_output, "\n" );
|
|
- }
|
|
- for (i = 0; i < cmd_reply.count; i++) {
|
|
- client_print_definitions (&cmd_reply.defs [i]);
|
|
-
|
|
- if (flush){
|
|
- fflush (dict_output);
|
|
- }
|
|
-
|
|
- client_free_text( cmd_reply.defs[i].data );
|
|
- cmd_reply.defs[i].data = NULL;
|
|
- }
|
|
- xfree( cmd_reply.defs );
|
|
- cmd_reply.count = 0;
|
|
-
|
|
- } else if (cmd_reply.matches) {
|
|
- fprintf( dict_error,
|
|
- "No definitions found for \"%s\", perhaps you mean:",
|
|
- c->word );
|
|
- fprintf( dict_error, "\n" );
|
|
-
|
|
- old = dict_output;
|
|
- dict_output = dict_error;
|
|
- client_print_matches( cmd_reply.data, 0, c->word );
|
|
- dict_output = old;
|
|
+ if (!cmd_reply.s) {
|
|
+ c = lst_top(cmd_list);
|
|
+ if (c->command != CMD_CONNECT) {
|
|
+ err_internal(__func__, "Not connected, but no CMD_CONNECT\n");
|
|
+ }
|
|
+ if ((cmd_reply.s = net_connect_tcp(c->host,
|
|
+ c->service
|
|
+ ? c->service
|
|
+ : DICT_DEFAULT_SERVICE)) < 0) {
|
|
+ const char *message;
|
|
+
|
|
+ switch (cmd_reply.s) {
|
|
+ case NET_NOHOST:
|
|
+ message = "Can't get host entry for";
|
|
+ break;
|
|
+ case NET_NOSERVICE:
|
|
+ message = "Can't get service entry for";
|
|
+ break;
|
|
+ case NET_NOPROTOCOL:
|
|
+ message = "Can't get protocol entry for";
|
|
+ break;
|
|
+ case NET_NOCONNECT:
|
|
+ message = "Can't connect to";
|
|
+ break;
|
|
+ default:
|
|
+ message = "Unknown error for";
|
|
+ break;
|
|
+ }
|
|
+ PRINTF(DBG_VERBOSE, ("%s %s.%s\n",
|
|
+ message,
|
|
+ c->host,
|
|
+ c->service ? c->
|
|
+ service : DICT_DEFAULT_SERVICE));
|
|
+ if (lst_length(cmd_list) > 1) {
|
|
+ c = lst_nth_get(cmd_list, 2);
|
|
+ if (c->command == CMD_CONNECT) {
|
|
+ /* undo pipelining */
|
|
+ cmd_reply.s = 0;
|
|
+ if (!dbg_test(DBG_SERIAL)) {
|
|
+ LST_ITERATE(cmd_list, pos, c) c->sent = 0;
|
|
+ }
|
|
+ return;
|
|
+ }
|
|
+ }
|
|
+ client_close_pager();
|
|
+ fprintf(stderr, "Cannot connect to any servers%s\n",
|
|
+ dbg_test(DBG_VERBOSE) ? "" : " (use -v to see why)");
|
|
+ exit(EXST_CONNECTION_FAILED);
|
|
+ }
|
|
+ cmd_reply.host = c->host;
|
|
+ cmd_reply.service = c->service ? c->service : DICT_DEFAULT_SERVICE;
|
|
+ cmd_reply.user = c->user;
|
|
+ cmd_reply.key = c->key;
|
|
+ }
|
|
+ if ((len = strlen(buffer))) {
|
|
+ char *pt;
|
|
+
|
|
+ PRINTF(DBG_PIPE,
|
|
+ ("* Sending %d commands (%d bytes)\n", count, len));
|
|
+ PRINTF(DBG_RAW, ("* Send/%d: %s", c->command, buffer));
|
|
+ pt = alloca(2 * len);
|
|
+ client_crlf(pt, buffer);
|
|
+ net_write(cmd_reply.s, pt, strlen(pt));
|
|
+ } else {
|
|
+ PRINTF(DBG_PIPE, ("* Sending nothing\n"));
|
|
+ PRINTF(DBG_RAW, ("* Send/%d\n", c->command));
|
|
+ }
|
|
+}
|
|
+
|
|
+static void process(void)
|
|
+{
|
|
+ struct cmd *c;
|
|
+ int expected;
|
|
+ const char *message = NULL;
|
|
+ int i;
|
|
+ int *listed;
|
|
+ FILE *old;
|
|
+
|
|
+ while ((c = lst_top(cmd_list))) {
|
|
+ request(); /* Send requests */
|
|
+ lst_pop(cmd_list);
|
|
+ expected = CODE_OK;
|
|
+ switch (c->command) {
|
|
+ case CMD_PRINT:
|
|
+ if (!formatted) {
|
|
+ if (c->comment)
|
|
+ fprintf(dict_output, "%s", c->comment);
|
|
+ }
|
|
+
|
|
+ if (cmd_reply.match)
|
|
+ client_print_matches(cmd_reply.data, 1, cmd_reply.word);
|
|
+ else if (cmd_reply.listed)
|
|
+ client_print_listed(cmd_reply.data);
|
|
+ else
|
|
+ client_print_text(cmd_reply.data, 1);
|
|
|
|
- if (flush){
|
|
- fflush (dict_output);
|
|
+ if (flush) {
|
|
+ fflush(dict_output);
|
|
}
|
|
|
|
- client_free_text( cmd_reply.data );
|
|
+ client_free_text(cmd_reply.data);
|
|
cmd_reply.data = NULL;
|
|
- cmd_reply.matches = 0;
|
|
+ cmd_reply.matches = cmd_reply.match = cmd_reply.listed = 0;
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ case CMD_DEFPRINT:
|
|
+ if (cmd_reply.count) {
|
|
+ if (c->flag) {
|
|
+ fprintf(dict_output, "%d definition%s found",
|
|
+ cmd_reply.count,
|
|
+ cmd_reply.count == 1 ? "" : "s");
|
|
+ fprintf(dict_output, "\n");
|
|
+ }
|
|
+ for (i = 0; i < cmd_reply.count; i++) {
|
|
+ client_print_definitions(&cmd_reply.defs[i]);
|
|
+
|
|
+ if (flush) {
|
|
+ fflush(dict_output);
|
|
+ }
|
|
+
|
|
+ client_free_text(cmd_reply.defs[i].data);
|
|
+ cmd_reply.defs[i].data = NULL;
|
|
+ }
|
|
+ xfree(cmd_reply.defs);
|
|
+ cmd_reply.count = 0;
|
|
+
|
|
+ } else if (cmd_reply.matches) {
|
|
+ fprintf(dict_error,
|
|
+ "No definitions found for \"%s\", perhaps you mean:",
|
|
+ c->word);
|
|
+ fprintf(dict_error, "\n");
|
|
+
|
|
+ old = dict_output;
|
|
+ dict_output = dict_error;
|
|
+ client_print_matches(cmd_reply.data, 0, c->word);
|
|
+ dict_output = old;
|
|
+
|
|
+ if (flush) {
|
|
+ fflush(dict_output);
|
|
+ }
|
|
+
|
|
+ client_free_text(cmd_reply.data);
|
|
+ cmd_reply.data = NULL;
|
|
+ cmd_reply.matches = 0;
|
|
+
|
|
+ set_ex_status(EXST_APPROX_MATCHES);
|
|
+ } else {
|
|
+ fprintf(dict_error,
|
|
+ "No definitions found for \"%s\"\n", c->word);
|
|
+
|
|
+ set_ex_status(EXST_NO_MATCH);
|
|
+ }
|
|
+
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ case CMD_CONNECT:
|
|
+ if (!cmd_reply.s)
|
|
+ break; /* Connection failed, continue; */
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL, NULL,
|
|
+ &cmd_reply.msgid);
|
|
+ if (cmd_reply.retcode == CODE_ACCESS_DENIED) {
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL,
|
|
+ "Access to server %s.%s denied when connecting\n",
|
|
+ cmd_reply.host, cmd_reply.service);
|
|
+ }
|
|
+
|
|
+ /* */
|
|
+ host_connected = c->host;
|
|
+ if (c->service)
|
|
+ service_connected = c->service;
|
|
+ else
|
|
+ service_connected = DICT_DEFAULT_SERVICE;
|
|
+
|
|
+ /* */
|
|
+ expected = CODE_HELLO;
|
|
+ while (((struct cmd *) lst_top(cmd_list))->command ==
|
|
+ CMD_CONNECT)
|
|
+ lst_pop(cmd_list);
|
|
+
|
|
+ break;
|
|
+ case CMD_OPTION_MIME:
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL, NULL,
|
|
+ NULL);
|
|
+ if (cmd_reply.retcode != expected && dbg_test(DBG_VERBOSE))
|
|
+ fprintf(dict_error,
|
|
+ "Client command gave unexpected status code %d (%s)\n",
|
|
+ cmd_reply.retcode,
|
|
+ message ? message : "no message");
|
|
|
|
- set_ex_status (EXST_APPROX_MATCHES);
|
|
- } else {
|
|
- fprintf( dict_error,
|
|
- "No definitions found for \"%s\"\n", c->word );
|
|
-
|
|
- set_ex_status (EXST_NO_MATCH);
|
|
- }
|
|
-
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_CONNECT:
|
|
- if (!cmd_reply.s) break; /* Connection failed, continue; */
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL, NULL, NULL, NULL,
|
|
- &cmd_reply.msgid );
|
|
- if (cmd_reply.retcode == CODE_ACCESS_DENIED) {
|
|
- client_close_pager();
|
|
- err_fatal( NULL,
|
|
- "Access to server %s.%s denied when connecting\n",
|
|
- cmd_reply.host,
|
|
- cmd_reply.service );
|
|
- }
|
|
-
|
|
- /* */
|
|
- host_connected = c -> host;
|
|
- if (c -> service)
|
|
- service_connected = c -> service;
|
|
- else
|
|
- service_connected = DICT_DEFAULT_SERVICE;
|
|
-
|
|
- /* */
|
|
- expected = CODE_HELLO;
|
|
- while (((struct cmd *)lst_top(cmd_list))->command == CMD_CONNECT)
|
|
- lst_pop(cmd_list);
|
|
-
|
|
- break;
|
|
- case CMD_OPTION_MIME:
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL, NULL, NULL, NULL, NULL);
|
|
- if (cmd_reply.retcode != expected && dbg_test(DBG_VERBOSE))
|
|
- fprintf( dict_error, "Client command gave unexpected status code %d (%s)\n",
|
|
- cmd_reply.retcode, message ? message : "no message" );
|
|
-
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_CLIENT:
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL, NULL, NULL, NULL, NULL);
|
|
- if (cmd_reply.retcode != expected && dbg_test(DBG_VERBOSE))
|
|
- fprintf( dict_error, "Client command gave unexpected status code %d (%s)\n",
|
|
- cmd_reply.retcode, message ? message : "no message" );
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ case CMD_CLIENT:
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL, NULL,
|
|
+ NULL);
|
|
+ if (cmd_reply.retcode != expected && dbg_test(DBG_VERBOSE))
|
|
+ fprintf(dict_error,
|
|
+ "Client command gave unexpected status code %d (%s)\n",
|
|
+ cmd_reply.retcode,
|
|
+ message ? message : "no message");
|
|
|
|
/* set_ex_status (cmd_reply.retcode); */
|
|
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_AUTH:
|
|
- if (!cmd_reply.key || !cmd_reply.user || !cmd_reply.msgid) {
|
|
expected = cmd_reply.retcode;
|
|
break;
|
|
- }
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL, NULL, NULL, NULL, NULL);
|
|
- expected = CODE_AUTH_OK;
|
|
- if (cmd_reply.retcode == CODE_AUTH_DENIED) {
|
|
- err_warning( NULL,
|
|
- "Authentication to %s.%s denied\n",
|
|
- cmd_reply.host,
|
|
- cmd_reply.service );
|
|
- expected = CODE_AUTH_DENIED;
|
|
- }
|
|
- break;
|
|
- case CMD_INFO:
|
|
- expected = CODE_DATABASE_INFO;
|
|
- listed = NULL;
|
|
- goto gettext;
|
|
- case CMD_SERVER:
|
|
- expected = CODE_SERVER_INFO;
|
|
- listed = NULL;
|
|
- goto gettext;
|
|
- case CMD_HELP:
|
|
- expected = CODE_HELP;
|
|
- listed = NULL;
|
|
- goto gettext;
|
|
- case CMD_DBS:
|
|
- expected = CODE_DATABASE_LIST;
|
|
- listed = &cmd_reply.listed;
|
|
- goto gettext;
|
|
- case CMD_STRATS:
|
|
- expected = CODE_STRATEGY_LIST;
|
|
- listed = &cmd_reply.listed;
|
|
- goto gettext;
|
|
- gettext:
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- listed,
|
|
- NULL, NULL, NULL, NULL);
|
|
- if (cmd_reply.retcode == expected) {
|
|
- cmd_reply.data = client_read_text( cmd_reply.s );
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL,NULL,NULL,NULL,NULL);
|
|
- expected = CODE_OK;
|
|
- }
|
|
- break;
|
|
- case CMD_DEFINE:
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- &cmd_reply.count,
|
|
- NULL, NULL, NULL, NULL );
|
|
- if (!client_defines) tim_start( "define" );
|
|
- switch (expected = cmd_reply.retcode) {
|
|
- case CODE_DEFINITIONS_FOUND:
|
|
- cmd_reply.defs = xmalloc(cmd_reply.count*sizeof(struct def));
|
|
- expected = CODE_DEFINITION_FOLLOWS;
|
|
- for (i = 0; i < cmd_reply.count; i++) {
|
|
- ++client_defines;
|
|
- cmd_reply.retcode
|
|
- = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL,
|
|
- &cmd_reply.defs[i].word,
|
|
- &cmd_reply.defs[i].db,
|
|
- &cmd_reply.defs[i].dbname,
|
|
- NULL );
|
|
- if (cmd_reply.retcode != expected) goto error;
|
|
- cmd_reply.defs[i].data = client_read_text( cmd_reply.s );
|
|
- }
|
|
- expected = CODE_OK;
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL,NULL,NULL,NULL,NULL );
|
|
- break;
|
|
- case CODE_NO_MATCH:
|
|
- PRINTF(DBG_VERBOSE,
|
|
- ("No match found for \"%s\" in %s\n",c->word,c->database));
|
|
- break;
|
|
- case CODE_INVALID_DB:
|
|
- fprintf(stderr, "%s is not a valid database, use -D for a list\n",
|
|
- c->database );
|
|
- set_ex_status (EXST_INVALID_DB);
|
|
- break;
|
|
- case CODE_NO_DATABASES:
|
|
- fprintf( dict_error, "There are no databases currently available\n" );
|
|
-
|
|
- set_ex_status (EXST_NO_DATABASES);
|
|
-
|
|
- break;
|
|
- default:
|
|
- expected = CODE_OK;
|
|
- }
|
|
- error:
|
|
- break;
|
|
- case CMD_MATCH:
|
|
- cmd_reply.match = 1;
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- &cmd_reply.matches,
|
|
- NULL, NULL, NULL, NULL );
|
|
- switch (expected = cmd_reply.retcode) {
|
|
- case CODE_MATCHES_FOUND:
|
|
- cmd_reply.data = client_read_text( cmd_reply.s );
|
|
- expected = CODE_OK;
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL,NULL,NULL,NULL,NULL );
|
|
- break;
|
|
- case CODE_TEMPORARILY_UNAVAILABLE:
|
|
- fprintf (stderr,
|
|
- "Server temporarily unavailable\n");
|
|
-
|
|
- set_ex_status (EXST_TEMPORARILY_UNAVAILABLE);
|
|
-
|
|
- break;
|
|
- case CODE_NO_MATCH:
|
|
- PRINTF(DBG_VERBOSE,
|
|
- ("No match found in %s for \"%s\" using %s\n",
|
|
- c->database,c->word,c->strategy));
|
|
-
|
|
- set_ex_status (EXST_NO_MATCH);
|
|
-
|
|
- break;
|
|
- case CODE_INVALID_DB:
|
|
- fprintf( dict_error,
|
|
- "%s is not a valid database, use -D for a list\n",
|
|
- c->database );
|
|
-
|
|
- set_ex_status (EXST_INVALID_DB);
|
|
-
|
|
- break;
|
|
- case CODE_INVALID_STRATEGY:
|
|
- fprintf( dict_error,
|
|
- "%s is not a valid search strategy, use -S for a list\n",
|
|
- c->strategy );
|
|
-
|
|
- set_ex_status (EXST_INVALID_STRATEGY);
|
|
-
|
|
- break;
|
|
- case CODE_NO_DATABASES:
|
|
- fprintf( dict_error,
|
|
- "There are no databases currently available\n" );
|
|
-
|
|
- set_ex_status (EXST_NO_DATABASES);
|
|
-
|
|
- break;
|
|
- case CODE_NO_STRATEGIES:
|
|
- fprintf( dict_error,
|
|
- "There are no search strategies currently available\n" );
|
|
-
|
|
- set_ex_status (EXST_NO_STRATEGIES);
|
|
-
|
|
- break;
|
|
- default:
|
|
- expected = CODE_OK;
|
|
- }
|
|
- break;
|
|
- case CMD_SPELL:
|
|
- if (cmd_reply.retcode == CODE_NO_MATCH) {
|
|
- prepend_command( make_command( CMD_MATCH,
|
|
- c->database, DEF_STRAT, c->word ) );
|
|
- }
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_WIND:
|
|
- if (cmd_reply.matches) {
|
|
- if (!cmd_reply.data)
|
|
- err_internal( __func__,
|
|
- "%d matches, but no list\n", cmd_reply.matches );
|
|
-
|
|
- for (i = cmd_reply.matches; i > 0; --i) {
|
|
- /* skip MIME header */
|
|
- const char *line = lst_nth_get( cmd_reply.data, i );
|
|
- arg_List a;
|
|
- const char *orig, *s;
|
|
- char *escaped, *d;
|
|
-
|
|
- if (option_mime){
|
|
- if (line [0] == 0 ||
|
|
- (line [0] == '\r' && line [1] == '\0'))
|
|
- {
|
|
- break;
|
|
- }
|
|
- }
|
|
-
|
|
- /* */
|
|
- a = arg_argify( line, 0 );
|
|
- if (arg_count(a) != 2)
|
|
- err_internal( __func__,
|
|
- "MATCH command didn't return 2 args: \"%s\"\n",
|
|
- line );
|
|
-
|
|
- prepend_command( make_command( CMD_DEFPRINT,
|
|
- str_find(arg_get(a,0)),
|
|
- str_copy(arg_get(a,1)),
|
|
- 0 ) );
|
|
-
|
|
- /* Escape " and \ in word before sending */
|
|
- orig = arg_get(a,1);
|
|
- escaped = xmalloc(strlen(orig) * 2 + 1);
|
|
- for (s = orig, d = escaped; *s;) {
|
|
- switch (*s) {
|
|
- case '"':
|
|
- case '\\':
|
|
- *d++ = '\\';
|
|
- default:
|
|
- *d++ = *s++;
|
|
- }
|
|
- }
|
|
- *d++ = '\0';
|
|
- prepend_command( make_command( CMD_DEFINE,
|
|
- str_find(arg_get(a,0)),
|
|
- str_copy(escaped),
|
|
- 0 ) );
|
|
- xfree(escaped);
|
|
- arg_destroy(a);
|
|
- }
|
|
- client_free_text( cmd_reply.data );
|
|
- cmd_reply.matches = 0;
|
|
- } else {
|
|
- fprintf( dict_error, "No matches found for \"%s\"", c->word );
|
|
- fprintf( dict_error, "\n" );
|
|
-
|
|
- set_ex_status (EXST_NO_MATCH);
|
|
- }
|
|
- expected = cmd_reply.retcode;
|
|
- break;
|
|
- case CMD_CLOSE:
|
|
- cmd_reply.retcode = client_read_status( cmd_reply.s,
|
|
- &message,
|
|
- NULL, NULL, NULL, NULL, NULL);
|
|
- expected = CODE_GOODBYE;
|
|
- break;
|
|
- default:
|
|
- err_internal( __func__, "Illegal command %d\n", c->command );
|
|
- }
|
|
- if (cmd_reply.s && cmd_reply.retcode != expected) {
|
|
- client_close_pager();
|
|
- unexpected_status_code (expected, cmd_reply.retcode, message);
|
|
- }
|
|
- PRINTF(DBG_RAW,("* Processed %d\n",c->command));
|
|
- xfree(c);
|
|
- }
|
|
+ case CMD_AUTH:
|
|
+ if (!cmd_reply.key || !cmd_reply.user || !cmd_reply.msgid) {
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ }
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL, NULL,
|
|
+ NULL);
|
|
+ expected = CODE_AUTH_OK;
|
|
+ if (cmd_reply.retcode == CODE_AUTH_DENIED) {
|
|
+ err_warning(NULL,
|
|
+ "Authentication to %s.%s denied\n",
|
|
+ cmd_reply.host, cmd_reply.service);
|
|
+ expected = CODE_AUTH_DENIED;
|
|
+ }
|
|
+ break;
|
|
+ case CMD_INFO:
|
|
+ expected = CODE_DATABASE_INFO;
|
|
+ listed = NULL;
|
|
+ goto gettext;
|
|
+ case CMD_SERVER:
|
|
+ expected = CODE_SERVER_INFO;
|
|
+ listed = NULL;
|
|
+ goto gettext;
|
|
+ case CMD_HELP:
|
|
+ expected = CODE_HELP;
|
|
+ listed = NULL;
|
|
+ goto gettext;
|
|
+ case CMD_DBS:
|
|
+ expected = CODE_DATABASE_LIST;
|
|
+ listed = &cmd_reply.listed;
|
|
+ goto gettext;
|
|
+ case CMD_STRATS:
|
|
+ expected = CODE_STRATEGY_LIST;
|
|
+ listed = &cmd_reply.listed;
|
|
+ goto gettext;
|
|
+ gettext:
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ listed,
|
|
+ NULL, NULL, NULL, NULL);
|
|
+ if (cmd_reply.retcode == expected) {
|
|
+ cmd_reply.data = client_read_text(cmd_reply.s);
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL,
|
|
+ NULL, NULL);
|
|
+ expected = CODE_OK;
|
|
+ }
|
|
+ break;
|
|
+ case CMD_DEFINE:
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ &cmd_reply.count,
|
|
+ NULL, NULL, NULL, NULL);
|
|
+ if (!client_defines)
|
|
+ tim_start("define");
|
|
+ switch (expected = cmd_reply.retcode) {
|
|
+ case CODE_DEFINITIONS_FOUND:
|
|
+ cmd_reply.defs =
|
|
+ xmalloc(cmd_reply.count * sizeof(struct def));
|
|
+ expected = CODE_DEFINITION_FOLLOWS;
|
|
+ for (i = 0; i < cmd_reply.count; i++) {
|
|
+ ++client_defines;
|
|
+ cmd_reply.retcode
|
|
+ = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL,
|
|
+ &cmd_reply.defs[i].word,
|
|
+ &cmd_reply.defs[i].db,
|
|
+ &cmd_reply.defs[i].dbname,
|
|
+ NULL);
|
|
+ if (cmd_reply.retcode != expected)
|
|
+ goto error;
|
|
+ cmd_reply.defs[i].data = client_read_text(cmd_reply.s);
|
|
+ }
|
|
+ expected = CODE_OK;
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL,
|
|
+ NULL, NULL);
|
|
+ break;
|
|
+ case CODE_NO_MATCH:
|
|
+ PRINTF(DBG_VERBOSE,
|
|
+ ("No match found for \"%s\" in %s\n", c->word,
|
|
+ c->database));
|
|
+ break;
|
|
+ case CODE_INVALID_DB:
|
|
+ fprintf(stderr,
|
|
+ "%s is not a valid database, use -D for a list\n",
|
|
+ c->database);
|
|
+ set_ex_status(EXST_INVALID_DB);
|
|
+ break;
|
|
+ case CODE_NO_DATABASES:
|
|
+ fprintf(dict_error,
|
|
+ "There are no databases currently available\n");
|
|
+
|
|
+ set_ex_status(EXST_NO_DATABASES);
|
|
+
|
|
+ break;
|
|
+ default:
|
|
+ expected = CODE_OK;
|
|
+ }
|
|
+ error:
|
|
+ break;
|
|
+ case CMD_MATCH:
|
|
+ cmd_reply.match = 1;
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ &cmd_reply.matches,
|
|
+ NULL, NULL, NULL, NULL);
|
|
+ switch (expected = cmd_reply.retcode) {
|
|
+ case CODE_MATCHES_FOUND:
|
|
+ cmd_reply.data = client_read_text(cmd_reply.s);
|
|
+ expected = CODE_OK;
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL,
|
|
+ NULL, NULL);
|
|
+ break;
|
|
+ case CODE_TEMPORARILY_UNAVAILABLE:
|
|
+ fprintf(stderr, "Server temporarily unavailable\n");
|
|
+
|
|
+ set_ex_status(EXST_TEMPORARILY_UNAVAILABLE);
|
|
+
|
|
+ break;
|
|
+ case CODE_NO_MATCH:
|
|
+ PRINTF(DBG_VERBOSE,
|
|
+ ("No match found in %s for \"%s\" using %s\n",
|
|
+ c->database, c->word, c->strategy));
|
|
+
|
|
+ set_ex_status(EXST_NO_MATCH);
|
|
+
|
|
+ break;
|
|
+ case CODE_INVALID_DB:
|
|
+ fprintf(dict_error,
|
|
+ "%s is not a valid database, use -D for a list\n",
|
|
+ c->database);
|
|
+
|
|
+ set_ex_status(EXST_INVALID_DB);
|
|
+
|
|
+ break;
|
|
+ case CODE_INVALID_STRATEGY:
|
|
+ fprintf(dict_error,
|
|
+ "%s is not a valid search strategy, use -S for a list\n",
|
|
+ c->strategy);
|
|
+
|
|
+ set_ex_status(EXST_INVALID_STRATEGY);
|
|
+
|
|
+ break;
|
|
+ case CODE_NO_DATABASES:
|
|
+ fprintf(dict_error,
|
|
+ "There are no databases currently available\n");
|
|
+
|
|
+ set_ex_status(EXST_NO_DATABASES);
|
|
+
|
|
+ break;
|
|
+ case CODE_NO_STRATEGIES:
|
|
+ fprintf(dict_error,
|
|
+ "There are no search strategies currently available\n");
|
|
+
|
|
+ set_ex_status(EXST_NO_STRATEGIES);
|
|
+
|
|
+ break;
|
|
+ default:
|
|
+ expected = CODE_OK;
|
|
+ }
|
|
+ break;
|
|
+ case CMD_SPELL:
|
|
+ if (cmd_reply.retcode == CODE_NO_MATCH) {
|
|
+ prepend_command(make_command(CMD_MATCH,
|
|
+ c->database, DEF_STRAT,
|
|
+ c->word));
|
|
+ }
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ case CMD_WIND:
|
|
+ if (cmd_reply.matches) {
|
|
+ if (!cmd_reply.data)
|
|
+ err_internal(__func__,
|
|
+ "%d matches, but no list\n",
|
|
+ cmd_reply.matches);
|
|
+
|
|
+ for (i = cmd_reply.matches; i > 0; --i) {
|
|
+ /* skip MIME header */
|
|
+ const char *line = lst_nth_get(cmd_reply.data, i);
|
|
+ arg_List a;
|
|
+ const char *orig, *s;
|
|
+ char *escaped, *d;
|
|
+
|
|
+ if (option_mime) {
|
|
+ if (line[0] == 0 ||
|
|
+ (line[0] == '\r' && line[1] == '\0')) {
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ /* */
|
|
+ a = arg_argify(line, 0);
|
|
+ if (arg_count(a) != 2)
|
|
+ err_internal(__func__,
|
|
+ "MATCH command didn't return 2 args: \"%s\"\n",
|
|
+ line);
|
|
+
|
|
+ prepend_command(make_command(CMD_DEFPRINT,
|
|
+ str_find(arg_get(a, 0)),
|
|
+ str_copy(arg_get(a, 1)),
|
|
+ 0));
|
|
+
|
|
+ /* Escape " and \ in word before sending */
|
|
+ orig = arg_get(a, 1);
|
|
+ escaped = xmalloc(strlen(orig) * 2 + 1);
|
|
+ for (s = orig, d = escaped; *s;) {
|
|
+ switch (*s) {
|
|
+ case '"':
|
|
+ case '\\':
|
|
+ *d++ = '\\';
|
|
+ default:
|
|
+ *d++ = *s++;
|
|
+ }
|
|
+ }
|
|
+ *d++ = '\0';
|
|
+ prepend_command(make_command(CMD_DEFINE,
|
|
+ str_find(arg_get(a, 0)),
|
|
+ str_copy(escaped), 0));
|
|
+ xfree(escaped);
|
|
+ arg_destroy(a);
|
|
+ }
|
|
+ client_free_text(cmd_reply.data);
|
|
+ cmd_reply.matches = 0;
|
|
+ } else {
|
|
+ fprintf(dict_error, "No matches found for \"%s\"",
|
|
+ c->word);
|
|
+ fprintf(dict_error, "\n");
|
|
+
|
|
+ set_ex_status(EXST_NO_MATCH);
|
|
+ }
|
|
+ expected = cmd_reply.retcode;
|
|
+ break;
|
|
+ case CMD_CLOSE:
|
|
+ cmd_reply.retcode = client_read_status(cmd_reply.s,
|
|
+ &message,
|
|
+ NULL, NULL, NULL, NULL,
|
|
+ NULL);
|
|
+ expected = CODE_GOODBYE;
|
|
+ break;
|
|
+ default:
|
|
+ err_internal(__func__, "Illegal command %d\n", c->command);
|
|
+ }
|
|
+ if (cmd_reply.s && cmd_reply.retcode != expected) {
|
|
+ client_close_pager();
|
|
+ unexpected_status_code(expected, cmd_reply.retcode, message);
|
|
+ }
|
|
+ PRINTF(DBG_RAW, ("* Processed %d\n", c->command));
|
|
+ xfree(c);
|
|
+ }
|
|
}
|
|
|
|
#if 0
|
|
-static void handler( int sig )
|
|
+static void handler(int sig)
|
|
{
|
|
- const char *name = NULL;
|
|
-
|
|
- switch (sig) {
|
|
- case SIGHUP: name = "SIGHUP"; break;
|
|
- case SIGINT: name = "SIGINT"; break;
|
|
- case SIGQUIT: name = "SIGQUIT"; break;
|
|
- case SIGILL: name = "SIGILL"; break;
|
|
- case SIGTRAP: name = "SIGTRAP"; break;
|
|
- case SIGTERM: name = "SIGTERM"; break;
|
|
- case SIGPIPE: name = "SIGPIPE"; break;
|
|
- }
|
|
-
|
|
- if (name)
|
|
- err_fatal( __func__, "Caught %s, exiting\n", name );
|
|
- else
|
|
- err_fatal( __func__, "Caught signal %d, exiting\n", sig );
|
|
-
|
|
- exit(0);
|
|
-}
|
|
-
|
|
-static void setsig( int sig, void (*f)(int) )
|
|
-{
|
|
- struct sigaction sa;
|
|
-
|
|
- sa.sa_handler = f;
|
|
- sigemptyset(&sa.sa_mask);
|
|
- sa.sa_flags = 0;
|
|
- sigaction(sig, &sa, NULL);
|
|
+ const char *name = NULL;
|
|
+
|
|
+ switch (sig) {
|
|
+ case SIGHUP:
|
|
+ name = "SIGHUP";
|
|
+ break;
|
|
+ case SIGINT:
|
|
+ name = "SIGINT";
|
|
+ break;
|
|
+ case SIGQUIT:
|
|
+ name = "SIGQUIT";
|
|
+ break;
|
|
+ case SIGILL:
|
|
+ name = "SIGILL";
|
|
+ break;
|
|
+ case SIGTRAP:
|
|
+ name = "SIGTRAP";
|
|
+ break;
|
|
+ case SIGTERM:
|
|
+ name = "SIGTERM";
|
|
+ break;
|
|
+ case SIGPIPE:
|
|
+ name = "SIGPIPE";
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ if (name)
|
|
+ err_fatal(__func__, "Caught %s, exiting\n", name);
|
|
+ else
|
|
+ err_fatal(__func__, "Caught signal %d, exiting\n", sig);
|
|
+
|
|
+ exit(0);
|
|
+}
|
|
+
|
|
+static void setsig(int sig, void (*f) (int))
|
|
+{
|
|
+ struct sigaction sa;
|
|
+
|
|
+ sa.sa_handler = f;
|
|
+ sigemptyset(&sa.sa_mask);
|
|
+ sa.sa_flags = 0;
|
|
+ sigaction(sig, &sa, NULL);
|
|
}
|
|
#endif
|
|
|
|
-static void client_config_print( FILE *stream, lst_List c )
|
|
+static void client_config_print(FILE * stream, lst_List c)
|
|
{
|
|
- FILE *s = stream ? stream : stderr;
|
|
- lst_Position p;
|
|
- dictServer *e;
|
|
-
|
|
- printf( "Configuration file:\n" );
|
|
- LST_ITERATE(dict_Servers,p,e) {
|
|
- if (e->port || e->user || e->secret) {
|
|
- fprintf( s, " server %s {\n", e->host );
|
|
- if (e->port) fprintf( s, " port %s\n", e->port );
|
|
- if (e->user) fprintf( s, " user %s %s\n",
|
|
- e->user,
|
|
- e->secret ? "*" : "(none)" );
|
|
- fprintf( s, " }\n" );
|
|
- } else {
|
|
- fprintf( s, " server %s\n", e->host );
|
|
- }
|
|
- }
|
|
-}
|
|
-
|
|
-static const char *id_string (void)
|
|
-{
|
|
- static char buffer[BUFFERSIZE];
|
|
-
|
|
- snprintf( buffer, BUFFERSIZE, "%s", DICT_VERSION );
|
|
-
|
|
- return buffer;
|
|
-}
|
|
-
|
|
-static const char *client_get_banner( void )
|
|
-{
|
|
- static char *buffer= NULL;
|
|
- struct utsname uts;
|
|
-
|
|
- if (buffer) return buffer;
|
|
- uname( &uts );
|
|
- buffer = xmalloc(256);
|
|
- snprintf( buffer, 256,
|
|
- "%s %s/rf on %s %s", err_program_name (), id_string (),
|
|
- uts.sysname, uts.release );
|
|
- return buffer;
|
|
-}
|
|
-
|
|
-static void banner( FILE *out_stream )
|
|
-{
|
|
- fprintf( out_stream , "%s\n", client_get_banner() );
|
|
- fprintf( out_stream,
|
|
- "Copyright 1997-2002 Rickard E. Faith (faith@dict.org)\n" );
|
|
- fprintf( out_stream,
|
|
- "Copyright 2002-2007 Aleksey Cheusov (vle@gmx.net)\n" );
|
|
- fprintf( out_stream, "\n" );
|
|
-}
|
|
-
|
|
-static void license( void )
|
|
-{
|
|
- static const char *license_msg[] = {
|
|
- "This program is free software; you can redistribute it and/or modify it",
|
|
- "under the terms of the GNU General Public License as published by the",
|
|
- "Free Software Foundation; either version 1, or (at your option) any",
|
|
- "later version.",
|
|
- "",
|
|
- "This program is distributed in the hope that it will be useful, but",
|
|
- "WITHOUT ANY WARRANTY; without even the implied warranty of",
|
|
- "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU",
|
|
- "General Public License for more details.",
|
|
- "",
|
|
- "You should have received a copy of the GNU General Public License along",
|
|
- "with this program; if not, write to the Free Software Foundation, Inc.,",
|
|
- "675 Mass Ave, Cambridge, MA 02139, USA.",
|
|
- 0 };
|
|
- const char **p = license_msg;
|
|
-
|
|
- banner ( stdout );
|
|
- while (*p) fprintf( stdout, " %s\n", *p++ );
|
|
-}
|
|
-
|
|
-static void help( FILE *out_stream )
|
|
-{
|
|
- static const char *help_msg[] = {
|
|
- "Usage: dict [options] [word]",
|
|
- "Query a dictd server for the definition of a word",
|
|
- "",
|
|
- "-h --host <server> specify server",
|
|
- "-p --port <service> specify port",
|
|
- "-d --database <dbname> select a database to search",
|
|
- "-m --match match instead of define",
|
|
- "-s --strategy <strategy> strategy for matching or defining",
|
|
- "-c --config <file> specify configuration file",
|
|
- "-C --nocorrect disable attempted spelling correction",
|
|
- "-D --dbs show available databases",
|
|
- "-S --strats show available search strategies",
|
|
- "-H --serverhelp show server help",
|
|
- "-i --info <dbname> show information about a database",
|
|
- "-I --serverinfo show information about the server",
|
|
- "-a --noauth disable authentication",
|
|
- "-u --user <username> username for authentication",
|
|
- "-k --key <key> shared secret for authentication",
|
|
- "-V --version display version information",
|
|
- "-L --license display copyright and license information",
|
|
- " --help display this help",
|
|
- "-v --verbose be verbose",
|
|
- "-r --raw trace raw transaction",
|
|
- " --debug <flag> set debugging flag",
|
|
- " --pipesize <size> specify buffer size for pipelining (256)",
|
|
- " --client <text> additional text for client command",
|
|
- "-M --mime send OPTION MIME command if server supports it",
|
|
- "-f --formatted use strict tabbed format of output",
|
|
- 0 };
|
|
- const char **p = help_msg;
|
|
-
|
|
- banner( out_stream );
|
|
- while (*p) fprintf( out_stream, "%s\n", *p++ );
|
|
-}
|
|
-
|
|
-int main( int argc, char **argv )
|
|
-{
|
|
- int c;
|
|
- const char *host = NULL;
|
|
- const char *service = NULL;
|
|
- const char *user = NULL;
|
|
- const char *key = NULL;
|
|
- const char *database = DEF_DB;
|
|
- const char *strategy = DEF_STRAT;
|
|
- const char *configFile = NULL;
|
|
- const char *word = NULL;
|
|
- int doauth = 1;
|
|
- int docorrect = 1;
|
|
- int offset = 0;
|
|
- int i;
|
|
- enum { DEFINE = 0x0001,
|
|
- MATCH = 0x0002,
|
|
- INFO = 0x0010,
|
|
- SERVER = 0x0020,
|
|
- DBS = 0x0040,
|
|
- STRATS = 0x0080,
|
|
- HELP = 0x0100,
|
|
- OPTION_MIME = 0x0200,
|
|
- } function = DEFINE;
|
|
- struct option longopts[] = {
|
|
- { "host", 1, 0, 'h' },
|
|
- { "port", 1, 0, 'p' },
|
|
- { "database", 1, 0, 'd' },
|
|
- { "info", 1, 0, 'i' },
|
|
- { "serverinfo", 0, 0, 'I' },
|
|
- { "match", 0, 0, 'm' },
|
|
- { "strategy", 1, 0, 's' },
|
|
- { "nocorrect", 0, 0, 'C' },
|
|
- { "config", 1, 0, 'c' },
|
|
- { "dbs", 0, 0, 'D' },
|
|
- { "strats", 0, 0, 'S' },
|
|
- { "serverhelp", 0, 0, 'H' },
|
|
- { "noauth", 0, 0, 'a' },
|
|
- { "user", 1, 0, 'u' },
|
|
- { "key", 1, 0, 'k' },
|
|
- { "version", 0, 0, 'V' },
|
|
- { "license", 0, 0, 'L' },
|
|
- { "help", 0, 0, 501 },
|
|
- { "verbose", 0, 0, 'v' },
|
|
- { "raw", 0, 0, 'r' },
|
|
- { "pager", 1, 0, 'P' },
|
|
- { "debug", 1, 0, 502 },
|
|
- { "pipesize", 1, 0, 504 },
|
|
- { "client", 1, 0, 505 },
|
|
- { "mime", 1, 0, 'M' },
|
|
- { "formatted", 0, 0, 'f' },
|
|
- { "flush", 0, 0, 'F' },
|
|
- { 0, 0, 0, 0 }
|
|
- };
|
|
-
|
|
- dict_output = stdout;
|
|
- dict_error = stderr;
|
|
-
|
|
- maa_init (argv[0]);
|
|
-
|
|
- dbg_register( DBG_VERBOSE, "verbose" );
|
|
- dbg_register( DBG_RAW, "raw" );
|
|
- dbg_register( DBG_SCAN, "scan" );
|
|
- dbg_register( DBG_PARSE, "parse" );
|
|
- dbg_register( DBG_PIPE, "pipe" );
|
|
- dbg_register( DBG_SERIAL, "serial" );
|
|
- dbg_register( DBG_TIME, "time" );
|
|
- dbg_register( DBG_URL, "url" );
|
|
+ FILE *s = stream ? stream : stderr;
|
|
+ lst_Position p;
|
|
+ dictServer *e;
|
|
+
|
|
+ printf("Configuration file:\n");
|
|
+ LST_ITERATE(dict_Servers, p, e) {
|
|
+ if (e->port || e->user || e->secret) {
|
|
+ fprintf(s, " server %s {\n", e->host);
|
|
+ if (e->port)
|
|
+ fprintf(s, " port %s\n", e->port);
|
|
+ if (e->user)
|
|
+ fprintf(s, " user %s %s\n",
|
|
+ e->user, e->secret ? "*" : "(none)");
|
|
+ fprintf(s, " }\n");
|
|
+ } else {
|
|
+ fprintf(s, " server %s\n", e->host);
|
|
+ }
|
|
+ }
|
|
+}
|
|
+
|
|
+static const char *id_string(void)
|
|
+{
|
|
+ static char buffer[BUFFERSIZE];
|
|
+
|
|
+ snprintf(buffer, BUFFERSIZE, "%s", DICT_VERSION);
|
|
+
|
|
+ return buffer;
|
|
+}
|
|
+
|
|
+static const char *client_get_banner(void)
|
|
+{
|
|
+ static char *buffer = NULL;
|
|
+ struct utsname uts;
|
|
+
|
|
+ if (buffer)
|
|
+ return buffer;
|
|
+ uname(&uts);
|
|
+ buffer = xmalloc(256);
|
|
+ snprintf(buffer, 256,
|
|
+ "%s %s/rf on %s %s", err_program_name(), id_string(),
|
|
+ uts.sysname, uts.release);
|
|
+ return buffer;
|
|
+}
|
|
+
|
|
+static void banner(FILE * out_stream)
|
|
+{
|
|
+ fprintf(out_stream, "%s\n", client_get_banner());
|
|
+ fprintf(out_stream,
|
|
+ "Copyright 1997-2002 Rickard E. Faith (faith@dict.org)\n");
|
|
+ fprintf(out_stream,
|
|
+ "Copyright 2002-2007 Aleksey Cheusov (vle@gmx.net)\n");
|
|
+ fprintf(out_stream, "\n");
|
|
+}
|
|
+
|
|
+static void license(void)
|
|
+{
|
|
+ static const char *license_msg[] = {
|
|
+ "This program is free software; you can redistribute it and/or modify it",
|
|
+ "under the terms of the GNU General Public License as published by the",
|
|
+ "Free Software Foundation; either version 1, or (at your option) any",
|
|
+ "later version.",
|
|
+ "",
|
|
+ "This program is distributed in the hope that it will be useful, but",
|
|
+ "WITHOUT ANY WARRANTY; without even the implied warranty of",
|
|
+ "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU",
|
|
+ "General Public License for more details.",
|
|
+ "",
|
|
+ "You should have received a copy of the GNU General Public License along",
|
|
+ "with this program; if not, write to the Free Software Foundation, Inc.,",
|
|
+ "675 Mass Ave, Cambridge, MA 02139, USA.",
|
|
+ 0
|
|
+ };
|
|
+ const char **p = license_msg;
|
|
+
|
|
+ banner(stdout);
|
|
+ while (*p)
|
|
+ fprintf(stdout, " %s\n", *p++);
|
|
+}
|
|
+
|
|
+static void help(FILE * out_stream)
|
|
+{
|
|
+ static const char *help_msg[] = {
|
|
+ "Usage: dict [options] [word]",
|
|
+ "Query a dictd server for the definition of a word",
|
|
+ "",
|
|
+ "-h --host <server> specify server",
|
|
+ "-p --port <service> specify port",
|
|
+ "-d --database <dbname> select a database to search",
|
|
+ "-m --match match instead of define",
|
|
+ "-s --strategy <strategy> strategy for matching or defining",
|
|
+ "-c --config <file> specify configuration file",
|
|
+ "-C --nocorrect disable attempted spelling correction",
|
|
+ "-D --dbs show available databases",
|
|
+ "-S --strats show available search strategies",
|
|
+ "-H --serverhelp show server help",
|
|
+ "-i --info <dbname> show information about a database",
|
|
+ "-I --serverinfo show information about the server",
|
|
+ "-a --noauth disable authentication",
|
|
+ "-u --user <username> username for authentication",
|
|
+ "-k --key <key> shared secret for authentication",
|
|
+ "-V --version display version information",
|
|
+ "-L --license display copyright and license information",
|
|
+ " --help display this help",
|
|
+ "-v --verbose be verbose",
|
|
+ "-r --raw trace raw transaction",
|
|
+ " --debug <flag> set debugging flag",
|
|
+ " --pipesize <size> specify buffer size for pipelining (256)",
|
|
+ " --client <text> additional text for client command",
|
|
+ "-M --mime send OPTION MIME command if server supports it",
|
|
+ "-f --formatted use strict tabbed format of output",
|
|
+ 0
|
|
+ };
|
|
+ const char **p = help_msg;
|
|
+
|
|
+ banner(out_stream);
|
|
+ while (*p)
|
|
+ fprintf(out_stream, "%s\n", *p++);
|
|
+}
|
|
+
|
|
+int main(int argc, char **argv)
|
|
+{
|
|
+ int c;
|
|
+ const char *host = NULL;
|
|
+ const char *service = NULL;
|
|
+ const char *user = NULL;
|
|
+ const char *key = NULL;
|
|
+ const char *database = DEF_DB;
|
|
+ const char *strategy = DEF_STRAT;
|
|
+ const char *configFile = NULL;
|
|
+ const char *word = NULL;
|
|
+ int doauth = 1;
|
|
+ int docorrect = 1;
|
|
+ int offset = 0;
|
|
+ int i;
|
|
+ enum { DEFINE = 0x0001,
|
|
+ MATCH = 0x0002,
|
|
+ INFO = 0x0010,
|
|
+ SERVER = 0x0020,
|
|
+ DBS = 0x0040,
|
|
+ STRATS = 0x0080,
|
|
+ HELP = 0x0100,
|
|
+ OPTION_MIME = 0x0200,
|
|
+ } function = DEFINE;
|
|
+ struct option longopts[] = {
|
|
+ {"host", 1, 0, 'h'},
|
|
+ {"port", 1, 0, 'p'},
|
|
+ {"database", 1, 0, 'd'},
|
|
+ {"info", 1, 0, 'i'},
|
|
+ {"serverinfo", 0, 0, 'I'},
|
|
+ {"match", 0, 0, 'm'},
|
|
+ {"strategy", 1, 0, 's'},
|
|
+ {"nocorrect", 0, 0, 'C'},
|
|
+ {"config", 1, 0, 'c'},
|
|
+ {"dbs", 0, 0, 'D'},
|
|
+ {"strats", 0, 0, 'S'},
|
|
+ {"serverhelp", 0, 0, 'H'},
|
|
+ {"noauth", 0, 0, 'a'},
|
|
+ {"user", 1, 0, 'u'},
|
|
+ {"key", 1, 0, 'k'},
|
|
+ {"version", 0, 0, 'V'},
|
|
+ {"license", 0, 0, 'L'},
|
|
+ {"help", 0, 0, 501},
|
|
+ {"verbose", 0, 0, 'v'},
|
|
+ {"raw", 0, 0, 'r'},
|
|
+ {"pager", 1, 0, 'P'},
|
|
+ {"debug", 1, 0, 502},
|
|
+ {"pipesize", 1, 0, 504},
|
|
+ {"client", 1, 0, 505},
|
|
+ {"mime", 1, 0, 'M'},
|
|
+ {"formatted", 0, 0, 'f'},
|
|
+ {"flush", 0, 0, 'F'},
|
|
+ {0, 0, 0, 0}
|
|
+ };
|
|
+
|
|
+ dict_output = stdout;
|
|
+ dict_error = stderr;
|
|
+
|
|
+ maa_init(argv[0]);
|
|
+
|
|
+ dbg_register(DBG_VERBOSE, "verbose");
|
|
+ dbg_register(DBG_RAW, "raw");
|
|
+ dbg_register(DBG_SCAN, "scan");
|
|
+ dbg_register(DBG_PARSE, "parse");
|
|
+ dbg_register(DBG_PIPE, "pipe");
|
|
+ dbg_register(DBG_SERIAL, "serial");
|
|
+ dbg_register(DBG_TIME, "time");
|
|
+ dbg_register(DBG_URL, "url");
|
|
|
|
- while ((c = getopt_long( argc, argv,
|
|
+ while ((c = getopt_long(argc, argv,
|
|
"h:p:d:i:Ims:DSHau:c:Ck:VLvrP:MfF",
|
|
- longopts, NULL )) != EOF)
|
|
- {
|
|
- switch (c) {
|
|
- case 'h': host = optarg; break;
|
|
- case 'p': service = optarg; break;
|
|
- case 'd': database = optarg; break;
|
|
- case 'i': database = optarg; function |= INFO; break;
|
|
- case 'I': function |= SERVER; break;
|
|
- case 'm': function &= ~DEFINE; function |= MATCH; break;
|
|
- case 's': strategy = optarg; break;
|
|
- case 'D': function |= DBS; break;
|
|
- case 'S': function |= STRATS; break;
|
|
- case 'H': function |= HELP; break;
|
|
- case 'M': option_mime = 1; function |= OPTION_MIME; break;
|
|
- case 'c': configFile = optarg; break;
|
|
- case 'C': docorrect = 0; break;
|
|
- case 'a': doauth = 0; break;
|
|
- case 'u': user = optarg; break;
|
|
- case 'k': key = optarg; break;
|
|
- case 'V': banner( stdout ); exit(1); break;
|
|
- case 'L': license(); exit(1); break;
|
|
- case 'v': dbg_set( "verbose" ); break;
|
|
- case 'r': dbg_set( "raw" ); break;
|
|
- case 'P':
|
|
- if (strcmp (optarg, "-")){
|
|
- fprintf (stderr, "Option --pager is now deprecated");
|
|
- exit (1);
|
|
- }
|
|
- break;
|
|
- case 505: client_text = optarg; break;
|
|
- case 504: client_pipesize = atoi(optarg); break;
|
|
- case 502: dbg_set( optarg ); break;
|
|
- case 501: help( stdout ); exit(1); break;
|
|
- case 'f': formatted = 1; break;
|
|
- case 'F': flush = 1; break;
|
|
- default: help( stderr ); exit(1); break;
|
|
- }
|
|
- }
|
|
-
|
|
- if (optind == argc && (!(function & ~(DEFINE|MATCH)))) {
|
|
- banner( stderr );
|
|
- fprintf( stderr, "Use --help for help\n" );
|
|
- exit(1);
|
|
- }
|
|
-
|
|
- if (client_pipesize > 1000000) client_pipesize = 1000000;
|
|
-
|
|
- if (dbg_test(DBG_PARSE)) prs_set_debug(1);
|
|
- if (dbg_test(DBG_SCAN)) yy_flex_debug = 1;
|
|
- else yy_flex_debug = 0;
|
|
-
|
|
- if (configFile) {
|
|
- prs_file_nocpp( configFile );
|
|
- } else {
|
|
- char b[256];
|
|
- char *env = getenv("HOME");
|
|
-
|
|
- snprintf( b, 256, "%s/%s", env ? env : "./", DICT_RC_NAME );
|
|
- PRINTF(DBG_VERBOSE,("Trying %s...\n",b));
|
|
- if (!access(b, R_OK)) {
|
|
- prs_file_nocpp(b);
|
|
- } else {
|
|
- PRINTF(DBG_VERBOSE,("Trying %s...\n",
|
|
- DICT_CONFIG_PATH DICT_CONFIG_NAME));
|
|
- if (!access( DICT_CONFIG_PATH DICT_CONFIG_NAME, R_OK ))
|
|
- prs_file_nocpp( DICT_CONFIG_PATH DICT_CONFIG_NAME );
|
|
- }
|
|
- }
|
|
- if (dbg_test(DBG_VERBOSE)) {
|
|
- if (dict_Servers) client_config_print( NULL, dict_Servers );
|
|
- else fprintf( stderr, "No configuration\n" );
|
|
- }
|
|
-
|
|
- if (optind < argc && !strncmp(argv[optind],"dict://",7)) {
|
|
- char *p;
|
|
- int state, fin;
|
|
- char *s;
|
|
+ longopts, NULL)) != EOF) {
|
|
+ switch (c) {
|
|
+ case 'h':
|
|
+ host = optarg;
|
|
+ break;
|
|
+ case 'p':
|
|
+ service = optarg;
|
|
+ break;
|
|
+ case 'd':
|
|
+ database = optarg;
|
|
+ break;
|
|
+ case 'i':
|
|
+ database = optarg;
|
|
+ function |= INFO;
|
|
+ break;
|
|
+ case 'I':
|
|
+ function |= SERVER;
|
|
+ break;
|
|
+ case 'm':
|
|
+ function &= ~DEFINE;
|
|
+ function |= MATCH;
|
|
+ break;
|
|
+ case 's':
|
|
+ strategy = optarg;
|
|
+ break;
|
|
+ case 'D':
|
|
+ function |= DBS;
|
|
+ break;
|
|
+ case 'S':
|
|
+ function |= STRATS;
|
|
+ break;
|
|
+ case 'H':
|
|
+ function |= HELP;
|
|
+ break;
|
|
+ case 'M':
|
|
+ option_mime = 1;
|
|
+ function |= OPTION_MIME;
|
|
+ break;
|
|
+ case 'c':
|
|
+ configFile = optarg;
|
|
+ break;
|
|
+ case 'C':
|
|
+ docorrect = 0;
|
|
+ break;
|
|
+ case 'a':
|
|
+ doauth = 0;
|
|
+ break;
|
|
+ case 'u':
|
|
+ user = optarg;
|
|
+ break;
|
|
+ case 'k':
|
|
+ key = optarg;
|
|
+ break;
|
|
+ case 'V':
|
|
+ banner(stdout);
|
|
+ exit(1);
|
|
+ break;
|
|
+ case 'L':
|
|
+ license();
|
|
+ exit(1);
|
|
+ break;
|
|
+ case 'v':
|
|
+ dbg_set("verbose");
|
|
+ break;
|
|
+ case 'r':
|
|
+ dbg_set("raw");
|
|
+ break;
|
|
+ case 'P':
|
|
+ if (strcmp(optarg, "-")) {
|
|
+ fprintf(stderr, "Option --pager is now deprecated");
|
|
+ exit(1);
|
|
+ }
|
|
+ break;
|
|
+ case 505:
|
|
+ client_text = optarg;
|
|
+ break;
|
|
+ case 504:
|
|
+ client_pipesize = atoi(optarg);
|
|
+ break;
|
|
+ case 502:
|
|
+ dbg_set(optarg);
|
|
+ break;
|
|
+ case 501:
|
|
+ help(stdout);
|
|
+ exit(1);
|
|
+ break;
|
|
+ case 'f':
|
|
+ formatted = 1;
|
|
+ break;
|
|
+ case 'F':
|
|
+ flush = 1;
|
|
+ break;
|
|
+ default:
|
|
+ help(stderr);
|
|
+ exit(1);
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (optind == argc && (!(function & ~(DEFINE | MATCH)))) {
|
|
+ banner(stderr);
|
|
+ fprintf(stderr, "Use --help for help\n");
|
|
+ exit(1);
|
|
+ }
|
|
+
|
|
+ if (client_pipesize > 1000000)
|
|
+ client_pipesize = 1000000;
|
|
+
|
|
+ if (dbg_test(DBG_PARSE))
|
|
+ prs_set_debug(1);
|
|
+ if (dbg_test(DBG_SCAN))
|
|
+ yy_flex_debug = 1;
|
|
+ else
|
|
+ yy_flex_debug = 0;
|
|
+
|
|
+ if (configFile) {
|
|
+ prs_file_nocpp(configFile);
|
|
+ } else {
|
|
+ char b[256];
|
|
+ char *env = getenv("HOME");
|
|
+
|
|
+ snprintf(b, 256, "%s/%s", env ? env : "./", DICT_RC_NAME);
|
|
+ PRINTF(DBG_VERBOSE, ("Trying %s...\n", b));
|
|
+ if (!access(b, R_OK)) {
|
|
+ prs_file_nocpp(b);
|
|
+ } else {
|
|
+ PRINTF(DBG_VERBOSE, ("Trying %s...\n",
|
|
+ DICT_CONFIG_PATH DICT_CONFIG_NAME));
|
|
+ if (!access(DICT_CONFIG_PATH DICT_CONFIG_NAME, R_OK))
|
|
+ prs_file_nocpp(DICT_CONFIG_PATH DICT_CONFIG_NAME);
|
|
+ }
|
|
+ }
|
|
+ if (dbg_test(DBG_VERBOSE)) {
|
|
+ if (dict_Servers)
|
|
+ client_config_print(NULL, dict_Servers);
|
|
+ else
|
|
+ fprintf(stderr, "No configuration\n");
|
|
+ }
|
|
+
|
|
+ if (optind < argc && !strncmp(argv[optind], "dict://", 7)) {
|
|
+ char *p;
|
|
+ int state, fin;
|
|
+ char *s;
|
|
|
|
/* dict://<user>:<passphrase>@<host>:<port>/d:<word>:<database>:<n>
|
|
000000 111111111111 222222 333333 4 555555 6666666666 777
|
|
@@ -1489,213 +1641,303 @@ int main( int argc, char **argv )
|
|
|
|
*/
|
|
|
|
- for (s = p = argv[optind] + 7, state = 0, fin = 0; !fin; ++p) {
|
|
- switch (*p) {
|
|
- case '\0': ++fin;
|
|
- case ':':
|
|
- switch (state) {
|
|
- case 0: *p = '\0'; host = user = cpy(s); ++state; s=p+1; break;
|
|
- case 2: *p = '\0'; host = cpy(s); ++state; s=p+1; break;
|
|
- case 4:
|
|
- if (s == p - 1) {
|
|
- if (*s == 'd') function = DEFINE;
|
|
- else if (*s == 'm') function = MATCH;
|
|
- else {
|
|
- PRINTF(DBG_URL,("State = %d, s = %s\n",state,s));
|
|
- client_close_pager();
|
|
- err_fatal( NULL, "Parse error at %s\n", p );
|
|
- }
|
|
- ++state; s=p+1; break;
|
|
- } else {
|
|
- PRINTF(DBG_URL,("State = %d, s = %s\n",state,s));
|
|
- client_close_pager();
|
|
- err_fatal( NULL, "Parse error at %s\n", p );
|
|
- }
|
|
- break;
|
|
- case 5: *p = '\0'; word = cpy(s); ++state; s=p+1; break;
|
|
- case 6: *p = '\0'; database = cpy(s); ++state; s=p+1; break;
|
|
- case 7: *p = '\0';
|
|
- if (function == DEFINE) offset = atoi(s);
|
|
- else strategy = cpy(s);
|
|
- ++state; s=p+1; break;
|
|
- case 8: *p = '\0';
|
|
- if (function == MATCH) offset = atoi(s); ++state; s=p+1; break;
|
|
- /* FALLTHROUGH */
|
|
- default:
|
|
- PRINTF(DBG_URL,("State = %d, s = %s\n",state,s));
|
|
- client_close_pager();
|
|
- err_fatal( NULL, "Parse error at %s\n", p );
|
|
+ for (s = p = argv[optind] + 7, state = 0, fin = 0; !fin; ++p) {
|
|
+ switch (*p) {
|
|
+ case '\0':
|
|
+ ++fin;
|
|
+ case '[':
|
|
+ state = 32;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case ']':
|
|
+ *p = '\0';
|
|
+ host = cpy(s);
|
|
+ state = 33;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case ':':
|
|
+ switch (state) {
|
|
+ case 32:
|
|
+ continue;
|
|
+ case 33:
|
|
+ state = 1;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 0:
|
|
+ *p = '\0';
|
|
+ host = user = cpy(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 2:
|
|
+ *p = '\0';
|
|
+ host = cpy(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 4:
|
|
+ if (s == p - 1) {
|
|
+ if (*s == 'd')
|
|
+ function = DEFINE;
|
|
+ else if (*s == 'm')
|
|
+ function = MATCH;
|
|
+ else {
|
|
+ PRINTF(DBG_URL,
|
|
+ ("State = %d, s = %s\n", state, s));
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL, "Parse error at %s\n", p);
|
|
+ }
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ } else {
|
|
+ PRINTF(DBG_URL,
|
|
+ ("State = %d, s = %s\n", state, s));
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL, "Parse error at %s\n", p);
|
|
+ }
|
|
+ break;
|
|
+ case 5:
|
|
+ *p = '\0';
|
|
+ word = cpy(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 6:
|
|
+ *p = '\0';
|
|
+ database = cpy(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 7:
|
|
+ *p = '\0';
|
|
+ if (function == DEFINE)
|
|
+ offset = atoi(s);
|
|
+ else
|
|
+ strategy = cpy(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 8:
|
|
+ *p = '\0';
|
|
+ if (function == MATCH)
|
|
+ offset = atoi(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ /* FALLTHROUGH */
|
|
+ default:
|
|
+ PRINTF(DBG_URL, ("State = %d, s = %s\n", state, s));
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL, "Parse error at %s\n", p);
|
|
+ }
|
|
+ break;
|
|
+ case '@':
|
|
+ switch (state) {
|
|
+ case 1:
|
|
+ *p = '\0';
|
|
+ key = xstrdup(s);
|
|
+ ++state;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ default:
|
|
+ PRINTF(DBG_URL, ("State = %d, s = %s\n", state, s));
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL, "Parse error at %s\n", p);
|
|
+ }
|
|
+ break;
|
|
+ case '/':
|
|
+ switch (state) {
|
|
+ case 0:
|
|
+ *p = '\0';
|
|
+ host = xstrdup(s);
|
|
+ state = 4;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 1:
|
|
+ case 3:
|
|
+ *p = '\0';
|
|
+ service = xstrdup(s);
|
|
+ state = 4;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 2:
|
|
+ *p = '\0';
|
|
+ host = xstrdup(s);
|
|
+ state = 4;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ case 33:
|
|
+ state = 4;
|
|
+ s = p + 1;
|
|
+ break;
|
|
+ default:
|
|
+ PRINTF(DBG_URL, ("State = %d, s = %s\n", state, s));
|
|
+ client_close_pager();
|
|
+ err_fatal(NULL, "Parse error at %s\n", p);
|
|
+ }
|
|
+ break;
|
|
}
|
|
- break;
|
|
- case '@':
|
|
- switch (state) {
|
|
- case 1: *p = '\0'; key = xstrdup(s); ++state; s=p+1; break;
|
|
- default:
|
|
- PRINTF(DBG_URL,("State = %d, s = %s\n",state,s));
|
|
- client_close_pager();
|
|
- err_fatal( NULL, "Parse error at %s\n", p );
|
|
- }
|
|
- break;
|
|
- case '/':
|
|
- switch (state) {
|
|
- case 0: *p = '\0'; host = xstrdup(s); state = 4; s=p+1; break;
|
|
- case 1: *p = '\0'; service = xstrdup(s); state = 4; s=p+1; break;
|
|
- case 2: *p = '\0'; host = xstrdup(s); state = 4; s=p+1; break;
|
|
- default:
|
|
- PRINTF(DBG_URL,("State = %d, s = %s\n",state,s));
|
|
- client_close_pager();
|
|
- err_fatal( NULL, "Parse error at %s\n", p );
|
|
- }
|
|
- break;
|
|
- }
|
|
- }
|
|
-
|
|
- if (!key) user = NULL;
|
|
- if (!database) database = DEF_DB;
|
|
- if (!strategy) strategy = DEF_STRAT;
|
|
-
|
|
- if (dbg_test(DBG_URL)) {
|
|
- printf( "user = %s, passphrase = %s\n",
|
|
- user ? user : "(null)", key ? key : "(null)" );
|
|
- printf( "host = %s, port = %s\n",
|
|
- host ? host : "(null)", service ? service : "(null)" );
|
|
- printf( "word = %s, database = %s, strategy = %s\n",
|
|
- word ? word : "(null)",
|
|
- strategy ? strategy : "(null)",
|
|
- database ? database : "(null)" );
|
|
- }
|
|
- if (host && !host[0]) { /* Empty host causes default to be used. */
|
|
- xfree((void *)host);
|
|
- host = NULL;
|
|
- }
|
|
- }
|
|
+ }
|
|
|
|
+ if (!key)
|
|
+ user = NULL;
|
|
+ if (!database)
|
|
+ database = DEF_DB;
|
|
+ if (!strategy)
|
|
+ strategy = DEF_STRAT;
|
|
+
|
|
+ if (dbg_test(DBG_URL)) {
|
|
+ printf("user = %s, passphrase = %s\n",
|
|
+ user ? user : "(null)", key ? key : "(null)");
|
|
+ printf("host = %s, port = %s\n",
|
|
+ host ? host : "(null)", service ? service : "(null)");
|
|
+ printf("word = %s, database = %s, strategy = %s\n",
|
|
+ word ? word : "(null)",
|
|
+ strategy ? strategy : "(null)",
|
|
+ database ? database : "(null)");
|
|
+ }
|
|
+ if (host && !host[0]) { /* Empty host causes default to be used. */
|
|
+ xfree((void *) host);
|
|
+ host = NULL;
|
|
+ }
|
|
+ }
|
|
#if 0
|
|
- setsig(SIGHUP, handler);
|
|
- setsig(SIGINT, handler);
|
|
- setsig(SIGQUIT, handler);
|
|
- setsig(SIGILL, handler);
|
|
- setsig(SIGTRAP, handler);
|
|
- setsig(SIGTERM, handler);
|
|
- setsig(SIGPIPE, handler);
|
|
+ setsig(SIGHUP, handler);
|
|
+ setsig(SIGINT, handler);
|
|
+ setsig(SIGQUIT, handler);
|
|
+ setsig(SIGILL, handler);
|
|
+ setsig(SIGTRAP, handler);
|
|
+ setsig(SIGTERM, handler);
|
|
+ setsig(SIGPIPE, handler);
|
|
#endif
|
|
|
|
- fflush(stdout);
|
|
- fflush(stderr);
|
|
+ fflush(stdout);
|
|
+ fflush(stderr);
|
|
|
|
- tim_start("total");
|
|
+ tim_start("total");
|
|
|
|
- if (host) {
|
|
- append_command( make_command( CMD_CONNECT, host, service, user, key ) );
|
|
- } else {
|
|
- lst_Position p;
|
|
- dictServer *s;
|
|
-
|
|
- if (dict_Servers) {
|
|
- LST_ITERATE(dict_Servers,p,s) {
|
|
- append_command( make_command( CMD_CONNECT,
|
|
- s->host,
|
|
- s->port,
|
|
- s->user,
|
|
- s->secret ) );
|
|
- }
|
|
+ if (host) {
|
|
+ append_command(make_command
|
|
+ (CMD_CONNECT, host, service, user, key));
|
|
+ } else {
|
|
+ lst_Position p;
|
|
+ dictServer *s;
|
|
+
|
|
+ if (dict_Servers) {
|
|
+ LST_ITERATE(dict_Servers, p, s) {
|
|
+ append_command(make_command(CMD_CONNECT,
|
|
+ s->host,
|
|
+ s->port, s->user, s->secret));
|
|
+ }
|
|
#ifdef USE_DICT_ORG
|
|
- }
|
|
- append_command(make_command(CMD_CONNECT,"dict.org", NULL,user,key));
|
|
- append_command(make_command(CMD_CONNECT,"alt0.dict.org",NULL,user,key));
|
|
+ }
|
|
+ append_command(make_command
|
|
+ (CMD_CONNECT, "dict.org", NULL, user, key));
|
|
+ append_command(make_command
|
|
+ (CMD_CONNECT, "alt0.dict.org", NULL, user, key));
|
|
#else
|
|
- }else{
|
|
- fprintf (stderr, "'dict.conf' doesn't specify any dict server\n");
|
|
- exit (1);
|
|
- }
|
|
+ } else {
|
|
+ fprintf(stderr,
|
|
+ "'dict.conf' doesn't specify any dict server\n");
|
|
+ exit(1);
|
|
+ }
|
|
#endif
|
|
- }
|
|
- append_command( make_command( CMD_CLIENT, client_get_banner() ) );
|
|
- if (doauth) append_command( make_command( CMD_AUTH ) );
|
|
- if (function & OPTION_MIME) {
|
|
- append_command( make_command( CMD_OPTION_MIME ) );
|
|
- append_command( make_command( CMD_PRINT, NULL ) );
|
|
- }
|
|
- if (function & INFO) {
|
|
- append_command( make_command( CMD_INFO, database ) );
|
|
- append_command( make_command( CMD_PRINT, NULL ) );
|
|
- }
|
|
- if (function & SERVER) {
|
|
- append_command( make_command( CMD_SERVER ) );
|
|
- append_command( make_command( CMD_PRINT, NULL ) );
|
|
- }
|
|
- if (function & DBS) {
|
|
- append_command( make_command( CMD_DBS ) );
|
|
- append_command( make_command( CMD_PRINT, "Databases available:\n" ) );
|
|
- }
|
|
- if (function & STRATS) {
|
|
- append_command( make_command( CMD_STRATS ) );
|
|
- append_command( make_command( CMD_PRINT, "Strategies available:\n" ) );
|
|
- }
|
|
- if (function & HELP) {
|
|
- append_command( make_command( CMD_HELP ) );
|
|
- append_command( make_command( CMD_PRINT, "Server help:\n" ) );
|
|
- }
|
|
-
|
|
- if (function & MATCH) {
|
|
- if (word) {
|
|
- append_command( make_command( CMD_MATCH, database, strategy, word ) );
|
|
- append_command( make_command( CMD_PRINT, NULL ) );
|
|
- } else {
|
|
- for (i = optind; i < argc; i++) {
|
|
- append_command( make_command( CMD_MATCH,
|
|
- database, strategy, argv[i] ) );
|
|
- append_command( make_command( CMD_PRINT, NULL ) );
|
|
- }
|
|
- }
|
|
- } else if (function & DEFINE) {
|
|
- if (word) {
|
|
- append_command( make_command( CMD_DEFINE, database, word ) );
|
|
- append_command( make_command( CMD_DEFPRINT, database, word, 1 ) );
|
|
- } else {
|
|
- for (i = optind; i < argc; i++) {
|
|
- if (!strcmp(strategy, DEF_STRAT)) {
|
|
- append_command( make_command( CMD_DEFINE, database, argv[i] ) );
|
|
- if (docorrect)
|
|
- append_command( make_command( CMD_SPELL, database, argv[i]));
|
|
- append_command( make_command(CMD_DEFPRINT,database,argv[i],1) );
|
|
- } else {
|
|
- append_command( make_command( CMD_MATCH,
|
|
- database, strategy, argv[i] ) );
|
|
- append_command( make_command( CMD_WIND,
|
|
- database, strategy, argv[i] ) );
|
|
- }
|
|
- }
|
|
- }
|
|
- }
|
|
- append_command( make_command( CMD_CLOSE ) );
|
|
-
|
|
- if (!dbg_test(DBG_VERBOSE|DBG_TIME)) client_open_pager();
|
|
- process();
|
|
- client_close_pager();
|
|
-
|
|
- if (dbg_test(DBG_TIME)) {
|
|
- fprintf( dict_output, "\n" );
|
|
- tim_stop("total");
|
|
- if (client_defines) {
|
|
- tim_stop("define");
|
|
- fprintf( stderr,
|
|
- "* %ld definitions in %.2fr %.2fu %.2fs"
|
|
- " => %.1f d/sec\n",
|
|
- client_defines,
|
|
- tim_get_real( "define" ),
|
|
- tim_get_user( "define" ),
|
|
- tim_get_system( "define" ),
|
|
- client_defines / tim_get_real( "define" ) );
|
|
- }
|
|
- fprintf( stderr,
|
|
- "* %ld bytes total in %.2fr %.2fu %.2fs => %.0f bps\n",
|
|
- client_bytes,
|
|
- tim_get_real( "total" ),
|
|
- tim_get_user( "total" ),
|
|
- tim_get_system( "total" ),
|
|
- client_bytes / tim_get_real( "total" ) );
|
|
- }
|
|
+ }
|
|
+ append_command(make_command(CMD_CLIENT, client_get_banner()));
|
|
+ if (doauth)
|
|
+ append_command(make_command(CMD_AUTH));
|
|
+ if (function & OPTION_MIME) {
|
|
+ append_command(make_command(CMD_OPTION_MIME));
|
|
+ append_command(make_command(CMD_PRINT, NULL));
|
|
+ }
|
|
+ if (function & INFO) {
|
|
+ append_command(make_command(CMD_INFO, database));
|
|
+ append_command(make_command(CMD_PRINT, NULL));
|
|
+ }
|
|
+ if (function & SERVER) {
|
|
+ append_command(make_command(CMD_SERVER));
|
|
+ append_command(make_command(CMD_PRINT, NULL));
|
|
+ }
|
|
+ if (function & DBS) {
|
|
+ append_command(make_command(CMD_DBS));
|
|
+ append_command(make_command(CMD_PRINT, "Databases available:\n"));
|
|
+ }
|
|
+ if (function & STRATS) {
|
|
+ append_command(make_command(CMD_STRATS));
|
|
+ append_command(make_command(CMD_PRINT, "Strategies available:\n"));
|
|
+ }
|
|
+ if (function & HELP) {
|
|
+ append_command(make_command(CMD_HELP));
|
|
+ append_command(make_command(CMD_PRINT, "Server help:\n"));
|
|
+ }
|
|
+
|
|
+ if (function & MATCH) {
|
|
+ if (word) {
|
|
+ append_command(make_command
|
|
+ (CMD_MATCH, database, strategy, word));
|
|
+ append_command(make_command(CMD_PRINT, NULL));
|
|
+ } else {
|
|
+ for (i = optind; i < argc; i++) {
|
|
+ append_command(make_command(CMD_MATCH,
|
|
+ database, strategy, argv[i]));
|
|
+ append_command(make_command(CMD_PRINT, NULL));
|
|
+ }
|
|
+ }
|
|
+ } else if (function & DEFINE) {
|
|
+ if (word) {
|
|
+ append_command(make_command(CMD_DEFINE, database, word));
|
|
+ append_command(make_command(CMD_DEFPRINT, database, word, 1));
|
|
+ } else {
|
|
+ for (i = optind; i < argc; i++) {
|
|
+ if (!strcmp(strategy, DEF_STRAT)) {
|
|
+ append_command(make_command
|
|
+ (CMD_DEFINE, database, argv[i]));
|
|
+ if (docorrect)
|
|
+ append_command(make_command
|
|
+ (CMD_SPELL, database, argv[i]));
|
|
+ append_command(make_command
|
|
+ (CMD_DEFPRINT, database, argv[i], 1));
|
|
+ } else {
|
|
+ append_command(make_command(CMD_MATCH,
|
|
+ database, strategy,
|
|
+ argv[i]));
|
|
+ append_command(make_command
|
|
+ (CMD_WIND, database, strategy,
|
|
+ argv[i]));
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+ append_command(make_command(CMD_CLOSE));
|
|
+
|
|
+ if (!dbg_test(DBG_VERBOSE | DBG_TIME))
|
|
+ client_open_pager();
|
|
+ process();
|
|
+ client_close_pager();
|
|
+
|
|
+ if (dbg_test(DBG_TIME)) {
|
|
+ fprintf(dict_output, "\n");
|
|
+ tim_stop("total");
|
|
+ if (client_defines) {
|
|
+ tim_stop("define");
|
|
+ fprintf(stderr,
|
|
+ "* %ld definitions in %.2fr %.2fu %.2fs"
|
|
+ " => %.1f d/sec\n",
|
|
+ client_defines,
|
|
+ tim_get_real("define"),
|
|
+ tim_get_user("define"),
|
|
+ tim_get_system("define"),
|
|
+ client_defines / tim_get_real("define"));
|
|
+ }
|
|
+ fprintf(stderr,
|
|
+ "* %ld bytes total in %.2fr %.2fu %.2fs => %.0f bps\n",
|
|
+ client_bytes,
|
|
+ tim_get_real("total"),
|
|
+ tim_get_user("total"),
|
|
+ tim_get_system("total"),
|
|
+ client_bytes / tim_get_real("total"));
|
|
+ }
|
|
|
|
- return ex_status;
|
|
+ return ex_status;
|
|
}
|
|
--- a/dictd.h
|
|
+++ b/dictd.h
|
|
@@ -42,9 +42,9 @@
|
|
*/
|
|
|
|
extern const char *site_info;
|
|
-extern int site_info_no_banner;
|
|
-extern int site_info_no_uptime;
|
|
-extern int site_info_no_dblist;
|
|
+extern int site_info_no_banner;
|
|
+extern int site_info_no_uptime;
|
|
+extern int site_info_no_dblist;
|
|
|
|
extern const char *daemon_service;
|
|
extern int client_delay;
|
|
@@ -79,59 +79,60 @@ extern int bind_to_set;
|
|
|
|
|
|
|
|
-extern void dict_disable_strat (dictDatabase *db, const char* strat);
|
|
+extern void dict_disable_strat(dictDatabase * db, const char *strat);
|
|
|
|
-extern void dict_dump_list( lst_List list );
|
|
-extern void dict_destroy_list( lst_List list );
|
|
+extern void dict_dump_list(lst_List list);
|
|
+extern void dict_destroy_list(lst_List list);
|
|
|
|
-extern int dict_destroy_datum( const void *datum );
|
|
+extern int dict_destroy_datum(const void *datum);
|
|
|
|
#ifdef USE_PLUGIN
|
|
-extern int dict_plugin_open (dictDatabase *db);
|
|
-extern void dict_plugin_close (dictDatabase *db);
|
|
+extern int dict_plugin_open(dictDatabase * db);
|
|
+extern void dict_plugin_close(dictDatabase * db);
|
|
#endif
|
|
|
|
/* dictd.c */
|
|
|
|
-extern void set_minimal (void);
|
|
+extern void set_minimal(void);
|
|
|
|
-extern void dict_initsetproctitle( int argc, char **argv, char **envp );
|
|
-extern void dict_setproctitle( const char *format, ... );
|
|
-extern const char *dict_format_time( double t );
|
|
-extern const char *dict_get_hostname( void );
|
|
-extern const char *dict_get_banner( int shortFlag );
|
|
-
|
|
-extern dictConfig *DictConfig; /* GLOBAL VARIABLE */
|
|
-extern int _dict_comparisons; /* GLOBAL VARIABLE */
|
|
-extern int _dict_forks; /* GLOBAL VARIABLE */
|
|
+extern void dict_initsetproctitle(int argc, char **argv, char **envp);
|
|
+extern void dict_setproctitle(const char *format, ...);
|
|
+extern const char *dict_format_time(double t);
|
|
+extern const char *dict_get_hostname(void);
|
|
+extern const char *dict_get_banner(int shortFlag);
|
|
+
|
|
+extern dictConfig *DictConfig; /* GLOBAL VARIABLE */
|
|
+extern int _dict_comparisons; /* GLOBAL VARIABLE */
|
|
+extern int _dict_forks; /* GLOBAL VARIABLE */
|
|
extern const char *locale;
|
|
|
|
extern const char *locale;
|
|
-extern int inetd; /* 1 if --inetd is applied, 0 otherwise */
|
|
+extern int inetd; /* 1 if --inetd is applied, 0 otherwise */
|
|
|
|
/*
|
|
If the filename doesn't start with / or .,
|
|
it is prepended with DICT_DIR
|
|
*/
|
|
-extern const char *postprocess_dict_filename (const char *fn);
|
|
+extern const char *postprocess_dict_filename(const char *fn);
|
|
/*
|
|
If the filename doesn't start with / or .,
|
|
it is prepended with PLUGIN_DIR
|
|
*/
|
|
-extern const char *postprocess_plugin_filename (const char *fn);
|
|
+extern const char *postprocess_plugin_filename(const char *fn);
|
|
|
|
/* daemon.c */
|
|
|
|
-extern int dict_daemon( int s, struct sockaddr_in *csin, int error );
|
|
-extern int dict_inetd( int error );
|
|
-extern void daemon_terminate( int sig, const char *name );
|
|
+extern int dict_daemon(int s, struct sockaddr_storage *csin, char ***argv0,
|
|
+ int delay, int error);
|
|
+extern int dict_inetd(int error);
|
|
+extern void daemon_terminate(int sig, const char *name);
|
|
|
|
/* */
|
|
extern int utf8_mode;
|
|
extern int stdin2stdout_mode;
|
|
/* dmalloc must be last */
|
|
#ifdef DMALLOC_FUNC_CHECK
|
|
-# include "dmalloc.h"
|
|
+#include "dmalloc.h"
|
|
#endif
|
|
|
|
#endif
|
|
--- a/daemon.c
|
|
+++ b/daemon.c
|
|
@@ -2,17 +2,17 @@
|
|
* Created: Fri Feb 28 18:17:56 1997 by faith@dict.org
|
|
* Copyright 1997, 1998, 1999, 2000, 2002 Rickard E. Faith (faith@dict.org)
|
|
* Copyright 2002-2008 Aleksey Cheusov (vle@gmx.net)
|
|
- *
|
|
+ *
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 1, or (at your option) any
|
|
* later version.
|
|
- *
|
|
+ *
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
- *
|
|
+ *
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
@@ -30,1579 +30,1652 @@
|
|
#include <ctype.h>
|
|
#include <setjmp.h>
|
|
|
|
-int stdin2stdout_mode = 0; /* copy stdin to stdout ( dict_dictd function ) */
|
|
+int stdin2stdout_mode = 0; /* copy stdin to stdout ( dict_dictd function ) */
|
|
|
|
-static int _dict_defines, _dict_matches;
|
|
-static int daemonS_in = 0;
|
|
-static int daemonS_out = 1;
|
|
-static const char *daemonHostname = NULL;
|
|
-static const char *daemonIP = NULL;
|
|
-static int daemonPort = -1;
|
|
-static char daemonStamp[256] = "";
|
|
-static jmp_buf env;
|
|
-
|
|
-static int daemonMime;
|
|
-
|
|
-static void daemon_define( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_match( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_show_db( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_show_strat( const char *cmdline, int argc, const char **argv );
|
|
-void daemon_show_info( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_show_server( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_show( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_option_mime( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_option( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_client( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_auth( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_status( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_help( const char *cmdline, int argc, const char **argv );
|
|
-static void daemon_quit( const char *cmdline, int argc, const char **argv );
|
|
+static int _dict_defines, _dict_matches;
|
|
+static int daemonS_in = 0;
|
|
+static int daemonS_out = 1;
|
|
+static const char *daemonHostname = NULL;
|
|
+static const char *daemonIP = NULL;
|
|
+static int daemonPort = -1;
|
|
+static char daemonStamp[256] = "";
|
|
+static jmp_buf env;
|
|
+
|
|
+static int daemonMime;
|
|
+
|
|
+static void daemon_define(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_match(const char *cmdline, int argc, const char **argv);
|
|
+static void daemon_show_db(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_show_strat(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+void daemon_show_info(const char *cmdline, int argc, const char **argv);
|
|
+static void daemon_show_server(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_show(const char *cmdline, int argc, const char **argv);
|
|
+static void daemon_option_mime(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_option(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_client(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_auth(const char *cmdline, int argc, const char **argv);
|
|
+static void daemon_status(const char *cmdline, int argc,
|
|
+ const char **argv);
|
|
+static void daemon_help(const char *cmdline, int argc, const char **argv);
|
|
+static void daemon_quit(const char *cmdline, int argc, const char **argv);
|
|
|
|
#define MAXARGCS 3
|
|
static struct {
|
|
- int argc;
|
|
- const char *name[MAXARGCS];
|
|
- void (*f)( const char *cmdline, int argc, const char **argv );
|
|
+ int argc;
|
|
+ const char *name[MAXARGCS];
|
|
+ void (*f) (const char *cmdline, int argc, const char **argv);
|
|
} commandInfo[] = {
|
|
- { 1, {"define"}, daemon_define },
|
|
- { 1, {"d"}, daemon_define },
|
|
- { 1, {"match"}, daemon_match },
|
|
- { 1, {"m"}, daemon_match },
|
|
- { 2, {"show", "db"}, daemon_show_db },
|
|
- { 2, {"show", "databases"}, daemon_show_db },
|
|
- { 2, {"show", "strat"}, daemon_show_strat },
|
|
- { 2, {"show", "strategies"}, daemon_show_strat },
|
|
- { 2, {"show", "info"}, daemon_show_info },
|
|
- { 2, {"show", "server"}, daemon_show_server },
|
|
- { 1, {"show"}, daemon_show },
|
|
- { 2, {"option", "mime"}, daemon_option_mime },
|
|
- { 1, {"option"}, daemon_option },
|
|
- { 1, {"client"}, daemon_client },
|
|
- { 1, {"auth"}, daemon_auth },
|
|
- { 1, {"status"}, daemon_status },
|
|
- { 1, {"s"}, daemon_status },
|
|
- { 1, {"help"}, daemon_help },
|
|
- { 1, {"h"}, daemon_help },
|
|
- { 1, {"quit"}, daemon_quit },
|
|
- { 1, {"q"}, daemon_quit },
|
|
-};
|
|
+ {
|
|
+ 1, {
|
|
+ "define"}, daemon_define}, {
|
|
+ 1, {
|
|
+ "d"}, daemon_define}, {
|
|
+ 1, {
|
|
+ "match"}, daemon_match}, {
|
|
+ 1, {
|
|
+ "m"}, daemon_match}, {
|
|
+ 2, {
|
|
+ "show", "db"}, daemon_show_db}, {
|
|
+ 2, {
|
|
+ "show", "databases"}, daemon_show_db}, {
|
|
+ 2, {
|
|
+ "show", "strat"}, daemon_show_strat}, {
|
|
+ 2, {
|
|
+ "show", "strategies"}, daemon_show_strat}, {
|
|
+ 2, {
|
|
+ "show", "info"}, daemon_show_info}, {
|
|
+ 2, {
|
|
+ "show", "server"}, daemon_show_server}, {
|
|
+ 1, {
|
|
+ "show"}, daemon_show}, {
|
|
+ 2, {
|
|
+ "option", "mime"}, daemon_option_mime}, {
|
|
+ 1, {
|
|
+ "option"}, daemon_option}, {
|
|
+ 1, {
|
|
+ "client"}, daemon_client}, {
|
|
+ 1, {
|
|
+ "auth"}, daemon_auth}, {
|
|
+ 1, {
|
|
+ "status"}, daemon_status}, {
|
|
+ 1, {
|
|
+ "s"}, daemon_status}, {
|
|
+ 1, {
|
|
+ "help"}, daemon_help}, {
|
|
+ 1, {
|
|
+ "h"}, daemon_help}, {
|
|
+ 1, {
|
|
+ "quit"}, daemon_quit}, {
|
|
+ 1, {
|
|
+"q"}, daemon_quit},};
|
|
+
|
|
#define COMMANDS (sizeof(commandInfo)/sizeof(commandInfo[0]))
|
|
|
|
-static void *(lookup_command)( int argc, const char **argv )
|
|
-{
|
|
- size_t i;
|
|
- int j;
|
|
- int err;
|
|
-
|
|
- for (i = 0; i < COMMANDS; i++) {
|
|
- if (argc >= commandInfo[i].argc) {
|
|
- for (err = 0, j = 0; j < commandInfo[i].argc; j++) {
|
|
- if (strcasecmp(argv[j], commandInfo[i].name[j])) {
|
|
- err = 1;
|
|
- break;
|
|
+static void *(lookup_command) (int argc, const char **argv) {
|
|
+ size_t i;
|
|
+ int j;
|
|
+ int err;
|
|
+
|
|
+ for (i = 0; i < COMMANDS; i++) {
|
|
+ if (argc >= commandInfo[i].argc) {
|
|
+ for (err = 0, j = 0; j < commandInfo[i].argc; j++) {
|
|
+ if (strcasecmp(argv[j], commandInfo[i].name[j])) {
|
|
+ err = 1;
|
|
+ break;
|
|
+ }
|
|
}
|
|
- }
|
|
- if (!err) return commandInfo[i].f;
|
|
- }
|
|
- }
|
|
- return NULL;
|
|
+ if (!err)
|
|
+ return commandInfo[i].f;
|
|
+ }
|
|
+ }
|
|
+ return NULL;
|
|
}
|
|
|
|
static unsigned long daemon_compute_mask(int bits)
|
|
{
|
|
- unsigned long mask = 0xffffffff;
|
|
-
|
|
- if (bits < 1) return 0;
|
|
- if (bits < 32) mask -= (1 << (32-bits)) - 1;
|
|
- return mask;
|
|
-}
|
|
-
|
|
-static void daemon_log( int type, const char *format, ... )
|
|
-{
|
|
- va_list ap;
|
|
- char buf[8*1024];
|
|
- char *buf2;
|
|
- size_t len;
|
|
- char *s, *d;
|
|
- int c;
|
|
- char marker = '?';
|
|
-
|
|
- switch (type) {
|
|
- case DICT_LOG_TERM:
|
|
- if (!flg_test(LOG_STATS)) return; marker = 'I'; break;
|
|
- case DICT_LOG_TRACE:
|
|
- if (!flg_test(LOG_SERVER)) return; marker = 'I'; break;
|
|
- case DICT_LOG_CLIENT:
|
|
- if (!flg_test(LOG_CLIENT)) return; marker = 'C'; break;
|
|
- case DICT_LOG_CONNECT:
|
|
- if (!flg_test(LOG_CONNECT)) return; marker = 'K'; break;
|
|
- case DICT_LOG_DEFINE:
|
|
- if (!flg_test(LOG_FOUND)) return; marker = 'D'; break;
|
|
- case DICT_LOG_MATCH:
|
|
- if (!flg_test(LOG_FOUND)) return; marker = 'M'; break;
|
|
- case DICT_LOG_NOMATCH:
|
|
- if (!flg_test(LOG_NOTFOUND)) return; marker = 'N'; break;
|
|
- case DICT_LOG_COMMAND:
|
|
- if (!flg_test(LOG_COMMAND)) return; marker = 'T'; break;
|
|
- case DICT_LOG_AUTH:
|
|
- if (!flg_test(LOG_AUTH)) return; marker = 'A'; break;
|
|
- }
|
|
-
|
|
- if (dbg_test(DBG_PORT))
|
|
- snprintf(
|
|
- buf, sizeof (buf)/2,
|
|
- ":%c: %s:%d ", marker, daemonHostname, daemonPort );
|
|
- else if (flg_test(LOG_HOST))
|
|
- snprintf(
|
|
- buf, sizeof (buf)/2,
|
|
- ":%c: %s ", marker, daemonHostname );
|
|
- else
|
|
- snprintf(
|
|
- buf, sizeof (buf)/2,
|
|
- ":%c: ", marker );
|
|
-
|
|
- len = strlen( buf );
|
|
-
|
|
- va_start( ap, format );
|
|
- vsnprintf( buf+len, sizeof (buf)/2-len, format, ap );
|
|
- va_end( ap );
|
|
- len = strlen( buf );
|
|
-
|
|
- if (len >= sizeof (buf)/2) {
|
|
- log_info( ":E: buffer overflow (%d)\n", len );
|
|
- buf[2048] = '\0';
|
|
- len = strlen(buf);
|
|
- }
|
|
-
|
|
- buf2 = alloca( 3*(len+3) );
|
|
-
|
|
- for (s = buf, d = buf2; *s; s++) {
|
|
- c = (unsigned char)*s;
|
|
- if (c == '\t') *d++ = ' ';
|
|
- else if (c == '\n') *d++ = c;
|
|
- else {
|
|
- if (c < 32) { *d++ = '^'; *d++ = c + 64; }
|
|
- else if (c == 127) { *d++ = '^'; *d++ = '?'; }
|
|
- else *d++ = c;
|
|
- }
|
|
- }
|
|
- *d = '\0';
|
|
- log_info( "%s", buf2 );
|
|
+ unsigned long mask = 0xffffffff;
|
|
|
|
- if (d != buf2) d[-1] = '\0'; /* kill newline */
|
|
- dict_setproctitle( "dictd %s", buf2 );
|
|
+ if (bits < 1)
|
|
+ return 0;
|
|
+ if (bits < 32)
|
|
+ mask -= (1 << (32 - bits)) - 1;
|
|
+ return mask;
|
|
+}
|
|
+
|
|
+static void daemon_log(int type, const char *format, ...)
|
|
+{
|
|
+ va_list ap;
|
|
+ char buf[8 * 1024];
|
|
+ char *buf2;
|
|
+ size_t len;
|
|
+ char *s, *d;
|
|
+ int c;
|
|
+ char marker = '?';
|
|
+
|
|
+ switch (type) {
|
|
+ case DICT_LOG_TERM:
|
|
+ if (!flg_test(LOG_STATS))
|
|
+ return;
|
|
+ marker = 'I';
|
|
+ break;
|
|
+ case DICT_LOG_TRACE:
|
|
+ if (!flg_test(LOG_SERVER))
|
|
+ return;
|
|
+ marker = 'I';
|
|
+ break;
|
|
+ case DICT_LOG_CLIENT:
|
|
+ if (!flg_test(LOG_CLIENT))
|
|
+ return;
|
|
+ marker = 'C';
|
|
+ break;
|
|
+ case DICT_LOG_CONNECT:
|
|
+ if (!flg_test(LOG_CONNECT))
|
|
+ return;
|
|
+ marker = 'K';
|
|
+ break;
|
|
+ case DICT_LOG_DEFINE:
|
|
+ if (!flg_test(LOG_FOUND))
|
|
+ return;
|
|
+ marker = 'D';
|
|
+ break;
|
|
+ case DICT_LOG_MATCH:
|
|
+ if (!flg_test(LOG_FOUND))
|
|
+ return;
|
|
+ marker = 'M';
|
|
+ break;
|
|
+ case DICT_LOG_NOMATCH:
|
|
+ if (!flg_test(LOG_NOTFOUND))
|
|
+ return;
|
|
+ marker = 'N';
|
|
+ break;
|
|
+ case DICT_LOG_COMMAND:
|
|
+ if (!flg_test(LOG_COMMAND))
|
|
+ return;
|
|
+ marker = 'T';
|
|
+ break;
|
|
+ case DICT_LOG_AUTH:
|
|
+ if (!flg_test(LOG_AUTH))
|
|
+ return;
|
|
+ marker = 'A';
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ if (dbg_test(DBG_PORT))
|
|
+ snprintf(buf, sizeof(buf) / 2,
|
|
+ ":%c: %s:%d ", marker, daemonHostname, daemonPort);
|
|
+ else if (flg_test(LOG_HOST))
|
|
+ snprintf(buf, sizeof(buf) / 2, ":%c: %s ", marker, daemonHostname);
|
|
+ else
|
|
+ snprintf(buf, sizeof(buf) / 2, ":%c: ", marker);
|
|
+
|
|
+ len = strlen(buf);
|
|
+
|
|
+ va_start(ap, format);
|
|
+ vsnprintf(buf + len, sizeof(buf) / 2 - len, format, ap);
|
|
+ va_end(ap);
|
|
+ len = strlen(buf);
|
|
+
|
|
+ if (len >= sizeof(buf) / 2) {
|
|
+ log_info(":E: buffer overflow (%d)\n", len);
|
|
+ buf[2048] = '\0';
|
|
+ len = strlen(buf);
|
|
+ }
|
|
+
|
|
+ buf2 = alloca(3 * (len + 3));
|
|
+
|
|
+ for (s = buf, d = buf2; *s; s++) {
|
|
+ c = (unsigned char) *s;
|
|
+ if (c == '\t')
|
|
+ *d++ = ' ';
|
|
+ else if (c == '\n')
|
|
+ *d++ = c;
|
|
+ else {
|
|
+ if (c < 32) {
|
|
+ *d++ = '^';
|
|
+ *d++ = c + 64;
|
|
+ } else if (c == 127) {
|
|
+ *d++ = '^';
|
|
+ *d++ = '?';
|
|
+ } else
|
|
+ *d++ = c;
|
|
+ }
|
|
+ }
|
|
+ *d = '\0';
|
|
+ log_info("%s", buf2);
|
|
+
|
|
+ if (d != buf2)
|
|
+ d[-1] = '\0'; /* kill newline */
|
|
+ dict_setproctitle("dictd %s", buf2);
|
|
}
|
|
|
|
static int daemon_check_mask(const char *spec, const char *ip)
|
|
{
|
|
- char *tmp = alloca(strlen(spec) + 1);
|
|
- char *pt;
|
|
- char tstring[64], mstring[64];
|
|
- struct in_addr target, mask;
|
|
- int bits;
|
|
- unsigned long bitmask;
|
|
-
|
|
- strcpy(tmp, spec);
|
|
- if (!(pt = strchr(tmp, '/'))) {
|
|
- log_info( ":E: No / in %s, denying access to %s\n", spec, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
- *pt++ = '\0';
|
|
- if (!*pt) {
|
|
- log_info( ":E: No bit count after / in %s, denying access to %s\n",
|
|
- spec, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
-
|
|
- inet_aton(ip, &target);
|
|
- inet_aton(tmp, &mask);
|
|
- bits = strtol(pt, NULL, 10);
|
|
- strcpy(tstring, inet_ntoa(target));
|
|
- strcpy(mstring, inet_ntoa(mask));
|
|
- if (bits < 0 || bits > 32) {
|
|
- log_info( ":E: Bit count (%d) out of range, denying access to %s\n",
|
|
- bits, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
-
|
|
- bitmask = daemon_compute_mask(bits);
|
|
- if ((ntohl(target.s_addr) & bitmask) == (ntohl(mask.s_addr) & bitmask)) {
|
|
- PRINTF(DBG_AUTH, ("%s matches %s/%d\n", tstring, mstring, bits));
|
|
- return DICT_MATCH;
|
|
- }
|
|
- PRINTF(DBG_AUTH, ("%s does NOT match %s/%d\n", tstring, mstring, bits));
|
|
- return DICT_NOMATCH;
|
|
+ char *tmp = alloca(strlen(spec) + 1);
|
|
+ char *pt;
|
|
+ char tstring[64], mstring[64];
|
|
+ struct in_addr target, mask;
|
|
+ int bits;
|
|
+ unsigned long bitmask;
|
|
+
|
|
+ strcpy(tmp, spec);
|
|
+ if (!(pt = strchr(tmp, '/'))) {
|
|
+ log_info(":E: No / in %s, denying access to %s\n", spec, ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+ *pt++ = '\0';
|
|
+ if (!*pt) {
|
|
+ log_info(":E: No bit count after / in %s, denying access to %s\n",
|
|
+ spec, ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+
|
|
+ inet_aton(ip, &target);
|
|
+ inet_aton(tmp, &mask);
|
|
+ bits = strtol(pt, NULL, 10);
|
|
+ strcpy(tstring, inet_ntoa(target));
|
|
+ strcpy(mstring, inet_ntoa(mask));
|
|
+ if (bits < 0 || bits > 32) {
|
|
+ log_info(":E: Bit count (%d) out of range, denying access to %s\n",
|
|
+ bits, ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+
|
|
+ bitmask = daemon_compute_mask(bits);
|
|
+ if ((ntohl(target.s_addr) & bitmask) == (ntohl(mask.s_addr) & bitmask)) {
|
|
+ PRINTF(DBG_AUTH, ("%s matches %s/%d\n", tstring, mstring, bits));
|
|
+ return DICT_MATCH;
|
|
+ }
|
|
+ PRINTF(DBG_AUTH,
|
|
+ ("%s does NOT match %s/%d\n", tstring, mstring, bits));
|
|
+ return DICT_NOMATCH;
|
|
}
|
|
|
|
static int daemon_check_range(const char *spec, const char *ip)
|
|
{
|
|
- char *tmp = alloca(strlen(spec) + 1);
|
|
- char *pt;
|
|
- char tstring[64], minstring[64], maxstring[64];
|
|
- struct in_addr target, min, max;
|
|
-
|
|
- strcpy(tmp, spec);
|
|
- if (!(pt = strchr(tmp, ':'))) {
|
|
- log_info( ":E: No : in range %s, denying access to %s\n", spec, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
- *pt++ = '\0';
|
|
- if (strchr(pt, ':')) {
|
|
- log_info( ":E: More than one : in range %s, denying access to %s\n",
|
|
- spec, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
- if (!*pt) {
|
|
- log_info( ":E: Misformed range %s, denying access to %s\n", spec, ip);
|
|
- return DICT_DENY;
|
|
- }
|
|
-
|
|
- inet_aton(ip, &target);
|
|
- inet_aton(tmp, &min);
|
|
- inet_aton(pt, &max);
|
|
- strcpy(tstring, inet_ntoa(target));
|
|
- strcpy(minstring, inet_ntoa(min));
|
|
- strcpy(maxstring, inet_ntoa(max));
|
|
- if (ntohl(target.s_addr) >= ntohl(min.s_addr)
|
|
- && ntohl(target.s_addr) <= ntohl(max.s_addr)) {
|
|
- PRINTF(DBG_AUTH,("%s in range from %s to %s\n",
|
|
- tstring, minstring, maxstring));
|
|
- return DICT_MATCH;
|
|
- }
|
|
- PRINTF(DBG_AUTH,("%s NOT in range from %s to %s\n",
|
|
- tstring, minstring, maxstring));
|
|
- return DICT_NOMATCH;
|
|
+ char *tmp = alloca(strlen(spec) + 1);
|
|
+ char *pt;
|
|
+ char tstring[64], minstring[64], maxstring[64];
|
|
+ struct in_addr target, min, max;
|
|
+
|
|
+ strcpy(tmp, spec);
|
|
+ if (!(pt = strchr(tmp, ':'))) {
|
|
+ log_info(":E: No : in range %s, denying access to %s\n", spec, ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+ *pt++ = '\0';
|
|
+ if (strchr(pt, ':')) {
|
|
+ log_info(":E: More than one : in range %s, denying access to %s\n",
|
|
+ spec, ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+ if (!*pt) {
|
|
+ log_info(":E: Misformed range %s, denying access to %s\n", spec,
|
|
+ ip);
|
|
+ return DICT_DENY;
|
|
+ }
|
|
+
|
|
+ inet_aton(ip, &target);
|
|
+ inet_aton(tmp, &min);
|
|
+ inet_aton(pt, &max);
|
|
+ strcpy(tstring, inet_ntoa(target));
|
|
+ strcpy(minstring, inet_ntoa(min));
|
|
+ strcpy(maxstring, inet_ntoa(max));
|
|
+ if (ntohl(target.s_addr) >= ntohl(min.s_addr)
|
|
+ && ntohl(target.s_addr) <= ntohl(max.s_addr)) {
|
|
+ PRINTF(DBG_AUTH, ("%s in range from %s to %s\n",
|
|
+ tstring, minstring, maxstring));
|
|
+ return DICT_MATCH;
|
|
+ }
|
|
+ PRINTF(DBG_AUTH, ("%s NOT in range from %s to %s\n",
|
|
+ tstring, minstring, maxstring));
|
|
+ return DICT_NOMATCH;
|
|
}
|
|
|
|
static int daemon_check_wildcard(const char *spec, const char *ip)
|
|
{
|
|
- char regbuf[256];
|
|
- char erbuf[100];
|
|
- int err;
|
|
- const char *s;
|
|
- char *d;
|
|
- regex_t re;
|
|
-
|
|
- for (d = regbuf, s = spec; s && *s; ++s) {
|
|
- switch (*s) {
|
|
- case '*': *d++ = '.'; *d++ = '*'; break;
|
|
- case '.': *d++ = '\\'; *d++ = '.'; break;
|
|
- case '?': *d++ = '.'; break;
|
|
- default: *d++ = *s; break;
|
|
- }
|
|
- }
|
|
- *d = '\0';
|
|
- if ((err = regcomp(&re, regbuf, REG_ICASE|REG_NOSUB))) {
|
|
- regerror(err, &re, erbuf, sizeof(erbuf));
|
|
- log_info( ":E: regcomp(%s): %s\n", regbuf, erbuf );
|
|
- return DICT_DENY; /* Err on the side of safety */
|
|
- }
|
|
- if (!regexec(&re, daemonHostname, 0, NULL, 0)
|
|
- || !regexec(&re, daemonIP, 0, NULL, 0)) {
|
|
- PRINTF(DBG_AUTH,
|
|
- ("Match %s with %s/%s\n", spec, daemonHostname, daemonIP));
|
|
- regfree(&re);
|
|
- return DICT_MATCH;
|
|
- }
|
|
- regfree(&re);
|
|
- PRINTF(DBG_AUTH,
|
|
- ("No match (%s with %s/%s)\n", spec, daemonHostname, daemonIP));
|
|
- return DICT_NOMATCH;
|
|
-}
|
|
-
|
|
-static int daemon_check_list( const char *user, lst_List acl )
|
|
-{
|
|
- lst_Position p;
|
|
- dictAccess *a;
|
|
- int retcode;
|
|
-
|
|
- if (!acl)
|
|
- return DICT_ALLOW;
|
|
-
|
|
- for (p = lst_init_position(acl); p; p = lst_next_position(p)) {
|
|
- a = lst_get_position(p);
|
|
- switch (a->type) {
|
|
- case DICT_DENY:
|
|
- case DICT_ALLOW:
|
|
- case DICT_AUTHONLY:
|
|
- if (strchr(a->spec, '/'))
|
|
- retcode = daemon_check_mask(a->spec, daemonIP);
|
|
- else if (strchr(a->spec, ':'))
|
|
- retcode = daemon_check_range(a->spec, daemonIP);
|
|
- else
|
|
- retcode = daemon_check_wildcard(a->spec, daemonIP);
|
|
- switch (retcode) {
|
|
- case DICT_DENY:
|
|
- return DICT_DENY;
|
|
- case DICT_MATCH:
|
|
- if (a->type == DICT_DENY) {
|
|
- daemon_log( DICT_LOG_AUTH, "spec %s denies %s/%s\n",
|
|
- a->spec, daemonHostname, daemonIP);
|
|
- }
|
|
- return a->type;
|
|
- }
|
|
- break;
|
|
- case DICT_USER:
|
|
- if (user && !strcmp(user,a->spec)) return DICT_ALLOW;
|
|
- case DICT_GROUP: /* Groups are not yet implemented. */
|
|
- break;
|
|
- }
|
|
- }
|
|
- return DICT_DENY;
|
|
-}
|
|
-
|
|
-static int daemon_check_auth( const char *user )
|
|
-{
|
|
- lst_Position p;
|
|
- lst_List dbl = DictConfig->dbl;
|
|
- dictDatabase *db;
|
|
-
|
|
- switch (daemon_check_list( user, DictConfig->acl )) {
|
|
- default:
|
|
- case DICT_DENY:
|
|
- return 1;
|
|
- case DICT_AUTHONLY:
|
|
- if (!user) return 0;
|
|
- case DICT_ALLOW:
|
|
- for (p = lst_init_position(dbl); p; p = lst_next_position(p)) {
|
|
- db = lst_get_position(p);
|
|
- switch (daemon_check_list(user, db->acl)) {
|
|
- case DICT_ALLOW: db->available = 1; continue;
|
|
- default: db->available = 0; continue;
|
|
- }
|
|
- }
|
|
- break;
|
|
- }
|
|
- return 0;
|
|
-}
|
|
-
|
|
-void daemon_terminate( int sig, const char *name )
|
|
-{
|
|
- alarm(0);
|
|
- tim_stop( "t" );
|
|
- close(daemonS_in);
|
|
- close(daemonS_out);
|
|
- if (name) {
|
|
- daemon_log( DICT_LOG_TERM,
|
|
- "%s: d/m/c = %d/%d/%d; %sr %su %ss\n",
|
|
- name,
|
|
- _dict_defines,
|
|
- _dict_matches,
|
|
- _dict_comparisons,
|
|
- dict_format_time( tim_get_real( "t" ) ),
|
|
- dict_format_time( tim_get_user( "t" ) ),
|
|
- dict_format_time( tim_get_system( "t" ) ) );
|
|
- } else {
|
|
- daemon_log( DICT_LOG_TERM,
|
|
- "signal %d: d/m/c = %d/%d/%d; %sr %su %ss\n",
|
|
- sig,
|
|
- _dict_defines,
|
|
- _dict_matches,
|
|
- _dict_comparisons,
|
|
- dict_format_time( tim_get_real( "t" ) ),
|
|
- dict_format_time( tim_get_user( "t" ) ),
|
|
- dict_format_time( tim_get_system( "t" ) ) );
|
|
- }
|
|
-
|
|
- log_close();
|
|
- longjmp(env,1);
|
|
- if (sig) exit(sig+128);
|
|
-
|
|
- exit(0);
|
|
-}
|
|
-
|
|
-
|
|
-static void daemon_write( const char *buf, int len )
|
|
-{
|
|
- int left = len;
|
|
- int count;
|
|
-
|
|
- while (left) {
|
|
- if ((count = write(daemonS_out, buf, left)) != left) {
|
|
- if (count <= 0) {
|
|
- if (errno == EPIPE) {
|
|
- daemon_terminate( 0, "pipe" );
|
|
+ char regbuf[256];
|
|
+ char erbuf[100];
|
|
+ int err;
|
|
+ const char *s;
|
|
+ char *d;
|
|
+ regex_t re;
|
|
+
|
|
+ for (d = regbuf, s = spec; s && *s; ++s) {
|
|
+ switch (*s) {
|
|
+ case '*':
|
|
+ *d++ = '.';
|
|
+ *d++ = '*';
|
|
+ break;
|
|
+ case '.':
|
|
+ *d++ = '\\';
|
|
+ *d++ = '.';
|
|
+ break;
|
|
+ case '?':
|
|
+ *d++ = '.';
|
|
+ break;
|
|
+ default:
|
|
+ *d++ = *s;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+ *d = '\0';
|
|
+ if ((err = regcomp(&re, regbuf, REG_ICASE | REG_NOSUB))) {
|
|
+ regerror(err, &re, erbuf, sizeof(erbuf));
|
|
+ log_info(":E: regcomp(%s): %s\n", regbuf, erbuf);
|
|
+ return DICT_DENY; /* Err on the side of safety */
|
|
+ }
|
|
+ if (!regexec(&re, daemonHostname, 0, NULL, 0)
|
|
+ || !regexec(&re, daemonIP, 0, NULL, 0)) {
|
|
+ PRINTF(DBG_AUTH,
|
|
+ ("Match %s with %s/%s\n", spec, daemonHostname, daemonIP));
|
|
+ regfree(&re);
|
|
+ return DICT_MATCH;
|
|
+ }
|
|
+ regfree(&re);
|
|
+ PRINTF(DBG_AUTH,
|
|
+ ("No match (%s with %s/%s)\n", spec, daemonHostname, daemonIP));
|
|
+ return DICT_NOMATCH;
|
|
+}
|
|
+
|
|
+static int daemon_check_list(const char *user, lst_List acl)
|
|
+{
|
|
+ lst_Position p;
|
|
+ dictAccess *a;
|
|
+ int retcode;
|
|
+
|
|
+ if (!acl)
|
|
+ return DICT_ALLOW;
|
|
+
|
|
+ for (p = lst_init_position(acl); p; p = lst_next_position(p)) {
|
|
+ a = lst_get_position(p);
|
|
+ switch (a->type) {
|
|
+ case DICT_DENY:
|
|
+ case DICT_ALLOW:
|
|
+ case DICT_AUTHONLY:
|
|
+ if (strchr(a->spec, '/'))
|
|
+ retcode = daemon_check_mask(a->spec, daemonIP);
|
|
+ else if (strchr(a->spec, ':'))
|
|
+ retcode = daemon_check_range(a->spec, daemonIP);
|
|
+ else
|
|
+ retcode = daemon_check_wildcard(a->spec, daemonIP);
|
|
+ switch (retcode) {
|
|
+ case DICT_DENY:
|
|
+ return DICT_DENY;
|
|
+ case DICT_MATCH:
|
|
+ if (a->type == DICT_DENY) {
|
|
+ daemon_log(DICT_LOG_AUTH, "spec %s denies %s/%s\n",
|
|
+ a->spec, daemonHostname, daemonIP);
|
|
+ }
|
|
+ return a->type;
|
|
+ }
|
|
+ break;
|
|
+ case DICT_USER:
|
|
+ if (user && !strcmp(user, a->spec))
|
|
+ return DICT_ALLOW;
|
|
+ case DICT_GROUP: /* Groups are not yet implemented. */
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+ return DICT_DENY;
|
|
+}
|
|
+
|
|
+static int daemon_check_auth(const char *user)
|
|
+{
|
|
+ lst_Position p;
|
|
+ lst_List dbl = DictConfig->dbl;
|
|
+ dictDatabase *db;
|
|
+
|
|
+ switch (daemon_check_list(user, DictConfig->acl)) {
|
|
+ default:
|
|
+ case DICT_DENY:
|
|
+ return 1;
|
|
+ case DICT_AUTHONLY:
|
|
+ if (!user)
|
|
+ return 0;
|
|
+ case DICT_ALLOW:
|
|
+ for (p = lst_init_position(dbl); p; p = lst_next_position(p)) {
|
|
+ db = lst_get_position(p);
|
|
+ switch (daemon_check_list(user, db->acl)) {
|
|
+ case DICT_ALLOW:
|
|
+ db->available = 1;
|
|
+ continue;
|
|
+ default:
|
|
+ db->available = 0;
|
|
+ continue;
|
|
}
|
|
+ }
|
|
+ break;
|
|
+ }
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+void daemon_terminate(int sig, const char *name)
|
|
+{
|
|
+ alarm(0);
|
|
+ tim_stop("t");
|
|
+ close(daemonS_in);
|
|
+ close(daemonS_out);
|
|
+ if (name) {
|
|
+ daemon_log(DICT_LOG_TERM,
|
|
+ "%s: d/m/c = %d/%d/%d; %sr %su %ss\n",
|
|
+ name,
|
|
+ _dict_defines,
|
|
+ _dict_matches,
|
|
+ _dict_comparisons,
|
|
+ dict_format_time(tim_get_real("t")),
|
|
+ dict_format_time(tim_get_user("t")),
|
|
+ dict_format_time(tim_get_system("t")));
|
|
+ } else {
|
|
+ daemon_log(DICT_LOG_TERM,
|
|
+ "signal %d: d/m/c = %d/%d/%d; %sr %su %ss\n",
|
|
+ sig,
|
|
+ _dict_defines,
|
|
+ _dict_matches,
|
|
+ _dict_comparisons,
|
|
+ dict_format_time(tim_get_real("t")),
|
|
+ dict_format_time(tim_get_user("t")),
|
|
+ dict_format_time(tim_get_system("t")));
|
|
+ }
|
|
+
|
|
+ log_close();
|
|
+ longjmp(env, 1);
|
|
+ if (sig)
|
|
+ exit(sig + 128);
|
|
+
|
|
+ exit(0);
|
|
+}
|
|
+
|
|
+
|
|
+static void daemon_write(const char *buf, int len)
|
|
+{
|
|
+ int left = len;
|
|
+ int count;
|
|
+
|
|
+ while (left) {
|
|
+ if ((count = write(daemonS_out, buf, left)) != left) {
|
|
+ if (count <= 0) {
|
|
+ if (errno == EPIPE) {
|
|
+ daemon_terminate(0, "pipe");
|
|
+ }
|
|
#if HAVE_STRERROR
|
|
- log_info( ":E: writing %d of %d bytes:"
|
|
- " retval = %d, errno = %d (%s)\n",
|
|
- left, len, count, errno, strerror(errno) );
|
|
+ log_info(":E: writing %d of %d bytes:"
|
|
+ " retval = %d, errno = %d (%s)\n",
|
|
+ left, len, count, errno, strerror(errno));
|
|
#else
|
|
- log_info( ":E: writing %d of %d bytes:"
|
|
- " retval = %d, errno = %d\n",
|
|
- left, len, count, errno );
|
|
+ log_info(":E: writing %d of %d bytes:"
|
|
+ " retval = %d, errno = %d\n",
|
|
+ left, len, count, errno);
|
|
#endif
|
|
- daemon_terminate( 0, __func__ );
|
|
- }
|
|
- }
|
|
- left -= count;
|
|
- }
|
|
-}
|
|
-
|
|
-static void daemon_crlf( char *d, const char *s, int dot )
|
|
-{
|
|
- int first = 1;
|
|
-
|
|
- while (*s) {
|
|
- if (*s == '\n') {
|
|
- *d++ = '\r';
|
|
- *d++ = '\n';
|
|
- first = 1;
|
|
- ++s;
|
|
- } else {
|
|
- if (dot && first && *s == '.' && s[1] == '\n')
|
|
- *d++ = '.'; /* double first dot on line */
|
|
- first = 0;
|
|
- *d++ = *s++;
|
|
- }
|
|
- }
|
|
- if (dot) { /* add final . */
|
|
- if (!first){
|
|
- *d++ = '\r';
|
|
- *d++ = '\n';
|
|
- }
|
|
- *d++ = '.';
|
|
- *d++ = '\r';
|
|
- *d++ = '\n';
|
|
- }
|
|
- *d = '\0';
|
|
-}
|
|
-
|
|
-static void daemon_printf( const char *format, ... )
|
|
-{
|
|
- va_list ap;
|
|
- char buf[BUFFERSIZE];
|
|
- char *pt;
|
|
- int len;
|
|
-
|
|
- va_start( ap, format );
|
|
- vsnprintf( buf, sizeof (buf), format, ap );
|
|
- va_end( ap );
|
|
- if ((len = strlen( buf )) >= BUFFERSIZE) {
|
|
- log_info( ":E: buffer overflow: %d\n", len );
|
|
- daemon_terminate( 0, __func__ );
|
|
- }
|
|
-
|
|
- pt = alloca(2*len + 10); /* +10 for the case when buf == "\n"*/
|
|
- daemon_crlf(pt, buf, 0);
|
|
- daemon_write(pt, strlen(pt));
|
|
-}
|
|
-
|
|
-static void daemon_mime( void )
|
|
-{
|
|
- if (daemonMime) daemon_write( "\r\n", 2 );
|
|
-}
|
|
-
|
|
-static void daemon_mime_definition (const dictDatabase *db)
|
|
-{
|
|
- if (daemonMime){
|
|
- if (db -> mime_header){
|
|
- daemon_printf ("%s", db -> mime_header);
|
|
- }
|
|
-
|
|
- daemon_write ("\r\n", 2);
|
|
- }
|
|
-}
|
|
-
|
|
-static void daemon_text( const char *text, int dot )
|
|
-{
|
|
- char *pt = alloca( 2*strlen(text) + 10 );
|
|
-
|
|
- daemon_crlf(pt, text, dot);
|
|
- daemon_write(pt, strlen(pt));
|
|
-}
|
|
-
|
|
-static int daemon_read( char *buf, int count )
|
|
-{
|
|
- return net_read( daemonS_in, buf, count );
|
|
-}
|
|
-
|
|
-static void daemon_ok( int code, const char *string, const char *timer )
|
|
-{
|
|
- static int lastDefines = 0;
|
|
- static int lastMatches = 0;
|
|
- static int lastComparisons = 0;
|
|
-
|
|
- if (code == CODE_STATUS) {
|
|
- lastDefines = 0;
|
|
- lastMatches = 0;
|
|
- lastComparisons = 0;
|
|
- }
|
|
-
|
|
- if (!timer) {
|
|
- daemon_printf("%d %s\n", code, string);
|
|
- } else {
|
|
- tim_stop( timer );
|
|
- daemon_printf("%d %s [d/m/c = %d/%d/%d; %sr %su %ss]\n",
|
|
- code,
|
|
- string,
|
|
- _dict_defines - lastDefines,
|
|
- _dict_matches - lastMatches,
|
|
- _dict_comparisons - lastComparisons,
|
|
- dict_format_time( tim_get_real( timer ) ),
|
|
- dict_format_time( tim_get_user( timer ) ),
|
|
- dict_format_time( tim_get_system( timer ) ) );
|
|
- }
|
|
-
|
|
- lastDefines = _dict_defines;
|
|
- lastMatches = _dict_matches;
|
|
- lastComparisons = _dict_comparisons;
|
|
-}
|
|
-
|
|
-static int daemon_count_defs( lst_List list )
|
|
-{
|
|
- lst_Position p;
|
|
- dictWord *dw;
|
|
- unsigned long previousStart = 0;
|
|
- unsigned long previousEnd = 0;
|
|
- const char *previousDef = NULL;
|
|
- int count = 0;
|
|
-
|
|
- LST_ITERATE(list,p,dw) {
|
|
- if (
|
|
- previousStart == dw->start &&
|
|
- previousEnd == dw->end &&
|
|
- previousDef == dw->def)
|
|
- {
|
|
- continue;
|
|
- }
|
|
-
|
|
- previousStart = dw->start;
|
|
- previousEnd = dw->end;
|
|
- previousDef = dw->def;
|
|
-
|
|
- ++count;
|
|
- }
|
|
- return count;
|
|
-}
|
|
-
|
|
-static void daemon_dump_defs( lst_List list )
|
|
-{
|
|
- lst_Position p;
|
|
- char *buf;
|
|
- dictWord *dw;
|
|
- const dictDatabase *db = NULL;
|
|
- const dictDatabase *db_visible = NULL;
|
|
- unsigned long previousStart = 0;
|
|
- unsigned long previousEnd = 0;
|
|
- const char * previousDef = NULL;
|
|
- int count;
|
|
-
|
|
- LST_ITERATE(list,p,dw) {
|
|
- db = dw->database;
|
|
-
|
|
- if (
|
|
- previousStart == dw->start &&
|
|
- previousEnd == dw->end &&
|
|
- previousDef == dw->def)
|
|
- {
|
|
- continue;
|
|
- }
|
|
-
|
|
- previousStart = dw->start;
|
|
- previousEnd = dw->end;
|
|
- previousDef = dw->def;
|
|
-
|
|
- buf = dict_data_obtain ( db, dw );
|
|
-
|
|
- if (dw -> database_visible){
|
|
- db_visible = dw -> database_visible;
|
|
- }else{
|
|
- db_visible = db;
|
|
- }
|
|
-
|
|
- daemon_printf (
|
|
- "%d \"%s\" %s \"%s\"\n",
|
|
- CODE_DEFINITION_FOLLOWS,
|
|
- dw->word,
|
|
- db_visible -> invisible ? "*" : db_visible -> databaseName,
|
|
- db_visible -> invisible ? "" : db_visible -> databaseShort);
|
|
-
|
|
- daemon_mime_definition (db);
|
|
-
|
|
- if (db->filter){
|
|
- count = strlen(buf);
|
|
- daemon_log( DICT_LOG_AUTH, "filtering with: %s\ncount: %d\n",
|
|
- db->filter, count );
|
|
-
|
|
- dict_data_filter(buf, &count, strlen (buf), db->filter);
|
|
- buf[count] = '\0';
|
|
- }
|
|
-
|
|
- daemon_text(buf, 1);
|
|
- xfree( buf );
|
|
- }
|
|
-}
|
|
-
|
|
-static int daemon_count_matches( lst_List list )
|
|
-{
|
|
- lst_Position p;
|
|
- dictWord *dw;
|
|
- const char *prevword = NULL;
|
|
- const dictDatabase *prevdb = NULL;
|
|
- int count = 0;
|
|
-
|
|
- LST_ITERATE(list,p,dw) {
|
|
- if (prevdb == dw->database && prevword && !strcmp(prevword,dw->word))
|
|
- continue;
|
|
-
|
|
- prevword = dw->word;
|
|
- prevdb = dw->database;
|
|
-
|
|
- ++count;
|
|
- }
|
|
- return count;
|
|
-}
|
|
-
|
|
-static void daemon_dump_matches( lst_List list )
|
|
-{
|
|
- lst_Position p;
|
|
- dictWord *dw;
|
|
- const char *prevword = NULL;
|
|
- const dictDatabase *prevdb = NULL;
|
|
- const dictDatabase *db = NULL;
|
|
-
|
|
- daemon_mime();
|
|
- LST_ITERATE(list,p,dw) {
|
|
- db = dw -> database;
|
|
-
|
|
- if (prevdb == dw->database && prevword && !strcmp(prevword,dw->word))
|
|
- continue;
|
|
-
|
|
- prevword = dw->word;
|
|
- prevdb = dw->database;
|
|
-
|
|
- if (dw -> database_visible){
|
|
- db = dw -> database_visible;
|
|
- }
|
|
-
|
|
- daemon_printf (
|
|
- "%s \"%s\"\n",
|
|
- db -> invisible ? "*" : db -> databaseName,
|
|
- dw -> word );
|
|
- }
|
|
- daemon_printf( ".\n" );
|
|
-}
|
|
-
|
|
-static void daemon_banner( void )
|
|
-{
|
|
- time_t t;
|
|
-
|
|
- time(&t);
|
|
-
|
|
- snprintf( daemonStamp, sizeof (daemonStamp), "<%d.%d.%lu@%s>",
|
|
- _dict_forks,
|
|
- (int) getpid(),
|
|
- (long unsigned)t,
|
|
- net_hostname() );
|
|
- daemon_printf( "%d %s %s <auth.mime> %s\n",
|
|
- CODE_HELLO,
|
|
- net_hostname(),
|
|
- dict_get_banner(0),
|
|
- daemonStamp );
|
|
-}
|
|
-
|
|
-static void daemon_define( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- lst_List list = lst_create();
|
|
- int matches = 0;
|
|
- const char *word;
|
|
- const char *databaseName;
|
|
- int extension = (argv[0][0] == 'd' && argv[0][1] == '\0');
|
|
- int db_found = 0;
|
|
-
|
|
- if (extension) {
|
|
- switch (argc) {
|
|
- case 2: databaseName = "*"; word = argv[1]; break;
|
|
- case 3: databaseName = argv[1]; word = argv[2]; break;
|
|
- default:
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
- } else if (argc == 3) {
|
|
- databaseName = argv[1];
|
|
- word = argv[2];
|
|
- } else {
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
-
|
|
- matches = abs(dict_search_databases (
|
|
- list, NULL,
|
|
- databaseName, word, DICT_STRAT_EXACT,
|
|
- &db_found));
|
|
-
|
|
- if (db_found && matches > 0) {
|
|
- int actual_matches = daemon_count_defs( list );
|
|
-
|
|
- _dict_defines += actual_matches;
|
|
- daemon_log( DICT_LOG_DEFINE,
|
|
- "%s \"%s\" %d\n", databaseName, word, actual_matches);
|
|
- daemon_printf( "%d %d definitions retrieved\n",
|
|
- CODE_DEFINITIONS_FOUND,
|
|
- actual_matches );
|
|
- daemon_dump_defs( list );
|
|
- daemon_ok( CODE_OK, "ok", "c" );
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- return;
|
|
- }
|
|
+ daemon_terminate(0, __func__);
|
|
+ }
|
|
+ }
|
|
+ left -= count;
|
|
+ }
|
|
+}
|
|
|
|
- if (!db_found) {
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- daemon_printf( "%d invalid database, use SHOW DB for list\n",
|
|
- CODE_INVALID_DB );
|
|
- return;
|
|
- }
|
|
+static void daemon_crlf(char *d, const char *s, int dot)
|
|
+{
|
|
+ int first = 1;
|
|
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- daemon_log( DICT_LOG_NOMATCH,
|
|
- "%s exact \"%s\"\n", databaseName, word );
|
|
- daemon_ok( CODE_NO_MATCH, "no match", "c" );
|
|
-}
|
|
-
|
|
-static void daemon_match( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- lst_List list = lst_create();
|
|
- int matches = 0;
|
|
- const char *word;
|
|
- const char *databaseName;
|
|
- const char *strategy;
|
|
- int strategyNumber;
|
|
- int extension = (argv[0][0] == 'm' && argv[0][1] == '\0');
|
|
- int db_found = 0;
|
|
-
|
|
- if (extension) {
|
|
- switch (argc) {
|
|
- case 2:databaseName = "*"; strategy = "."; word = argv[1]; break;
|
|
- case 3:databaseName = "*"; strategy = argv[1]; word = argv[2]; break;
|
|
- case 4:databaseName = argv[1]; strategy = argv[2]; word = argv[3]; break;
|
|
- default:
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
- } else if (argc == 4) {
|
|
- databaseName = argv[1];
|
|
- strategy = argv[2];
|
|
- word = argv[3];
|
|
- } else {
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
-
|
|
- if ((strategyNumber = lookup_strategy(strategy)) < 0) {
|
|
- daemon_printf( "%d invalid strategy, use SHOW STRAT for a list\n",
|
|
- CODE_INVALID_STRATEGY );
|
|
- return;
|
|
- }
|
|
-
|
|
- matches = abs(dict_search_databases (
|
|
- list, NULL,
|
|
- databaseName, word, strategyNumber | DICT_MATCH_MASK,
|
|
- &db_found));
|
|
-
|
|
- if (db_found && matches > 0) {
|
|
- int actual_matches = daemon_count_matches( list );
|
|
-
|
|
- _dict_matches += actual_matches;
|
|
- daemon_log( DICT_LOG_MATCH,
|
|
- "%s %s \"%s\" %d\n",
|
|
- databaseName, strategy, word, actual_matches);
|
|
- daemon_printf( "%d %d matches found\n",
|
|
- CODE_MATCHES_FOUND, actual_matches );
|
|
- daemon_dump_matches( list );
|
|
- daemon_ok( CODE_OK, "ok", "c" );
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- return;
|
|
- }
|
|
+ while (*s) {
|
|
+ if (*s == '\n') {
|
|
+ *d++ = '\r';
|
|
+ *d++ = '\n';
|
|
+ first = 1;
|
|
+ ++s;
|
|
+ } else {
|
|
+ if (dot && first && *s == '.' && s[1] == '\n')
|
|
+ *d++ = '.'; /* double first dot on line */
|
|
+ first = 0;
|
|
+ *d++ = *s++;
|
|
+ }
|
|
+ }
|
|
+ if (dot) { /* add final . */
|
|
+ if (!first) {
|
|
+ *d++ = '\r';
|
|
+ *d++ = '\n';
|
|
+ }
|
|
+ *d++ = '.';
|
|
+ *d++ = '\r';
|
|
+ *d++ = '\n';
|
|
+ }
|
|
+ *d = '\0';
|
|
+}
|
|
|
|
- if (!db_found) {
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- daemon_printf( "%d invalid database, use SHOW DB for list\n",
|
|
- CODE_INVALID_DB );
|
|
- return;
|
|
- }
|
|
+static void daemon_printf(const char *format, ...)
|
|
+{
|
|
+ va_list ap;
|
|
+ char buf[BUFFERSIZE];
|
|
+ char *pt;
|
|
+ int len;
|
|
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
- dict_destroy_list( list );
|
|
- daemon_log( DICT_LOG_NOMATCH,
|
|
- "%s %s \"%s\"\n", databaseName, strategy, word );
|
|
- daemon_ok( CODE_NO_MATCH, "no match", "c" );
|
|
+ va_start(ap, format);
|
|
+ vsnprintf(buf, sizeof(buf), format, ap);
|
|
+ va_end(ap);
|
|
+ if ((len = strlen(buf)) >= BUFFERSIZE) {
|
|
+ log_info(":E: buffer overflow: %d\n", len);
|
|
+ daemon_terminate(0, __func__);
|
|
+ }
|
|
+
|
|
+ pt = alloca(2 * len + 10); /* +10 for the case when buf == "\n" */
|
|
+ daemon_crlf(pt, buf, 0);
|
|
+ daemon_write(pt, strlen(pt));
|
|
+}
|
|
+
|
|
+static void daemon_mime(void)
|
|
+{
|
|
+ if (daemonMime)
|
|
+ daemon_write("\r\n", 2);
|
|
}
|
|
|
|
-static lst_Position first_database_pos (void)
|
|
+static void daemon_mime_definition(const dictDatabase * db)
|
|
{
|
|
- return lst_init_position (DictConfig->dbl);
|
|
+ if (daemonMime) {
|
|
+ if (db->mime_header) {
|
|
+ daemon_printf("%s", db->mime_header);
|
|
+ }
|
|
+
|
|
+ daemon_write("\r\n", 2);
|
|
+ }
|
|
}
|
|
|
|
-static dictDatabase *next_database (
|
|
- lst_Position *databasePosition,
|
|
- const char *name)
|
|
+static void daemon_text(const char *text, int dot)
|
|
{
|
|
- dictDatabase *db = NULL;
|
|
+ char *pt = alloca(2 * strlen(text) + 10);
|
|
|
|
- assert (databasePosition);
|
|
+ daemon_crlf(pt, text, dot);
|
|
+ daemon_write(pt, strlen(pt));
|
|
+}
|
|
|
|
- if (!name)
|
|
- return NULL;
|
|
+static int daemon_read(char *buf, int count)
|
|
+{
|
|
+ return net_read(daemonS_in, buf, count);
|
|
+}
|
|
|
|
- if (*name == '*' || *name == '!') {
|
|
- if (*databasePosition) {
|
|
- do {
|
|
- db = lst_get_position( *databasePosition );
|
|
+static void daemon_ok(int code, const char *string, const char *timer)
|
|
+{
|
|
+ static int lastDefines = 0;
|
|
+ static int lastMatches = 0;
|
|
+ static int lastComparisons = 0;
|
|
|
|
- *databasePosition = lst_next_position( *databasePosition );
|
|
- } while ( db && (!db->available || db->invisible));
|
|
- }
|
|
- return db;
|
|
- } else {
|
|
- while (*databasePosition) {
|
|
- db = lst_get_position( *databasePosition );
|
|
- *databasePosition = lst_next_position( *databasePosition );
|
|
+ if (code == CODE_STATUS) {
|
|
+ lastDefines = 0;
|
|
+ lastMatches = 0;
|
|
+ lastComparisons = 0;
|
|
+ }
|
|
|
|
- if (db){
|
|
- if (
|
|
- !db -> invisible && db->available &&
|
|
- !strcmp(db -> databaseName,name))
|
|
- {
|
|
- return db;
|
|
- }
|
|
- }else{
|
|
- return NULL;
|
|
- }
|
|
- }
|
|
+ if (!timer) {
|
|
+ daemon_printf("%d %s\n", code, string);
|
|
+ } else {
|
|
+ tim_stop(timer);
|
|
+ daemon_printf("%d %s [d/m/c = %d/%d/%d; %sr %su %ss]\n",
|
|
+ code,
|
|
+ string,
|
|
+ _dict_defines - lastDefines,
|
|
+ _dict_matches - lastMatches,
|
|
+ _dict_comparisons - lastComparisons,
|
|
+ dict_format_time(tim_get_real(timer)),
|
|
+ dict_format_time(tim_get_user(timer)),
|
|
+ dict_format_time(tim_get_system(timer)));
|
|
+ }
|
|
|
|
- return NULL;
|
|
- }
|
|
+ lastDefines = _dict_defines;
|
|
+ lastMatches = _dict_matches;
|
|
+ lastComparisons = _dict_comparisons;
|
|
}
|
|
|
|
-static int count_databases( void )
|
|
+static int daemon_count_defs(lst_List list)
|
|
{
|
|
- int count = 0;
|
|
- const dictDatabase *db;
|
|
+ lst_Position p;
|
|
+ dictWord *dw;
|
|
+ unsigned long previousStart = 0;
|
|
+ unsigned long previousEnd = 0;
|
|
+ const char *previousDef = NULL;
|
|
+ int count = 0;
|
|
+
|
|
+ LST_ITERATE(list, p, dw) {
|
|
+ if (previousStart == dw->start &&
|
|
+ previousEnd == dw->end && previousDef == dw->def) {
|
|
+ continue;
|
|
+ }
|
|
|
|
- lst_Position databasePosition = first_database_pos ();
|
|
+ previousStart = dw->start;
|
|
+ previousEnd = dw->end;
|
|
+ previousDef = dw->def;
|
|
+
|
|
+ ++count;
|
|
+ }
|
|
+ return count;
|
|
+}
|
|
+
|
|
+static void daemon_dump_defs(lst_List list)
|
|
+{
|
|
+ lst_Position p;
|
|
+ char *buf;
|
|
+ dictWord *dw;
|
|
+ const dictDatabase *db = NULL;
|
|
+ const dictDatabase *db_visible = NULL;
|
|
+ unsigned long previousStart = 0;
|
|
+ unsigned long previousEnd = 0;
|
|
+ const char *previousDef = NULL;
|
|
+ int count;
|
|
|
|
- while (NULL != (db = next_database (&databasePosition, "*"))){
|
|
- assert (!db -> invisible);
|
|
+ LST_ITERATE(list, p, dw) {
|
|
+ db = dw->database;
|
|
|
|
- if (!db -> exit_db)
|
|
- ++count;
|
|
- }
|
|
+ if (previousStart == dw->start &&
|
|
+ previousEnd == dw->end && previousDef == dw->def) {
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ previousStart = dw->start;
|
|
+ previousEnd = dw->end;
|
|
+ previousDef = dw->def;
|
|
+
|
|
+ buf = dict_data_obtain(db, dw);
|
|
+
|
|
+ if (dw->database_visible) {
|
|
+ db_visible = dw->database_visible;
|
|
+ } else {
|
|
+ db_visible = db;
|
|
+ }
|
|
+
|
|
+ daemon_printf("%d \"%s\" %s \"%s\"\n",
|
|
+ CODE_DEFINITION_FOLLOWS,
|
|
+ dw->word,
|
|
+ db_visible->invisible ? "*" : db_visible->
|
|
+ databaseName,
|
|
+ db_visible->invisible ? "" : db_visible->
|
|
+ databaseShort);
|
|
+
|
|
+ daemon_mime_definition(db);
|
|
+
|
|
+ if (db->filter) {
|
|
+ count = strlen(buf);
|
|
+ daemon_log(DICT_LOG_AUTH, "filtering with: %s\ncount: %d\n",
|
|
+ db->filter, count);
|
|
+
|
|
+ dict_data_filter(buf, &count, strlen(buf), db->filter);
|
|
+ buf[count] = '\0';
|
|
+ }
|
|
+
|
|
+ daemon_text(buf, 1);
|
|
+ xfree(buf);
|
|
+ }
|
|
+}
|
|
+
|
|
+static int daemon_count_matches(lst_List list)
|
|
+{
|
|
+ lst_Position p;
|
|
+ dictWord *dw;
|
|
+ const char *prevword = NULL;
|
|
+ const dictDatabase *prevdb = NULL;
|
|
+ int count = 0;
|
|
+
|
|
+ LST_ITERATE(list, p, dw) {
|
|
+ if (prevdb == dw->database && prevword
|
|
+ && !strcmp(prevword, dw->word))
|
|
+ continue;
|
|
|
|
- return count;
|
|
+ prevword = dw->word;
|
|
+ prevdb = dw->database;
|
|
+
|
|
+ ++count;
|
|
+ }
|
|
+ return count;
|
|
}
|
|
|
|
-static void destroy_word_list (lst_List l)
|
|
+static void daemon_dump_matches(lst_List list)
|
|
{
|
|
- char *word;
|
|
+ lst_Position p;
|
|
+ dictWord *dw;
|
|
+ const char *prevword = NULL;
|
|
+ const dictDatabase *prevdb = NULL;
|
|
+ const dictDatabase *db = NULL;
|
|
|
|
- while (lst_length (l)){
|
|
- word = lst_pop (l);
|
|
- if (word)
|
|
- xfree (word);
|
|
- }
|
|
- lst_destroy (l);
|
|
-}
|
|
+ daemon_mime();
|
|
+ LST_ITERATE(list, p, dw) {
|
|
+ db = dw->database;
|
|
|
|
-/*
|
|
- Search for all words in word_list in the database db
|
|
- */
|
|
-static int dict_search_words (
|
|
- lst_List *l,
|
|
- lst_List word_list,
|
|
- const dictDatabase *db,
|
|
- int strategy,
|
|
- int *error, int *result,
|
|
- const dictPluginData **extra_result, int *extra_result_size)
|
|
-{
|
|
- lst_Position word_list_pos;
|
|
- int mc = 0;
|
|
- int matches_count = 0;
|
|
- const char *word;
|
|
-
|
|
- word_list_pos = lst_init_position (word_list);
|
|
- while (word_list_pos){
|
|
- word = lst_get_position (word_list_pos);
|
|
-
|
|
- if (word){
|
|
- matches_count = dict_search (
|
|
- l, word, db, strategy,
|
|
- daemonMime,
|
|
- result, extra_result, extra_result_size);
|
|
-
|
|
- if (*result == DICT_PLUGIN_RESULT_PREPROCESS){
|
|
- assert (matches_count > 0);
|
|
-
|
|
- xfree (lst_get_position (word_list_pos));
|
|
- lst_set_position (word_list_pos, NULL);
|
|
- }
|
|
-
|
|
- if (matches_count < 0){
|
|
- *error = 1;
|
|
- matches_count = abs (matches_count);
|
|
- mc += matches_count;
|
|
- break;
|
|
- }
|
|
+ if (prevdb == dw->database && prevword
|
|
+ && !strcmp(prevword, dw->word))
|
|
+ continue;
|
|
|
|
- mc += matches_count;
|
|
- }
|
|
+ prevword = dw->word;
|
|
+ prevdb = dw->database;
|
|
|
|
- word_list_pos = lst_next_position (word_list_pos);
|
|
- }
|
|
+ if (dw->database_visible) {
|
|
+ db = dw->database_visible;
|
|
+ }
|
|
|
|
- return mc;
|
|
+ daemon_printf("%s \"%s\"\n",
|
|
+ db->invisible ? "*" : db->databaseName, dw->word);
|
|
+ }
|
|
+ daemon_printf(".\n");
|
|
}
|
|
|
|
-int dict_search_databases (
|
|
- lst_List *l,
|
|
- lst_Position databasePosition, /* NULL for global database list */
|
|
- const char *databaseName, const char *word, int strategy,
|
|
- int *db_found)
|
|
+static void daemon_banner(void)
|
|
{
|
|
- int matches = -1;
|
|
- int matches_count = 0;
|
|
- int error = 0;
|
|
-
|
|
+ time_t t;
|
|
|
|
- const dictDatabase *db;
|
|
- dictWord *dw;
|
|
- char *p;
|
|
+ time(&t);
|
|
|
|
- lst_List preprocessed_words;
|
|
- int i;
|
|
-
|
|
- int result;
|
|
- const dictPluginData *extra_result;
|
|
- int extra_result_size;
|
|
+ snprintf(daemonStamp, sizeof(daemonStamp), "<%d.%d.%lu@%s>",
|
|
+ _dict_forks,
|
|
+ (int) getpid(), (long unsigned) t, net_hostname());
|
|
+ daemon_printf("%d %s %s <auth.mime> %s\n",
|
|
+ CODE_HELLO,
|
|
+ net_hostname(), dict_get_banner(0), daemonStamp);
|
|
+}
|
|
|
|
- *db_found = 0;
|
|
+static void daemon_define(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ lst_List list = lst_create();
|
|
+ int matches = 0;
|
|
+ const char *word;
|
|
+ const char *databaseName;
|
|
+ int extension = (argv[0][0] == 'd' && argv[0][1] == '\0');
|
|
+ int db_found = 0;
|
|
+
|
|
+ if (extension) {
|
|
+ switch (argc) {
|
|
+ case 2:
|
|
+ databaseName = "*";
|
|
+ word = argv[1];
|
|
+ break;
|
|
+ case 3:
|
|
+ databaseName = argv[1];
|
|
+ word = argv[2];
|
|
+ break;
|
|
+ default:
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+ } else if (argc == 3) {
|
|
+ databaseName = argv[1];
|
|
+ word = argv[2];
|
|
+ } else {
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ matches = abs(dict_search_databases(list, NULL,
|
|
+ databaseName, word,
|
|
+ DICT_STRAT_EXACT, &db_found));
|
|
+
|
|
+ if (db_found && matches > 0) {
|
|
+ int actual_matches = daemon_count_defs(list);
|
|
+
|
|
+ _dict_defines += actual_matches;
|
|
+ daemon_log(DICT_LOG_DEFINE,
|
|
+ "%s \"%s\" %d\n", databaseName, word, actual_matches);
|
|
+ daemon_printf("%d %d definitions retrieved\n",
|
|
+ CODE_DEFINITIONS_FOUND, actual_matches);
|
|
+ daemon_dump_defs(list);
|
|
+ daemon_ok(CODE_OK, "ok", "c");
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ return;
|
|
+ }
|
|
|
|
- if (!databasePosition)
|
|
- databasePosition = first_database_pos ();
|
|
+ if (!db_found) {
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ daemon_printf("%d invalid database, use SHOW DB for list\n",
|
|
+ CODE_INVALID_DB);
|
|
+ return;
|
|
+ }
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ daemon_log(DICT_LOG_NOMATCH, "%s exact \"%s\"\n", databaseName, word);
|
|
+ daemon_ok(CODE_NO_MATCH, "no match", "c");
|
|
+}
|
|
+
|
|
+static void daemon_match(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ lst_List list = lst_create();
|
|
+ int matches = 0;
|
|
+ const char *word;
|
|
+ const char *databaseName;
|
|
+ const char *strategy;
|
|
+ int strategyNumber;
|
|
+ int extension = (argv[0][0] == 'm' && argv[0][1] == '\0');
|
|
+ int db_found = 0;
|
|
+
|
|
+ if (extension) {
|
|
+ switch (argc) {
|
|
+ case 2:
|
|
+ databaseName = "*";
|
|
+ strategy = ".";
|
|
+ word = argv[1];
|
|
+ break;
|
|
+ case 3:
|
|
+ databaseName = "*";
|
|
+ strategy = argv[1];
|
|
+ word = argv[2];
|
|
+ break;
|
|
+ case 4:
|
|
+ databaseName = argv[1];
|
|
+ strategy = argv[2];
|
|
+ word = argv[3];
|
|
+ break;
|
|
+ default:
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+ } else if (argc == 4) {
|
|
+ databaseName = argv[1];
|
|
+ strategy = argv[2];
|
|
+ word = argv[3];
|
|
+ } else {
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if ((strategyNumber = lookup_strategy(strategy)) < 0) {
|
|
+ daemon_printf("%d invalid strategy, use SHOW STRAT for a list\n",
|
|
+ CODE_INVALID_STRATEGY);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ matches = abs(dict_search_databases(list, NULL,
|
|
+ databaseName, word,
|
|
+ strategyNumber | DICT_MATCH_MASK,
|
|
+ &db_found));
|
|
+
|
|
+ if (db_found && matches > 0) {
|
|
+ int actual_matches = daemon_count_matches(list);
|
|
+
|
|
+ _dict_matches += actual_matches;
|
|
+ daemon_log(DICT_LOG_MATCH,
|
|
+ "%s %s \"%s\" %d\n",
|
|
+ databaseName, strategy, word, actual_matches);
|
|
+ daemon_printf("%d %d matches found\n",
|
|
+ CODE_MATCHES_FOUND, actual_matches);
|
|
+ daemon_dump_matches(list);
|
|
+ daemon_ok(CODE_OK, "ok", "c");
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ return;
|
|
+ }
|
|
|
|
- preprocessed_words = lst_create ();
|
|
- lst_append (preprocessed_words, xstrdup(word));
|
|
+ if (!db_found) {
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ daemon_printf("%d invalid database, use SHOW DB for list\n",
|
|
+ CODE_INVALID_DB);
|
|
+ return;
|
|
+ }
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
+ dict_destroy_list(list);
|
|
+ daemon_log(DICT_LOG_NOMATCH,
|
|
+ "%s %s \"%s\"\n", databaseName, strategy, word);
|
|
+ daemon_ok(CODE_NO_MATCH, "no match", "c");
|
|
+}
|
|
|
|
- while (!error && (db = next_database (&databasePosition, databaseName))) {
|
|
- if (db -> exit_db)
|
|
- /* dictionary_exit */
|
|
- break;
|
|
+static lst_Position first_database_pos(void)
|
|
+{
|
|
+ return lst_init_position(DictConfig->dbl);
|
|
+}
|
|
|
|
- *db_found = 1;
|
|
+static dictDatabase *next_database(lst_Position * databasePosition,
|
|
+ const char *name)
|
|
+{
|
|
+ dictDatabase *db = NULL;
|
|
|
|
- result = DICT_PLUGIN_RESULT_NOTFOUND;
|
|
+ assert(databasePosition);
|
|
|
|
- matches_count = dict_search_words (
|
|
- l,
|
|
- preprocessed_words, db, strategy,
|
|
- &error,
|
|
- &result, &extra_result, &extra_result_size);
|
|
+ if (!name)
|
|
+ return NULL;
|
|
|
|
- if (matches < 0)
|
|
- matches = 0;
|
|
+ if (*name == '*' || *name == '!') {
|
|
+ if (*databasePosition) {
|
|
+ do {
|
|
+ db = lst_get_position(*databasePosition);
|
|
|
|
- if (result == DICT_PLUGIN_RESULT_PREPROCESS){
|
|
- for (i=0; i < matches_count; ++i){
|
|
- dw = lst_pop (l);
|
|
- switch (dw -> def_size){
|
|
- case -1:
|
|
- p = xstrdup (dw -> word);
|
|
- lst_append (preprocessed_words, p);
|
|
- break;
|
|
- case 0:
|
|
- break;
|
|
- default:
|
|
- p = xmalloc (1 + dw -> def_size);
|
|
- memcpy (p, dw -> def, dw -> def_size);
|
|
- p [dw -> def_size] = 0;
|
|
+ *databasePosition = lst_next_position(*databasePosition);
|
|
+ } while (db && (!db->available || db->invisible));
|
|
+ }
|
|
+ return db;
|
|
+ } else {
|
|
+ while (*databasePosition) {
|
|
+ db = lst_get_position(*databasePosition);
|
|
+ *databasePosition = lst_next_position(*databasePosition);
|
|
|
|
- lst_append (preprocessed_words, p);
|
|
+ if (db) {
|
|
+ if (!db->invisible && db->available &&
|
|
+ !strcmp(db->databaseName, name)) {
|
|
+ return db;
|
|
+ }
|
|
+ } else {
|
|
+ return NULL;
|
|
}
|
|
+ }
|
|
|
|
- dict_destroy_datum (dw);
|
|
- }
|
|
- }else{
|
|
- matches += matches_count;
|
|
+ return NULL;
|
|
+ }
|
|
+}
|
|
|
|
- if (result == DICT_PLUGIN_RESULT_EXIT)
|
|
- break;
|
|
+static int count_databases(void)
|
|
+{
|
|
+ int count = 0;
|
|
+ const dictDatabase *db;
|
|
|
|
- if (*databaseName == '*')
|
|
- continue;
|
|
- else if (!matches && *databaseName == '!')
|
|
- continue;
|
|
+ lst_Position databasePosition = first_database_pos();
|
|
|
|
- break;
|
|
- }
|
|
- }
|
|
+ while (NULL != (db = next_database(&databasePosition, "*"))) {
|
|
+ assert(!db->invisible);
|
|
|
|
- destroy_word_list (preprocessed_words);
|
|
+ if (!db->exit_db)
|
|
+ ++count;
|
|
+ }
|
|
|
|
- return error ? -matches : matches;
|
|
+ return count;
|
|
}
|
|
|
|
-static void daemon_show_db( const char *cmdline, int argc, const char **argv )
|
|
+static void destroy_word_list(lst_List l)
|
|
{
|
|
- int count;
|
|
- const dictDatabase *db;
|
|
-
|
|
- lst_Position databasePosition;
|
|
+ char *word;
|
|
|
|
- if (argc != 2) {
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
+ while (lst_length(l)) {
|
|
+ word = lst_pop(l);
|
|
+ if (word)
|
|
+ xfree(word);
|
|
+ }
|
|
+ lst_destroy(l);
|
|
+}
|
|
|
|
- if (!(count = count_databases())) {
|
|
- daemon_printf( "%d no databases present\n", CODE_NO_DATABASES );
|
|
- } else {
|
|
- daemon_printf( "%d %d databases present\n",
|
|
- CODE_DATABASE_LIST, count );
|
|
+/*
|
|
+ Search for all words in word_list in the database db
|
|
+ */
|
|
+static int dict_search_words(lst_List * l,
|
|
+ lst_List word_list,
|
|
+ const dictDatabase * db,
|
|
+ int strategy,
|
|
+ int *error, int *result,
|
|
+ const dictPluginData ** extra_result,
|
|
+ int *extra_result_size)
|
|
+{
|
|
+ lst_Position word_list_pos;
|
|
+ int mc = 0;
|
|
+ int matches_count = 0;
|
|
+ const char *word;
|
|
+
|
|
+ word_list_pos = lst_init_position(word_list);
|
|
+ while (word_list_pos) {
|
|
+ word = lst_get_position(word_list_pos);
|
|
+
|
|
+ if (word) {
|
|
+ matches_count = dict_search(l, word, db, strategy,
|
|
+ daemonMime,
|
|
+ result, extra_result,
|
|
+ extra_result_size);
|
|
|
|
- databasePosition = first_database_pos ();
|
|
+ if (*result == DICT_PLUGIN_RESULT_PREPROCESS) {
|
|
+ assert(matches_count > 0);
|
|
|
|
- daemon_mime();
|
|
- while ((db = next_database(&databasePosition, "*"))) {
|
|
- assert (!db->invisible);
|
|
+ xfree(lst_get_position(word_list_pos));
|
|
+ lst_set_position(word_list_pos, NULL);
|
|
+ }
|
|
|
|
- if (db -> exit_db)
|
|
- continue;
|
|
+ if (matches_count < 0) {
|
|
+ *error = 1;
|
|
+ matches_count = abs(matches_count);
|
|
+ mc += matches_count;
|
|
+ break;
|
|
+ }
|
|
|
|
- daemon_printf(
|
|
- "%s \"%s\"\n",
|
|
- db->databaseName, db->databaseShort );
|
|
- }
|
|
- daemon_printf( ".\n" );
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
- }
|
|
-}
|
|
-
|
|
-static void daemon_show_strat( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- int i;
|
|
-
|
|
- int strat_count = get_strategy_count ();
|
|
- dictStrategy const * const * strats = get_strategies ();
|
|
-
|
|
- if (argc != 2) {
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
-
|
|
- if (strat_count){
|
|
- daemon_printf( "%d %d strategies present\n",
|
|
- CODE_STRATEGY_LIST, strat_count );
|
|
- daemon_mime();
|
|
-
|
|
- for (i = 0; i < strat_count; i++) {
|
|
- daemon_printf( "%s \"%s\"\n",
|
|
- strats [i] -> name, strats [i] -> description );
|
|
- }
|
|
-
|
|
- daemon_printf( ".\n" );
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
- }else{
|
|
- daemon_printf( "%d no strategies available\n", CODE_NO_STRATEGIES );
|
|
- }
|
|
-}
|
|
-
|
|
-void daemon_show_info(
|
|
- const char *cmdline,
|
|
- int argc, const char **argv )
|
|
-{
|
|
- char *buf=NULL;
|
|
- dictWord *dw;
|
|
- const dictDatabase *db;
|
|
- lst_List list;
|
|
- const char *info_entry_name = DICT_INFO_ENTRY_NAME;
|
|
-
|
|
- lst_Position databasePosition = first_database_pos ();
|
|
-
|
|
- if (argc != 3) {
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- return;
|
|
- }
|
|
-
|
|
- if ((argv[2][0] == '*' || argv[2][0] == '!') && argv[2][1] == '\0') {
|
|
- daemon_printf( "%d invalid database, use SHOW DB for list\n",
|
|
- CODE_INVALID_DB );
|
|
- return;
|
|
- }
|
|
-
|
|
- list = lst_create();
|
|
- while ((db = next_database(&databasePosition, argv[2]))) {
|
|
- if (db -> databaseInfo && db -> databaseInfo [0] != '@'){
|
|
- daemon_printf( "%d information for %s\n",
|
|
- CODE_DATABASE_INFO, argv[2] );
|
|
- daemon_mime();
|
|
- daemon_text(db -> databaseInfo, 1);
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
- return;
|
|
- }
|
|
-
|
|
- if (db -> databaseInfo && db -> databaseInfo [0] == '@')
|
|
- info_entry_name = db -> databaseInfo + 1;
|
|
-
|
|
- if (dict_search (
|
|
- list,
|
|
- info_entry_name,
|
|
- db, DICT_STRAT_EXACT, 0,
|
|
- NULL, NULL, NULL))
|
|
- {
|
|
- int i=1;
|
|
- int list_size = lst_length (list);
|
|
-
|
|
- daemon_printf( "%d information for %s\n",
|
|
- CODE_DATABASE_INFO, argv[2] );
|
|
- daemon_mime();
|
|
-
|
|
- if (db -> virtual_db){
|
|
- daemon_printf ("The virtual dictionary `%s' includes the following:\n\n",
|
|
- db -> databaseName);
|
|
- }
|
|
+ mc += matches_count;
|
|
+ }
|
|
|
|
- for (i=1; i <= list_size; ++i){
|
|
- dw = lst_nth_get( list, i );
|
|
+ word_list_pos = lst_next_position(word_list_pos);
|
|
+ }
|
|
|
|
- daemon_printf ("============ %s ============\n",
|
|
- dw -> database -> databaseName);
|
|
+ return mc;
|
|
+}
|
|
|
|
- buf = dict_data_obtain( dw -> database, dw );
|
|
+int dict_search_databases(lst_List * l, lst_Position databasePosition, /* NULL for global database list */
|
|
+ const char *databaseName, const char *word,
|
|
+ int strategy, int *db_found)
|
|
+{
|
|
+ int matches = -1;
|
|
+ int matches_count = 0;
|
|
+ int error = 0;
|
|
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
|
|
- if (buf)
|
|
- daemon_text (buf, 0);
|
|
- }
|
|
+ const dictDatabase *db;
|
|
+ dictWord *dw;
|
|
+ char *p;
|
|
|
|
- daemon_text ("\n", 1);
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
+ lst_List preprocessed_words;
|
|
+ int i;
|
|
|
|
- dict_destroy_list (list);
|
|
+ int result;
|
|
+ const dictPluginData *extra_result;
|
|
+ int extra_result_size;
|
|
|
|
- return;
|
|
- } else {
|
|
-#ifdef USE_PLUGIN
|
|
- call_dictdb_free (DictConfig->dbl);
|
|
-#endif
|
|
+ *db_found = 0;
|
|
|
|
- dict_destroy_list( list );
|
|
- daemon_printf( "%d information for %s\n",
|
|
- CODE_DATABASE_INFO, argv[2] );
|
|
- daemon_mime();
|
|
- daemon_text( "No information available\n" , 1);
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
- return;
|
|
- }
|
|
- }
|
|
+ if (!databasePosition)
|
|
+ databasePosition = first_database_pos();
|
|
|
|
- dict_destroy_list( list );
|
|
- daemon_printf( "%d invalid database, use SHOW DB for list\n",
|
|
- CODE_INVALID_DB );
|
|
-}
|
|
+ preprocessed_words = lst_create();
|
|
+ lst_append(preprocessed_words, xstrdup(word));
|
|
|
|
-static int daemon_get_max_dbname_length ()
|
|
-{
|
|
- size_t max_len = 0;
|
|
- size_t curr_len = 0;
|
|
+ while (!error && (db = next_database(&databasePosition, databaseName))) {
|
|
+ if (db->exit_db)
|
|
+ /* dictionary_exit */
|
|
+ break;
|
|
+
|
|
+ *db_found = 1;
|
|
|
|
- const dictDatabase *db;
|
|
+ result = DICT_PLUGIN_RESULT_NOTFOUND;
|
|
|
|
- lst_Position databasePosition = first_database_pos ();
|
|
+ matches_count = dict_search_words(l,
|
|
+ preprocessed_words, db, strategy,
|
|
+ &error,
|
|
+ &result, &extra_result,
|
|
+ &extra_result_size);
|
|
+
|
|
+ if (matches < 0)
|
|
+ matches = 0;
|
|
+
|
|
+ if (result == DICT_PLUGIN_RESULT_PREPROCESS) {
|
|
+ for (i = 0; i < matches_count; ++i) {
|
|
+ dw = lst_pop(l);
|
|
+ switch (dw->def_size) {
|
|
+ case -1:
|
|
+ p = xstrdup(dw->word);
|
|
+ lst_append(preprocessed_words, p);
|
|
+ break;
|
|
+ case 0:
|
|
+ break;
|
|
+ default:
|
|
+ p = xmalloc(1 + dw->def_size);
|
|
+ memcpy(p, dw->def, dw->def_size);
|
|
+ p[dw->def_size] = 0;
|
|
|
|
- while (NULL != (db = next_database (&databasePosition, "*"))){
|
|
- assert (!db -> invisible);
|
|
+ lst_append(preprocessed_words, p);
|
|
+ }
|
|
|
|
- if (db -> databaseName){
|
|
- curr_len = strlen (db -> databaseName);
|
|
+ dict_destroy_datum(dw);
|
|
+ }
|
|
+ } else {
|
|
+ matches += matches_count;
|
|
|
|
- if (curr_len > max_len){
|
|
- max_len = curr_len;
|
|
- }
|
|
- }
|
|
- }
|
|
+ if (result == DICT_PLUGIN_RESULT_EXIT)
|
|
+ break;
|
|
|
|
- return (int) max_len;
|
|
-}
|
|
+ if (*databaseName == '*')
|
|
+ continue;
|
|
+ else if (!matches && *databaseName == '!')
|
|
+ continue;
|
|
|
|
-static void daemon_show_server (
|
|
- const char *cmdline,
|
|
- int argc, const char **argv)
|
|
-{
|
|
- FILE *str;
|
|
- char buffer[1024];
|
|
- const dictDatabase *db;
|
|
- double uptime;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
|
|
- int headwords;
|
|
+ destroy_word_list(preprocessed_words);
|
|
|
|
- int index_size;
|
|
- char index_size_uom;
|
|
+ return error ? -matches : matches;
|
|
+}
|
|
|
|
- int data_size;
|
|
- char data_size_uom;
|
|
- int data_length;
|
|
- char data_length_uom;
|
|
+static void daemon_show_db(const char *cmdline, int argc,
|
|
+ const char **argv)
|
|
+{
|
|
+ int count;
|
|
+ const dictDatabase *db;
|
|
+
|
|
+ lst_Position databasePosition;
|
|
+
|
|
+ if (argc != 2) {
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (!(count = count_databases())) {
|
|
+ daemon_printf("%d no databases present\n", CODE_NO_DATABASES);
|
|
+ } else {
|
|
+ daemon_printf("%d %d databases present\n",
|
|
+ CODE_DATABASE_LIST, count);
|
|
+
|
|
+ databasePosition = first_database_pos();
|
|
+
|
|
+ daemon_mime();
|
|
+ while ((db = next_database(&databasePosition, "*"))) {
|
|
+ assert(!db->invisible);
|
|
+
|
|
+ if (db->exit_db)
|
|
+ continue;
|
|
+
|
|
+ daemon_printf("%s \"%s\"\n",
|
|
+ db->databaseName, db->databaseShort);
|
|
+ }
|
|
+ daemon_printf(".\n");
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void daemon_show_strat(const char *cmdline, int argc,
|
|
+ const char **argv)
|
|
+{
|
|
+ int i;
|
|
+
|
|
+ int strat_count = get_strategy_count();
|
|
+ dictStrategy const *const *strats = get_strategies();
|
|
+
|
|
+ if (argc != 2) {
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (strat_count) {
|
|
+ daemon_printf("%d %d strategies present\n",
|
|
+ CODE_STRATEGY_LIST, strat_count);
|
|
+ daemon_mime();
|
|
+
|
|
+ for (i = 0; i < strat_count; i++) {
|
|
+ daemon_printf("%s \"%s\"\n",
|
|
+ strats[i]->name, strats[i]->description);
|
|
+ }
|
|
+
|
|
+ daemon_printf(".\n");
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+ } else {
|
|
+ daemon_printf("%d no strategies available\n", CODE_NO_STRATEGIES);
|
|
+ }
|
|
+}
|
|
+
|
|
+void daemon_show_info(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ char *buf = NULL;
|
|
+ dictWord *dw;
|
|
+ const dictDatabase *db;
|
|
+ lst_List list;
|
|
+ const char *info_entry_name = DICT_INFO_ENTRY_NAME;
|
|
+
|
|
+ lst_Position databasePosition = first_database_pos();
|
|
+
|
|
+ if (argc != 3) {
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if ((argv[2][0] == '*' || argv[2][0] == '!') && argv[2][1] == '\0') {
|
|
+ daemon_printf("%d invalid database, use SHOW DB for list\n",
|
|
+ CODE_INVALID_DB);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ list = lst_create();
|
|
+ while ((db = next_database(&databasePosition, argv[2]))) {
|
|
+ if (db->databaseInfo && db->databaseInfo[0] != '@') {
|
|
+ daemon_printf("%d information for %s\n",
|
|
+ CODE_DATABASE_INFO, argv[2]);
|
|
+ daemon_mime();
|
|
+ daemon_text(db->databaseInfo, 1);
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (db->databaseInfo && db->databaseInfo[0] == '@')
|
|
+ info_entry_name = db->databaseInfo + 1;
|
|
+
|
|
+ if (dict_search(list,
|
|
+ info_entry_name,
|
|
+ db, DICT_STRAT_EXACT, 0, NULL, NULL, NULL)) {
|
|
+ int i = 1;
|
|
+ int list_size = lst_length(list);
|
|
+
|
|
+ daemon_printf("%d information for %s\n",
|
|
+ CODE_DATABASE_INFO, argv[2]);
|
|
+ daemon_mime();
|
|
+
|
|
+ if (db->virtual_db) {
|
|
+ daemon_printf
|
|
+ ("The virtual dictionary `%s' includes the following:\n\n",
|
|
+ db->databaseName);
|
|
+ }
|
|
|
|
- int max_dbname_len;
|
|
+ for (i = 1; i <= list_size; ++i) {
|
|
+ dw = lst_nth_get(list, i);
|
|
|
|
- lst_Position databasePosition = first_database_pos ();
|
|
+ daemon_printf("============ %s ============\n",
|
|
+ dw->database->databaseName);
|
|
|
|
- daemon_printf( "%d server information\n", CODE_SERVER_INFO );
|
|
- daemon_mime();
|
|
+ buf = dict_data_obtain(dw->database, dw);
|
|
|
|
- /* banner: dictd and OS */
|
|
- if (!site_info_no_banner){
|
|
- daemon_printf( "%s\n", dict_get_banner(0) );
|
|
- }
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
|
|
- /* uptime and forks */
|
|
- if (!site_info_no_uptime && !inetd){
|
|
- tim_stop("dictd");
|
|
- uptime = tim_get_real("dictd");
|
|
+ if (buf)
|
|
+ daemon_text(buf, 0);
|
|
+ }
|
|
|
|
- daemon_printf (
|
|
- "On %s: up %s, %d fork%s (%0.1f/hour)\n",
|
|
- net_hostname(),
|
|
- dict_format_time( uptime ),
|
|
- _dict_forks,
|
|
- _dict_forks > 1 ? "s" : "",
|
|
- (_dict_forks/uptime)*3600.0 );
|
|
+ daemon_text("\n", 1);
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
|
|
- daemon_printf ("\n");
|
|
- }
|
|
+ dict_destroy_list(list);
|
|
|
|
- if (!site_info_no_dblist && count_databases()) {
|
|
- daemon_printf( "Database Headwords Index Data Uncompressed\n" );
|
|
+ return;
|
|
+ } else {
|
|
+#ifdef USE_PLUGIN
|
|
+ call_dictdb_free(DictConfig->dbl);
|
|
+#endif
|
|
|
|
- databasePosition = first_database_pos ();
|
|
+ dict_destroy_list(list);
|
|
+ daemon_printf("%d information for %s\n",
|
|
+ CODE_DATABASE_INFO, argv[2]);
|
|
+ daemon_mime();
|
|
+ daemon_text("No information available\n", 1);
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+ return;
|
|
+ }
|
|
+ }
|
|
|
|
- while (db = next_database (&databasePosition, "*"),
|
|
- db != NULL)
|
|
- {
|
|
- headwords = (db->index ? db->index->headwords : 0);
|
|
+ dict_destroy_list(list);
|
|
+ daemon_printf("%d invalid database, use SHOW DB for list\n",
|
|
+ CODE_INVALID_DB);
|
|
+}
|
|
|
|
- index_size = 0;
|
|
- index_size_uom = 'k';
|
|
+static int daemon_get_max_dbname_length()
|
|
+{
|
|
+ size_t max_len = 0;
|
|
+ size_t curr_len = 0;
|
|
|
|
- data_size = 0;
|
|
- data_size_uom = 'k';
|
|
- data_length = 0;
|
|
- data_length_uom= 'k';
|
|
+ const dictDatabase *db;
|
|
|
|
- max_dbname_len = 0;
|
|
+ lst_Position databasePosition = first_database_pos();
|
|
|
|
- assert (!db -> invisible);
|
|
+ while (NULL != (db = next_database(&databasePosition, "*"))) {
|
|
+ assert(!db->invisible);
|
|
|
|
- if (db->index){
|
|
- index_size = db->index->size/1024 > 10240 ?
|
|
- db->index->size/1024/1024 : db->index->size/1024;
|
|
- index_size_uom = db->index->size/1024 > 10240 ? 'M' : 'k';
|
|
- }
|
|
+ if (db->databaseName) {
|
|
+ curr_len = strlen(db->databaseName);
|
|
|
|
- if (db->data){
|
|
- data_size = db->data->size/1024 > 10240 ?
|
|
- db->data->size/1024/1024 : db->data->size/1024;
|
|
- data_size_uom = db->data->size/1024 > 10240 ? 'M' : 'k';
|
|
+ if (curr_len > max_len) {
|
|
+ max_len = curr_len;
|
|
+ }
|
|
+ }
|
|
+ }
|
|
|
|
- data_length = db->data->length/1024 > 10240 ?
|
|
- db->data->length/1024/1024 : db->data->length/1024;
|
|
- data_length_uom = db->data->length/1024 > 10240 ? 'M' : 'k';
|
|
- }
|
|
+ return (int) max_len;
|
|
+}
|
|
|
|
- max_dbname_len = daemon_get_max_dbname_length ();
|
|
+static void daemon_show_server(const char *cmdline,
|
|
+ int argc, const char **argv)
|
|
+{
|
|
+ FILE *str;
|
|
+ char buffer[1024];
|
|
+ const dictDatabase *db;
|
|
+ double uptime;
|
|
|
|
- daemon_printf(
|
|
- "%-*.*s %10i %10i %cB %10i %cB %10i %cB\n",
|
|
- max_dbname_len,
|
|
- max_dbname_len,
|
|
+ int headwords;
|
|
|
|
- db->databaseName,
|
|
- headwords,
|
|
+ int index_size;
|
|
+ char index_size_uom;
|
|
|
|
- index_size,
|
|
- index_size_uom,
|
|
+ int data_size;
|
|
+ char data_size_uom;
|
|
+ int data_length;
|
|
+ char data_length_uom;
|
|
|
|
- data_size,
|
|
- data_size_uom,
|
|
+ int max_dbname_len;
|
|
|
|
- data_length,
|
|
- data_length_uom);
|
|
- }
|
|
+ lst_Position databasePosition = first_database_pos();
|
|
|
|
- daemon_printf ("\n");
|
|
- }
|
|
+ daemon_printf("%d server information\n", CODE_SERVER_INFO);
|
|
+ daemon_mime();
|
|
|
|
- if (site_info && (str = fopen( site_info, "r" ))) {
|
|
- while ((fgets( buffer, 1000, str ))) daemon_printf( "%s", buffer );
|
|
- fclose( str );
|
|
- }
|
|
- daemon_printf( ".\n" );
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
-}
|
|
+ /* banner: dictd and OS */
|
|
+ if (!site_info_no_banner) {
|
|
+ daemon_printf("%s\n", dict_get_banner(0));
|
|
+ }
|
|
|
|
-static void daemon_show( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
-}
|
|
+ /* uptime and forks */
|
|
+ if (!site_info_no_uptime && !inetd) {
|
|
+ tim_stop("dictd");
|
|
+ uptime = tim_get_real("dictd");
|
|
|
|
-static void daemon_option_mime( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- ++daemonMime;
|
|
- daemon_ok( CODE_OK, "ok - using MIME headers", NULL );
|
|
-}
|
|
+ daemon_printf("On %s: up %s, %d fork%s (%0.1f/hour)\n",
|
|
+ net_hostname(),
|
|
+ dict_format_time(uptime),
|
|
+ _dict_forks,
|
|
+ _dict_forks > 1 ? "s" : "",
|
|
+ (_dict_forks / uptime) * 3600.0);
|
|
|
|
-static void daemon_option( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
-}
|
|
+ daemon_printf("\n");
|
|
+ }
|
|
|
|
-static void daemon_client( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- const char *pt = strchr( cmdline, ' ' );
|
|
-
|
|
- if (pt)
|
|
- daemon_log( DICT_LOG_CLIENT, "%.200s\n", pt + 1 );
|
|
- else
|
|
- daemon_log( DICT_LOG_CLIENT, "%.200s\n", cmdline );
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
-}
|
|
+ if (!site_info_no_dblist && count_databases()) {
|
|
+ daemon_printf
|
|
+ ("Database Headwords Index Data Uncompressed\n");
|
|
|
|
-static void daemon_auth( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- char *buf;
|
|
- hsh_HashTable h = DictConfig->usl;
|
|
- const char *secret;
|
|
- struct MD5Context ctx;
|
|
- unsigned char digest[16];
|
|
- char hex[33];
|
|
- int i;
|
|
- int buf_size;
|
|
+ databasePosition = first_database_pos();
|
|
|
|
- if (argc != 3)
|
|
- daemon_printf( "%d syntax error, illegal parameters\n",
|
|
- CODE_ILLEGAL_PARAM );
|
|
- if (!h || !(secret = hsh_retrieve(h, argv[1]))) {
|
|
- daemon_log( DICT_LOG_AUTH, "%s@%s/%s denied: invalid username\n",
|
|
- argv[1], daemonHostname, daemonIP );
|
|
- daemon_printf( "%d auth denied\n", CODE_AUTH_DENIED );
|
|
- return;
|
|
- }
|
|
+ while (db = next_database(&databasePosition, "*"), db != NULL) {
|
|
+ headwords = (db->index ? db->index->headwords : 0);
|
|
|
|
- buf_size = strlen(daemonStamp) + strlen(secret) + 10;
|
|
- buf = alloca(buf_size);
|
|
- snprintf( buf, buf_size, "%s%s", daemonStamp, secret );
|
|
+ index_size = 0;
|
|
+ index_size_uom = 'k';
|
|
|
|
- MD5Init(&ctx);
|
|
- MD5Update(&ctx, (const unsigned char *) buf, strlen(buf));
|
|
- MD5Final(digest, &ctx);
|
|
+ data_size = 0;
|
|
+ data_size_uom = 'k';
|
|
+ data_length = 0;
|
|
+ data_length_uom = 'k';
|
|
|
|
- for (i = 0; i < 16; i++)
|
|
- snprintf( hex+2*i, 3, "%02x", digest[i] );
|
|
+ max_dbname_len = 0;
|
|
|
|
- hex[32] = '\0';
|
|
+ assert(!db->invisible);
|
|
|
|
- PRINTF(DBG_AUTH,("Got %s expected %s\n", argv[2], hex ));
|
|
+ if (db->index) {
|
|
+ index_size = db->index->size / 1024 > 10240 ?
|
|
+ db->index->size / 1024 / 1024 : db->index->size / 1024;
|
|
+ index_size_uom =
|
|
+ db->index->size / 1024 > 10240 ? 'M' : 'k';
|
|
+ }
|
|
|
|
- if (strcmp(hex,argv[2])) {
|
|
- daemon_log( DICT_LOG_AUTH, "%s@%s/%s denied: hash mismatch\n",
|
|
- argv[1], daemonHostname, daemonIP );
|
|
- daemon_printf( "%d auth denied\n", CODE_AUTH_DENIED );
|
|
- } else {
|
|
- daemon_printf( "%d authenticated\n", CODE_AUTH_OK );
|
|
- daemon_check_auth( argv[1] );
|
|
- }
|
|
-}
|
|
+ if (db->data) {
|
|
+ data_size = db->data->size / 1024 > 10240 ?
|
|
+ db->data->size / 1024 / 1024 : db->data->size / 1024;
|
|
+ data_size_uom = db->data->size / 1024 > 10240 ? 'M' : 'k';
|
|
+
|
|
+ data_length = db->data->length / 1024 > 10240 ?
|
|
+ db->data->length / 1024 / 1024 : db->data->length /
|
|
+ 1024;
|
|
+ data_length_uom =
|
|
+ db->data->length / 1024 > 10240 ? 'M' : 'k';
|
|
+ }
|
|
|
|
-static void daemon_status( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- daemon_ok( CODE_STATUS, "status", "t" );
|
|
-}
|
|
+ max_dbname_len = daemon_get_max_dbname_length();
|
|
|
|
-static void daemon_help( const char *cmdline, int argc, const char **argv )
|
|
-{
|
|
- daemon_printf( "%d help text follows\n", CODE_HELP );
|
|
- daemon_mime();
|
|
- daemon_text(
|
|
- "DEFINE database word -- look up word in database\n"
|
|
- "MATCH database strategy word -- match word in database using strategy\n"
|
|
- "SHOW DB -- list all accessible databases\n"
|
|
- "SHOW DATABASES -- list all accessible databases\n"
|
|
- "SHOW STRAT -- list available matching strategies\n"
|
|
- "SHOW STRATEGIES -- list available matching strategies\n"
|
|
- "SHOW INFO database -- provide information about the database\n"
|
|
- "SHOW SERVER -- provide site-specific information\n"
|
|
- "OPTION MIME -- use MIME headers\n"
|
|
- "CLIENT info -- identify client to server\n"
|
|
- "AUTH user string -- provide authentication information\n"
|
|
- "STATUS -- display timing information\n"
|
|
- "HELP -- display this help information\n"
|
|
- "QUIT -- terminate connection\n\n"
|
|
- "The following commands are unofficial server extensions for debugging\n"
|
|
- "only. You may find them useful if you are using telnet as a client.\n"
|
|
- "If you are writing a client, you MUST NOT use these commands, since\n"
|
|
- "they won't be supported on any other server!\n\n"
|
|
- "D word -- DEFINE * word\n"
|
|
- "D database word -- DEFINE database word\n"
|
|
- "M word -- MATCH * . word\n"
|
|
- "M strategy word -- MATCH * strategy word\n"
|
|
- "M database strategy word -- MATCH database strategy word\n"
|
|
- "S -- STATUS\n"
|
|
- "H -- HELP\n"
|
|
- "Q -- QUIT\n"
|
|
- , 1);
|
|
- daemon_ok( CODE_OK, "ok", NULL );
|
|
+ daemon_printf("%-*.*s %10i %10i %cB %10i %cB %10i %cB\n",
|
|
+ max_dbname_len,
|
|
+ max_dbname_len,
|
|
+ db->databaseName,
|
|
+ headwords,
|
|
+ index_size,
|
|
+ index_size_uom,
|
|
+ data_size,
|
|
+ data_size_uom, data_length, data_length_uom);
|
|
+ }
|
|
+
|
|
+ daemon_printf("\n");
|
|
+ }
|
|
+
|
|
+ if (site_info && (str = fopen(site_info, "r"))) {
|
|
+ while ((fgets(buffer, 1000, str)))
|
|
+ daemon_printf("%s", buffer);
|
|
+ fclose(str);
|
|
+ }
|
|
+ daemon_printf(".\n");
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+}
|
|
+
|
|
+static void daemon_show(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+}
|
|
+
|
|
+static void daemon_option_mime(const char *cmdline, int argc,
|
|
+ const char **argv)
|
|
+{
|
|
+ ++daemonMime;
|
|
+ daemon_ok(CODE_OK, "ok - using MIME headers", NULL);
|
|
+}
|
|
+
|
|
+static void daemon_option(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+}
|
|
+
|
|
+static void daemon_client(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ const char *pt = strchr(cmdline, ' ');
|
|
+
|
|
+ if (pt)
|
|
+ daemon_log(DICT_LOG_CLIENT, "%.200s\n", pt + 1);
|
|
+ else
|
|
+ daemon_log(DICT_LOG_CLIENT, "%.200s\n", cmdline);
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
+}
|
|
+
|
|
+static void daemon_auth(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ char *buf;
|
|
+ hsh_HashTable h = DictConfig->usl;
|
|
+ const char *secret;
|
|
+ struct MD5Context ctx;
|
|
+ unsigned char digest[16];
|
|
+ char hex[33];
|
|
+ int i;
|
|
+ int buf_size;
|
|
+
|
|
+ if (argc != 3)
|
|
+ daemon_printf("%d syntax error, illegal parameters\n",
|
|
+ CODE_ILLEGAL_PARAM);
|
|
+ if (!h || !(secret = hsh_retrieve(h, argv[1]))) {
|
|
+ daemon_log(DICT_LOG_AUTH, "%s@%s/%s denied: invalid username\n",
|
|
+ argv[1], daemonHostname, daemonIP);
|
|
+ daemon_printf("%d auth denied\n", CODE_AUTH_DENIED);
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ buf_size = strlen(daemonStamp) + strlen(secret) + 10;
|
|
+ buf = alloca(buf_size);
|
|
+ snprintf(buf, buf_size, "%s%s", daemonStamp, secret);
|
|
+
|
|
+ MD5Init(&ctx);
|
|
+ MD5Update(&ctx, (const unsigned char *) buf, strlen(buf));
|
|
+ MD5Final(digest, &ctx);
|
|
+
|
|
+ for (i = 0; i < 16; i++)
|
|
+ snprintf(hex + 2 * i, 3, "%02x", digest[i]);
|
|
+
|
|
+ hex[32] = '\0';
|
|
+
|
|
+ PRINTF(DBG_AUTH, ("Got %s expected %s\n", argv[2], hex));
|
|
+
|
|
+ if (strcmp(hex, argv[2])) {
|
|
+ daemon_log(DICT_LOG_AUTH, "%s@%s/%s denied: hash mismatch\n",
|
|
+ argv[1], daemonHostname, daemonIP);
|
|
+ daemon_printf("%d auth denied\n", CODE_AUTH_DENIED);
|
|
+ } else {
|
|
+ daemon_printf("%d authenticated\n", CODE_AUTH_OK);
|
|
+ daemon_check_auth(argv[1]);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void daemon_status(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ daemon_ok(CODE_STATUS, "status", "t");
|
|
+}
|
|
+
|
|
+static void daemon_help(const char *cmdline, int argc, const char **argv)
|
|
+{
|
|
+ daemon_printf("%d help text follows\n", CODE_HELP);
|
|
+ daemon_mime();
|
|
+ daemon_text
|
|
+ ("DEFINE database word -- look up word in database\n"
|
|
+ "MATCH database strategy word -- match word in database using strategy\n"
|
|
+ "SHOW DB -- list all accessible databases\n"
|
|
+ "SHOW DATABASES -- list all accessible databases\n"
|
|
+ "SHOW STRAT -- list available matching strategies\n"
|
|
+ "SHOW STRATEGIES -- list available matching strategies\n"
|
|
+ "SHOW INFO database -- provide information about the database\n"
|
|
+ "SHOW SERVER -- provide site-specific information\n"
|
|
+ "OPTION MIME -- use MIME headers\n"
|
|
+ "CLIENT info -- identify client to server\n"
|
|
+ "AUTH user string -- provide authentication information\n"
|
|
+ "STATUS -- display timing information\n"
|
|
+ "HELP -- display this help information\n"
|
|
+ "QUIT -- terminate connection\n\n"
|
|
+ "The following commands are unofficial server extensions for debugging\n"
|
|
+ "only. You may find them useful if you are using telnet as a client.\n"
|
|
+ "If you are writing a client, you MUST NOT use these commands, since\n"
|
|
+ "they won't be supported on any other server!\n\n"
|
|
+ "D word -- DEFINE * word\n"
|
|
+ "D database word -- DEFINE database word\n"
|
|
+ "M word -- MATCH * . word\n"
|
|
+ "M strategy word -- MATCH * strategy word\n"
|
|
+ "M database strategy word -- MATCH database strategy word\n"
|
|
+ "S -- STATUS\n"
|
|
+ "H -- HELP\n"
|
|
+ "Q -- QUIT\n", 1);
|
|
+ daemon_ok(CODE_OK, "ok", NULL);
|
|
}
|
|
|
|
-static void daemon_quit( const char *cmdline, int argc, const char **argv )
|
|
+static void daemon_quit(const char *cmdline, int argc, const char **argv)
|
|
{
|
|
- daemon_ok( CODE_GOODBYE, "bye", "t" );
|
|
- daemon_terminate( 0, "quit" );
|
|
+ daemon_ok(CODE_GOODBYE, "bye", "t");
|
|
+ daemon_terminate(0, "quit");
|
|
}
|
|
|
|
/* The whole sub should be moved here, but I want to keep the diff small. */
|
|
-int _handleconn (int error);
|
|
+int _handleconn(int error);
|
|
|
|
-int dict_inetd (int error)
|
|
+int dict_inetd(int error)
|
|
{
|
|
- if (setjmp(env)) return 0;
|
|
+ if (setjmp(env))
|
|
+ return 0;
|
|
|
|
- daemonPort = -1;
|
|
- daemonIP = "inetd";
|
|
+ daemonPort = -1;
|
|
+ daemonIP = "inetd";
|
|
|
|
- daemonHostname = daemonIP;
|
|
+ daemonHostname = daemonIP;
|
|
|
|
- daemonS_in = 0;
|
|
- daemonS_out = 1;
|
|
+ daemonS_in = 0;
|
|
+ daemonS_out = 1;
|
|
|
|
- return _handleconn (error);
|
|
+ return _handleconn(error);
|
|
}
|
|
|
|
-int dict_daemon( int s, struct sockaddr_in *csin, int error )
|
|
+int dict_daemon(int s, struct sockaddr_storage *csin, char ***argv0,
|
|
+ int delay, int error)
|
|
{
|
|
- struct hostent *h;
|
|
-
|
|
- if (setjmp(env)) return 0;
|
|
+ static char hostname[NI_MAXHOST], service[NI_MAXSERV];
|
|
+
|
|
+ if (setjmp(env))
|
|
+ return 0;
|
|
|
|
- daemonPort = ntohs(csin->sin_port);
|
|
- daemonIP = str_find( inet_ntoa(csin->sin_addr) );
|
|
- if ((h = gethostbyaddr((void *)&csin->sin_addr,
|
|
- sizeof(csin->sin_addr), csin->sin_family))) {
|
|
- daemonHostname = str_find( h->h_name );
|
|
- } else
|
|
- daemonHostname = daemonIP;
|
|
+ getnameinfo((const struct sockaddr *) csin,
|
|
+ sizeof(struct sockaddr_storage), hostname, NI_MAXHOST,
|
|
+ service, NI_MAXSERV, NI_NUMERICSERV);
|
|
|
|
- daemonS_in = s;
|
|
- daemonS_out = s;
|
|
+ daemonPort = strtol(service, NULL, 10);
|
|
+ daemonIP = str_find(inet_ntopW((struct sockaddr *) csin));
|
|
+ daemonHostname = str_find(hostname);
|
|
+ daemonS_in = daemonS_out = s;
|
|
|
|
- return _handleconn (error);
|
|
+ return _handleconn(delay, error);
|
|
}
|
|
|
|
-int _handleconn (int error) {
|
|
- int query_count = 0;
|
|
- char buf[4096];
|
|
- int count;
|
|
- arg_List cmdline;
|
|
- int argc;
|
|
- char **argv;
|
|
- void (*command)(const char *, int, const char **);
|
|
+int _handleconn(int error)
|
|
+{
|
|
+ int query_count = 0;
|
|
+ char buf[4096];
|
|
+ int count;
|
|
+ arg_List cmdline;
|
|
+ int argc;
|
|
+ char **argv;
|
|
+ void (*command) (const char *, int, const char **);
|
|
|
|
- _dict_defines = 0;
|
|
- _dict_matches = 0;
|
|
- _dict_comparisons = 0;
|
|
+ _dict_defines = 0;
|
|
+ _dict_matches = 0;
|
|
+ _dict_comparisons = 0;
|
|
|
|
- tim_start( "t" );
|
|
- daemon_log( DICT_LOG_TRACE, "connected\n" );
|
|
- daemon_log( DICT_LOG_CONNECT, "%s/%s connected on port %d\n",
|
|
- daemonHostname, daemonIP, daemonPort );
|
|
- dict_setproctitle( "dictd: %s connected", daemonHostname );
|
|
+ tim_start("t");
|
|
+ daemon_log(DICT_LOG_TRACE, "connected\n");
|
|
+ daemon_log(DICT_LOG_CONNECT, "%s/%s connected on port %d\n",
|
|
+ daemonHostname, daemonIP, daemonPort);
|
|
+ dict_setproctitle("dictd: %s connected", daemonHostname);
|
|
|
|
- if (error) {
|
|
- daemon_printf( "%d server temporarily unavailable\n",
|
|
- CODE_TEMPORARILY_UNAVAILABLE );
|
|
- daemon_terminate( 0, "temporarily unavailable" );
|
|
- }
|
|
+ if (error) {
|
|
+ daemon_printf("%d server temporarily unavailable\n",
|
|
+ CODE_TEMPORARILY_UNAVAILABLE);
|
|
+ daemon_terminate(0, "temporarily unavailable");
|
|
+ }
|
|
|
|
- if (daemon_check_auth( NULL )) {
|
|
- daemon_log( DICT_LOG_AUTH, "%s/%s denied: ip/hostname rules\n",
|
|
- daemonHostname, daemonIP );
|
|
- daemon_printf( "%d access denied\n", CODE_ACCESS_DENIED );
|
|
- daemon_terminate( 0, "access denied" );
|
|
- }
|
|
+ if (daemon_check_auth(NULL)) {
|
|
+ daemon_log(DICT_LOG_AUTH, "%s/%s denied: ip/hostname rules\n",
|
|
+ daemonHostname, daemonIP);
|
|
+ daemon_printf("%d access denied\n", CODE_ACCESS_DENIED);
|
|
+ daemon_terminate(0, "access denied");
|
|
+ }
|
|
|
|
- daemon_banner();
|
|
+ daemon_banner();
|
|
|
|
- if (!_dict_daemon_limit_time)
|
|
- alarm (client_delay);
|
|
+ if (!_dict_daemon_limit_time)
|
|
+ alarm(client_delay);
|
|
|
|
- while (count = daemon_read( buf, 4000 ), count >= 0) {
|
|
- ++query_count;
|
|
+ while (count = daemon_read(buf, 4000), count >= 0) {
|
|
+ ++query_count;
|
|
|
|
- if (_dict_daemon_limit_queries &&
|
|
- query_count >= _dict_daemon_limit_queries)
|
|
- {
|
|
- daemon_terminate (0, "query limit");
|
|
- }
|
|
+ if (_dict_daemon_limit_queries &&
|
|
+ query_count >= _dict_daemon_limit_queries) {
|
|
+ daemon_terminate(0, "query limit");
|
|
+ }
|
|
|
|
- if (stdin2stdout_mode){
|
|
- daemon_printf( "# %s\n", buf );
|
|
- }
|
|
+ if (stdin2stdout_mode) {
|
|
+ daemon_printf("# %s\n", buf);
|
|
+ }
|
|
|
|
- if (!_dict_daemon_limit_time)
|
|
- alarm(0);
|
|
+ if (!_dict_daemon_limit_time)
|
|
+ alarm(0);
|
|
|
|
- tim_start( "c" );
|
|
- if (!count) {
|
|
+ tim_start("c");
|
|
+ if (!count) {
|
|
#if 0
|
|
- daemon_ok( CODE_OK, "ok", "c" );
|
|
+ daemon_ok(CODE_OK, "ok", "c");
|
|
#endif
|
|
- continue;
|
|
- }
|
|
+ continue;
|
|
+ }
|
|
|
|
- daemon_log( DICT_LOG_COMMAND, "%.80s\n", buf );
|
|
- cmdline = arg_argify(buf,0);
|
|
- arg_get_vector( cmdline, &argc, &argv );
|
|
- if ((command = lookup_command (argc, (const char **) argv))) {
|
|
- command(buf, argc, (const char **) argv);
|
|
- } else {
|
|
- daemon_printf( "%d unknown command\n", CODE_SYNTAX_ERROR );
|
|
- }
|
|
- arg_destroy(cmdline);
|
|
-
|
|
- if (!_dict_daemon_limit_time)
|
|
- alarm (client_delay);
|
|
- }
|
|
+ daemon_log(DICT_LOG_COMMAND, "%.80s\n", buf);
|
|
+ cmdline = arg_argify(buf, 0);
|
|
+ arg_get_vector(cmdline, &argc, &argv);
|
|
+ if ((command = lookup_command(argc, (const char **) argv))) {
|
|
+ command(buf, argc, (const char **) argv);
|
|
+ } else {
|
|
+ daemon_printf("%d unknown command\n", CODE_SYNTAX_ERROR);
|
|
+ }
|
|
+ arg_destroy(cmdline);
|
|
+
|
|
+ if (!_dict_daemon_limit_time)
|
|
+ alarm(client_delay);
|
|
+ }
|
|
#if 0
|
|
- printf( "%d %d\n", count, errno );
|
|
+ printf("%d %d\n", count, errno);
|
|
#endif
|
|
- daemon_terminate( 0, "close" );
|
|
- return 0;
|
|
+ daemon_terminate(0, "close");
|
|
+ return 0;
|
|
}
|
|
--- a/net.c
|
|
+++ b/net.c
|
|
@@ -1,18 +1,18 @@
|
|
-/* net.c --
|
|
+/* net.c --
|
|
* Created: Fri Feb 21 20:58:10 1997 by faith@dict.org
|
|
* Copyright 1997, 1998, 1999, 2000, 2002 Rickard E. Faith (faith@dict.org)
|
|
* Copyright 2002-2008 Aleksey Cheusov (vle@gmx.net)
|
|
- *
|
|
+ *
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 1, or (at your option) any
|
|
* later version.
|
|
- *
|
|
+ *
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
- *
|
|
+ *
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
@@ -25,7 +25,7 @@
|
|
#include "dictP.h"
|
|
|
|
#if HAVE_SYS_PARAM_H
|
|
-# include <sys/param.h>
|
|
+#include <sys/param.h>
|
|
#endif
|
|
|
|
#include <fcntl.h>
|
|
@@ -42,146 +42,180 @@
|
|
|
|
static char netHostname[MAXHOSTNAMELEN];
|
|
|
|
-const char *net_hostname( void )
|
|
+const char *inet_ntopW(struct sockaddr *sa)
|
|
{
|
|
- struct hostent *hostEntry;
|
|
- static char *hostname = NULL;
|
|
-
|
|
- if (!netHostname[0]) {
|
|
- memset( netHostname, 0, sizeof(netHostname) );
|
|
- gethostname( netHostname, sizeof(netHostname)-1 );
|
|
-
|
|
- if ((hostEntry = gethostbyname(netHostname))) {
|
|
- hostname = xstrdup(hostEntry->h_name);
|
|
- } else {
|
|
- hostname = xstrdup(netHostname);
|
|
- }
|
|
- }
|
|
-
|
|
- return hostname;
|
|
-}
|
|
-
|
|
-int net_connect_tcp( const char *host, const char *service )
|
|
-{
|
|
- struct hostent *hostEntry;
|
|
- struct servent *serviceEntry;
|
|
- struct protoent *protocolEntry;
|
|
- struct sockaddr_in ssin;
|
|
- int s;
|
|
- int hosts = 0;
|
|
- char **current;
|
|
-
|
|
- memset( &ssin, 0, sizeof(ssin) );
|
|
- ssin.sin_family = AF_INET;
|
|
-
|
|
- if ((serviceEntry = getservbyname(service, "tcp"))) {
|
|
- ssin.sin_port = serviceEntry->s_port;
|
|
- } else if (!(ssin.sin_port = htons(atoi(service))))
|
|
- return NET_NOSERVICE;
|
|
-
|
|
- if (!(protocolEntry = getprotobyname("tcp")))
|
|
- return NET_NOPROTOCOL;
|
|
-
|
|
- if ((hostEntry = gethostbyname(host))) {
|
|
- ++hosts;
|
|
- } else if ((ssin.sin_addr.s_addr = inet_addr(host)) == htonl(INADDR_NONE))
|
|
- return NET_NOHOST;
|
|
-
|
|
- if (hosts) {
|
|
- for (current = hostEntry->h_addr_list; *current; current++) {
|
|
- memcpy( &ssin.sin_addr.s_addr, *current, hostEntry->h_length );
|
|
- PRINTF(DBG_VERBOSE,
|
|
- ("Trying %s (%s)\n",host,inet_ntoa(ssin.sin_addr)));
|
|
- if ((s = socket(PF_INET, SOCK_STREAM, protocolEntry->p_proto)) < 0)
|
|
- err_fatal_errno( __func__, "Can't open socket on port %d\n",
|
|
- ntohs(ssin.sin_port) );
|
|
- if (connect(s, (struct sockaddr *)&ssin, sizeof(ssin)) >= 0)
|
|
+ static char str[40];
|
|
+
|
|
+ switch (sa->sa_family) {
|
|
+ case AF_INET:
|
|
+ return inet_ntop(sa->sa_family,
|
|
+ &(((struct sockaddr_in *) sa)->sin_addr), str,
|
|
+ 16);
|
|
+ case AF_INET6:
|
|
+ return inet_ntop(sa->sa_family,
|
|
+ &(((struct sockaddr_in6 *) sa)->sin6_addr), str,
|
|
+ 40);
|
|
+ default:
|
|
+ errno = EAFNOSUPPORT;
|
|
+ return NULL;
|
|
+ }
|
|
+}
|
|
+
|
|
+const char *net_hostname(void)
|
|
+{
|
|
+ struct hostent *hostEntry;
|
|
+ static char *hostname = NULL;
|
|
+
|
|
+ if (!netHostname[0]) {
|
|
+ memset(netHostname, 0, sizeof(netHostname));
|
|
+ gethostname(netHostname, sizeof(netHostname) - 1);
|
|
+
|
|
+ if ((hostEntry = gethostbyname(netHostname))) {
|
|
+ hostname = xstrdup(hostEntry->h_name);
|
|
+ } else {
|
|
+ hostname = xstrdup(netHostname);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ return hostname;
|
|
+}
|
|
+
|
|
+int net_connect_tcp(const char *host, const char *service)
|
|
+{
|
|
+ struct addrinfo hints, *r, *rtmp;
|
|
+ int s;
|
|
+
|
|
+ memset(&hints, 0, sizeof(struct addrinfo));
|
|
+ hints.ai_family = PF_UNSPEC;
|
|
+ hints.ai_protocol = IPPROTO_TCP;
|
|
+ hints.ai_socktype = SOCK_STREAM;
|
|
+ hints.ai_flags = AI_ADDRCONFIG;
|
|
+
|
|
+ if (getaddrinfo(host, service, &hints, &r) != 0)
|
|
+ return NET_NOHOST;
|
|
+
|
|
+ for (rtmp = r; r != NULL; r = r->ai_next) {
|
|
+ s = socket(r->ai_family, r->ai_socktype, r->ai_protocol);
|
|
+ if (s < 0) {
|
|
+ if (r->ai_next != NULL)
|
|
+ continue;
|
|
+ else
|
|
+ err_fatal_errno(__FUNCTION__, "Can't open socket\n");
|
|
+ }
|
|
+
|
|
+ PRINTF(DBG_VERBOSE,
|
|
+ ("Trying %s (%s)...", host, inet_ntopW(r->ai_addr)));
|
|
+
|
|
+ if (connect(s, r->ai_addr, r->ai_addrlen) >= 0) {
|
|
+ PRINTF(DBG_VERBOSE, ("Connected."));
|
|
+ freeaddrinfo(rtmp);
|
|
return s;
|
|
- close(s);
|
|
- }
|
|
- } else {
|
|
- if ((s = socket(PF_INET, SOCK_STREAM, protocolEntry->p_proto)) < 0)
|
|
- err_fatal_errno( __func__, "Can't open socket on port %d\n",
|
|
- ntohs(ssin.sin_port) );
|
|
- if (connect(s, (struct sockaddr *)&ssin, sizeof(ssin)) >= 0)
|
|
- return s;
|
|
- close(s);
|
|
- }
|
|
-
|
|
- return NET_NOCONNECT;
|
|
-}
|
|
-
|
|
-int net_open_tcp (
|
|
- const char *address,
|
|
- const char *service,
|
|
- int queueLength)
|
|
-{
|
|
- struct servent *serviceEntry;
|
|
- struct protoent *protocolEntry;
|
|
- struct sockaddr_in ssin;
|
|
- int s;
|
|
- const int one = 1;
|
|
-
|
|
- memset( &ssin, 0, sizeof(ssin) );
|
|
- ssin.sin_family = AF_INET;
|
|
- ssin.sin_addr.s_addr = address ? inet_addr(address) : htonl(INADDR_ANY);
|
|
-
|
|
- if ((serviceEntry = getservbyname(service, "tcp"))) {
|
|
- ssin.sin_port = serviceEntry->s_port;
|
|
- } else if (!(ssin.sin_port = htons(atoi(service))))
|
|
- err_fatal( __func__, "Can't get \"%s\" service entry\n", service );
|
|
-
|
|
- if (!(protocolEntry = getprotobyname("tcp")))
|
|
- err_fatal( __func__, "Can't get \"tcp\" protocol entry\n" );
|
|
-
|
|
- if ((s = socket(PF_INET, SOCK_STREAM, protocolEntry->p_proto)) < 0)
|
|
- err_fatal_errno( __func__, "Can't open socket on port %d\n",
|
|
- ntohs(ssin.sin_port) );
|
|
-
|
|
- setsockopt( s, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one) );
|
|
-
|
|
- if (bind(s, (struct sockaddr *)&ssin, sizeof(ssin)) < 0)
|
|
- err_fatal_errno( __func__, "Can't bind %s/tcp to port %d\n",
|
|
- service, ntohs(ssin.sin_port) );
|
|
-
|
|
- if (listen( s, queueLength ) < 0)
|
|
- err_fatal_errno( __func__, "Can't listen to %s/tcp on port %d\n",
|
|
- service, ntohs(ssin.sin_port) );
|
|
-
|
|
- return s;
|
|
-}
|
|
-
|
|
-int net_read( int s, char *buf, int maxlen )
|
|
-{
|
|
- int len;
|
|
- int n = 0;
|
|
- char c;
|
|
- char *pt = buf;
|
|
-
|
|
- *pt = '\0';
|
|
-
|
|
- for (len = 0; len < maxlen && (n = read( s, &c, 1 )) > 0; /*void*/) {
|
|
- switch (c) {
|
|
- case '\n': *pt = '\0'; return len;
|
|
- case '\r': break;
|
|
- default: *pt++ = c; ++len; break;
|
|
- }
|
|
- }
|
|
- *pt = '\0';
|
|
- if (!n) return len ? len : EOF;
|
|
- return n; /* error code */
|
|
-}
|
|
-
|
|
-int net_write( int s, const char *buf, int len )
|
|
-{
|
|
- int left = len;
|
|
- int count;
|
|
-
|
|
- while (left) {
|
|
- if ((count = write(s, buf, left)) != left) {
|
|
- if (count <= 0) return count; /* error code */
|
|
- }
|
|
- left -= count;
|
|
- }
|
|
- return len;
|
|
+ }
|
|
+
|
|
+ PRINTF(DBG_VERBOSE, ("Failed: %s\n", strerror(errno)));
|
|
+
|
|
+ close(s);
|
|
+ }
|
|
+ freeaddrinfo(rtmp);
|
|
+
|
|
+ return NET_NOCONNECT;
|
|
+}
|
|
+
|
|
+int net_open_tcp(const char *address, const char *service, int queueLength)
|
|
+{
|
|
+ struct addrinfo hints, *r, *rtmp;
|
|
+ int s = -1;
|
|
+
|
|
+ memset(&hints, 0, sizeof(struct addrinfo));
|
|
+ hints.ai_family = PF_UNSPEC;
|
|
+ hints.ai_protocol = IPPROTO_TCP;
|
|
+ hints.ai_socktype = SOCK_STREAM;
|
|
+ hints.ai_flags = AI_PASSIVE;
|
|
+
|
|
+ if (getaddrinfo(address, service, &hints, &r) != 0)
|
|
+ err_fatal(__FUNCTION__,
|
|
+ "getaddrinfo: Failed, address = \"%s\", service = \"%s\"\n",
|
|
+ address, service);
|
|
+
|
|
+ for (rtmp = r; r != NULL; r = r->ai_next) {
|
|
+ s = socket(r->ai_family, r->ai_socktype, r->ai_protocol);
|
|
+
|
|
+ if (s < 0) {
|
|
+ if (r->ai_next != NULL)
|
|
+ continue;
|
|
+ freeaddrinfo(rtmp);
|
|
+ err_fatal_errno(__FUNCTION__, "Can't open socket\n");
|
|
+ }
|
|
+
|
|
+ {
|
|
+ const int one = 1;
|
|
+ setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));
|
|
+ }
|
|
+
|
|
+ if (bind(s, r->ai_addr, r->ai_addrlen) < 0) {
|
|
+ if (r->ai_next != NULL) {
|
|
+ close(s);
|
|
+ continue;
|
|
+ }
|
|
+ freeaddrinfo(rtmp);
|
|
+ err_fatal_errno(__FUNCTION__, "Can't bind %s/tcp to %s\n",
|
|
+ service, address ? address : "ANY");
|
|
+ }
|
|
+
|
|
+ if (listen(s, queueLength) < 0) {
|
|
+ if (r->ai_next != NULL) {
|
|
+ close(s);
|
|
+ continue;
|
|
+ }
|
|
+ freeaddrinfo(rtmp);
|
|
+ err_fatal_errno(__FUNCTION__, "Can't listen to %s/tcp on %s\n",
|
|
+ service, address);
|
|
+ }
|
|
+ }
|
|
+ freeaddrinfo(rtmp);
|
|
+
|
|
+ return s;
|
|
+}
|
|
+
|
|
+int net_read(int s, char *buf, int maxlen)
|
|
+{
|
|
+ int len;
|
|
+ int n = 0;
|
|
+ char c;
|
|
+ char *pt = buf;
|
|
+
|
|
+ *pt = '\0';
|
|
+
|
|
+ for (len = 0; len < maxlen && (n = read(s, &c, 1)) > 0; /*void */ ) {
|
|
+ switch (c) {
|
|
+ case '\n':
|
|
+ *pt = '\0';
|
|
+ return len;
|
|
+ case '\r':
|
|
+ break;
|
|
+ default:
|
|
+ *pt++ = c;
|
|
+ ++len;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+ *pt = '\0';
|
|
+ if (!n)
|
|
+ return len ? len : EOF;
|
|
+ return n; /* error code */
|
|
+}
|
|
+
|
|
+int net_write(int s, const char *buf, int len)
|
|
+{
|
|
+ int left = len;
|
|
+ int count;
|
|
+
|
|
+ while (left) {
|
|
+ if ((count = write(s, buf, left)) != left) {
|
|
+ if (count <= 0)
|
|
+ return count; /* error code */
|
|
+ }
|
|
+ left -= count;
|
|
+ }
|
|
+ return len;
|
|
}
|
|
--- a/net.h
|
|
+++ b/net.h
|
|
@@ -18,14 +18,14 @@
|
|
* 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
-
|
|
-extern const char *net_hostname( void );
|
|
-extern int net_connect_tcp( const char *host, const char *service );
|
|
-extern int net_open_tcp (const char *address,
|
|
- const char *service, int queueLength);
|
|
-extern void net_detach( void );
|
|
-extern int net_read( int s, char *buf, int maxlen );
|
|
-extern int net_write( int s, const char *buf, int len );
|
|
+extern const char *inet_ntopW(struct sockaddr *sa);
|
|
+extern const char *net_hostname(void);
|
|
+extern int net_connect_tcp(const char *host, const char *service);
|
|
+extern int net_open_tcp(const char *address,
|
|
+ const char *service, int queueLength);
|
|
+extern void net_detach(void);
|
|
+extern int net_read(int s, char *buf, int maxlen);
|
|
+extern int net_write(int s, const char *buf, int len);
|
|
|
|
#define NET_NOHOST (-1)
|
|
#define NET_NOSERVICE (-2)
|