-.\" $Id: apropos.1,v 1.6 2011/11/20 16:29:50 kristaps Exp $
+.\" $Id: apropos.1,v 1.7 2011/11/20 21:36:00 kristaps Exp $
.\"
.\" Copyright (c) 2011 Kristaps Dzonsons <kristaps@bsd.lv>
.\"
.Nd search manual page databases
.Sh SYNOPSIS
.Nm
+.Op Fl M Ar manpath
.Op Fl m Ar manpath
.Op Fl S Ar arch
.Op Fl s Ar section
for each file in each database.
Its arguments are as follows:
.Bl -tag -width Ds
+.It Fl M Ar manpath
+Use the colon-separated path instead of the default list of paths
+searched for
+.Xr mandocdb 8
+databases.
+Invalid paths, or paths without manual databases, are ignored.
.It Fl m Ar manpath
-Append the colon-separated paths to the default list of paths searched
+Append the colon-separated paths to the list of paths searched
for
.Xr mandocdb 8
databases.
.Pp
By default,
.Nm
-searches for a
+searches for
.Xr mandocdb 8
-database in the current working directory and
+databases in the default paths stipulated by
+.Xr man 1 ,
parses terms as case-sensitive regular expressions
.Pq the Li \&~ operator
over manual names and descriptions
Comma-separated paths overriding the default list of paths searched for
manual databases.
Invalid paths, or paths without manual databases, are ignored.
+Takes precedence over
+.Fl M .
.El
.\" .Sh FILES
.Sh EXIT STATUS
-/* $Id: apropos.c,v 1.16 2011/11/20 16:29:50 kristaps Exp $ */
+/* $Id: apropos.c,v 1.17 2011/11/20 21:36:00 kristaps Exp $ */
/*
* Copyright (c) 2011 Kristaps Dzonsons <kristaps@bsd.lv>
* Copyright (c) 2011 Ingo Schwarze <schwarze@openbsd.org>
#endif
#include <assert.h>
+#include <ctype.h>
#include <getopt.h>
#include <limits.h>
#include <stdio.h>
#include "apropos_db.h"
#include "mandoc.h"
+/*
+ * FIXME: add support for manpath(1), which everybody but OpenBSD and
+ * NetBSD seem to use.
+ */
+#define MAN_CONF_FILE "/etc/man.conf"
+#define MAN_CONF_KEY "_whatdb"
+
/*
* List of paths to be searched for manual databases.
*/
static void list(struct res *, size_t, void *);
static void manpath_add(struct manpaths *, const char *);
static void manpath_parse(struct manpaths *, char *);
+static void manpath_parseconf(struct manpaths *);
static void usage(void);
static char *progname;
size_t terms;
struct opts opts;
struct expr *e;
+ char *defpaths, *auxpaths;
extern int optind;
extern char *optarg;
memset(&paths, 0, sizeof(struct manpaths));
memset(&opts, 0, sizeof(struct opts));
+ auxpaths = defpaths = NULL;
e = NULL;
rc = 0;
- while (-1 != (ch = getopt(argc, argv, "m:S:s:")))
+ while (-1 != (ch = getopt(argc, argv, "M:m:S:s:")))
switch (ch) {
+ case ('M'):
+ defpaths = optarg;
+ break;
case ('m'):
- manpath_parse(&paths, optarg);
+ auxpaths = optarg;
break;
case ('S'):
opts.arch = optarg;
goto out;
}
- /*
- * Let MANPATH override our default paths.
- */
-
if (NULL != getenv("MANPATH"))
- manpath_add(&paths, getenv("MANPATH"));
+ defpaths = getenv("MANPATH");
+
+ if (NULL == defpaths)
+ manpath_parseconf(&paths);
else
- manpath_add(&paths, ".");
+ manpath_parse(&paths, defpaths);
+
+ manpath_parse(&paths, auxpaths);
if (NULL == (e = exprcomp(argc, argv, &terms))) {
fprintf(stderr, "%s: Bad expression\n", progname);
{
fprintf(stderr, "usage: %s "
+ "[-M dirs] "
"[-m dirs] "
"[-S arch] "
"[-s section] "
{
char *dir;
+ if (NULL == path)
+ return;
+
for (dir = strtok(path, ":"); dir; dir = strtok(NULL, ":"))
manpath_add(dirs, dir);
}
{
char buf[PATH_MAX];
char *cp;
+ int i;
if (NULL == (cp = realpath(dir, buf)))
return;
+ for (i = 0; i < dirs->sz; i++)
+ if (0 == strcmp(dirs->paths[i], dir))
+ return;
+
dirs->paths = mandoc_realloc
(dirs->paths,
((size_t)dirs->sz + 1) * sizeof(char *));
dirs->paths[dirs->sz++] = mandoc_strdup(cp);
}
+
+static void
+manpath_parseconf(struct manpaths *dirs)
+{
+ FILE *stream;
+#ifdef USE_MANPATH
+ char *buf;
+ size_t sz, bsz;
+
+ stream = popen("manpath", "r");
+ if (NULL == stream)
+ return;
+
+ buf = NULL;
+ bsz = 0;
+
+ do {
+ buf = mandoc_realloc(buf, bsz + 1024);
+ sz = fread(buf + (int)bsz, 1, 1024, stream);
+ bsz += sz;
+ } while (sz > 0);
+
+ assert(bsz && '\n' == buf[bsz - 1]);
+ buf[bsz - 1] = '\0';
+
+ manpath_parse(dirs, buf);
+ free(buf);
+ pclose(stream);
+#else
+ char *p, *q;
+ size_t len, keysz;
+
+ keysz = strlen(MAN_CONF_KEY);
+ assert(keysz > 0);
+
+ if (NULL == (stream = fopen(MAN_CONF_FILE, "r")))
+ return;
+
+ while (NULL != (p = fgetln(stream, &len))) {
+ if (0 == len || '\n' == p[--len])
+ break;
+ p[len] = '\0';
+ while (isspace((unsigned char)*p))
+ p++;
+ if (strncmp(MAN_CONF_KEY, p, keysz))
+ continue;
+ p += keysz;
+ while (isspace(*p))
+ p++;
+ if ('\0' == *p)
+ continue;
+ if (NULL == (q = strrchr(p, '/')))
+ continue;
+ *q = '\0';
+ manpath_add(dirs, p);
+ }
+
+ fclose(stream);
+#endif
+}