1
0
Fork 0
man-db/lib/filenames.c
Daniel Baumann 1fa764a8d3
Adding upstream version 2.13.1.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-21 08:13:55 +02:00

165 lines
4.2 KiB
C

/*
* filenames.c: compose and dissect man page file names
*
* Copyright (C) 1994, 1995 Graeme W. Wilford. (Wilf.)
* Copyright (C) 2001, 2002 Colin Watson.
*
* This file is part of man-db.
*
* man-db 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 2 of the License, or
* (at your option) any later version.
*
* man-db 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 man-db; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif /* HAVE_CONFIG_H */
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "dirname.h"
#include "error.h"
#include "xalloc.h"
#include "xvasprintf.h"
#include "gettext.h"
#define _(String) gettext (String)
#include "manconfig.h"
#include "appendstr.h"
#include "compression.h"
#include "debug.h"
#include "filenames.h"
static void gripe_bogus_manpage (const char *manpage)
{
error (0, 0, _ ("warning: %s: ignoring bogus filename"), manpage);
}
char *make_filename (const char *path, const char *name, struct mandata *in,
const char *type)
{
static char *file;
if (!name)
name = in->name; /* comes from dblookup(), so non-NULL */
file = xasprintf ("%s/%s%s/%s.%s", path, type, in->sec, name, in->ext);
if (in->comp && *in->comp != '-') /* Is there an extension? */
file = appendstr (file, ".", in->comp, nullptr);
debug ("Checking physical location: %s\n", file);
if (!CAN_ACCESS (file, R_OK)) {
free (file);
return NULL;
}
return file;
}
/* Fill in a mandata structure with information about a file name.
* file is the name to examine. info points to the structure to be filled
* in.
*
* Returns either a pointer to the buffer which the fields in info point
* into, to be freed by the caller, or NULL on error. The buffer will
* contain either three or four null-terminated strings: the directory name,
* the base of the file name in that directory, the section extension, and
* optionally the compression extension.
*
* Only the fields name, ext, sec, and comp are filled in by this function.
*/
struct mandata *filename_info (const char *file, bool warn_if_bogus)
{
struct mandata *info;
char *basename, *dirname;
struct compression *comp;
info = XZALLOC (struct mandata);
basename = base_name (file);
/* Bogus files either have (i) no period, ie no extension, (ii)
a compression extension, but no sectional extension, (iii)
a mismatch between the section they are under and the
sectional part of their extension. */
comp = comp_info (basename, true);
if (comp) {
info->comp = xstrdup (comp->ext);
*(basename + strlen (comp->stem)) = '\0';
free (comp->stem);
} else
info->comp = NULL;
{
char *ext = strrchr (basename, '.');
if (!ext) {
/* no section extension */
if (warn_if_bogus)
gripe_bogus_manpage (file);
free (basename);
free_mandata_struct (info);
return NULL;
}
*ext++ = '\0'; /* set section ext */
info->ext = xstrdup (ext);
if (!*info->ext) {
/* zero-length section extension */
if (warn_if_bogus)
gripe_bogus_manpage (file);
free (basename);
free_mandata_struct (info);
return NULL;
}
}
/* Set section name. */
dirname = dir_name (file);
info->sec = xstrdup (strrchr (dirname, '/') + 4);
free (dirname);
if (strlen (info->sec) >= 1 && strlen (info->ext) >= 1 &&
info->sec[0] != info->ext[0]) {
/* mismatch in section */
if (warn_if_bogus)
gripe_bogus_manpage (file);
free (basename);
free_mandata_struct (info);
return NULL;
}
info->name = xstrdup (basename);
return info;
}
/* Free a mandata structure and its elements. */
void free_mandata_struct (struct mandata *pinfo)
{
if (pinfo) {
free (pinfo->name);
free (pinfo->ext);
free (pinfo->sec);
free (pinfo->pointer);
free (pinfo->comp);
free (pinfo->filter);
free (pinfo->whatis);
}
free (pinfo);
}