123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477 |
- /*
- Utilities for VFS modules.
- Copyright (C) 1988-2024
- Free Software Foundation, Inc.
- Copyright (C) 1995, 1996 Miguel de Icaza
- This file is part of the Midnight Commander.
- The Midnight Commander 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 3 of the License,
- or (at your option) any later version.
- The Midnight Commander 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, see <http://www.gnu.org/licenses/>.
- */
- /**
- * \file
- * \brief Source: Utilities for VFS modules
- * \author Miguel de Icaza
- * \date 1995, 1996
- */
- #include <config.h>
- #include <ctype.h>
- #include <sys/types.h>
- #include <pwd.h>
- #include <grp.h>
- #include <stdlib.h>
- #include <string.h>
- #if !defined (HAVE_UTIMENSAT) && defined (HAVE_UTIME_H)
- #include <utime.h>
- #endif
- #include "lib/global.h"
- #include "lib/unixcompat.h"
- #include "lib/widget.h" /* message() */
- #include "lib/strutil.h" /* INVALID_CONV */
- #include "vfs.h"
- #include "utilvfs.h"
- /*** global variables ****************************************************************************/
- /*** file scope macro definitions ****************************************************************/
- #ifndef TUNMLEN
- #define TUNMLEN 256
- #endif
- #ifndef TGNMLEN
- #define TGNMLEN 256
- #endif
- #define MC_HISTORY_VFS_PASSWORD "mc.vfs.password"
- /*
- * FIXME2, the "-993" is to reduce the chance of a hit on the first lookup.
- */
- #define GUID_DEFAULT_CONST -993
- /*** file scope type declarations ****************************************************************/
- /*** forward declarations (file scope functions) *************************************************/
- /*** file scope variables ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /*** file scope functions ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /** Get current username
- *
- * @return g_malloc()ed string with the name of the currently logged in
- * user ("anonymous" if uid is not registered in the system)
- */
- char *
- vfs_get_local_username (void)
- {
- struct passwd *p_i;
- p_i = getpwuid (geteuid ());
- /* Unknown UID, strange */
- return (p_i != NULL && p_i->pw_name != NULL) ? g_strdup (p_i->pw_name) : g_strdup ("anonymous");
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Look up a user or group name from a uid/gid, maintaining a cache.
- * FIXME, for now it's a one-entry cache.
- * This file should be modified for non-unix systems to do something
- * reasonable.
- */
- int
- vfs_finduid (const char *uname)
- {
- static int saveuid = GUID_DEFAULT_CONST;
- static char saveuname[TUNMLEN] = "\0";
- size_t uname_len;
- uname_len = strlen (uname);
- if (uname[0] != saveuname[0] /* Quick test w/o proc call */
- || strncmp (uname, saveuname, MIN (uname_len, TUNMLEN - 1)) != 0)
- {
- struct passwd *pw;
- g_strlcpy (saveuname, uname, TUNMLEN);
- pw = getpwnam (uname);
- if (pw != NULL)
- saveuid = pw->pw_uid;
- else
- {
- static int my_uid = GUID_DEFAULT_CONST;
- if (my_uid < 0)
- my_uid = getuid ();
- saveuid = my_uid;
- }
- }
- return saveuid;
- }
- /* --------------------------------------------------------------------------------------------- */
- int
- vfs_findgid (const char *gname)
- {
- static int savegid = GUID_DEFAULT_CONST;
- static char savegname[TGNMLEN] = "\0";
- size_t gname_len;
- gname_len = strlen (gname);
- if (gname[0] != savegname[0] /* Quick test w/o proc call */
- || strncmp (gname, savegname, MIN (gname_len, TGNMLEN - 1)) != 0)
- {
- struct group *gr;
- g_strlcpy (savegname, gname, TGNMLEN);
- gr = getgrnam (gname);
- if (gr != NULL)
- savegid = gr->gr_gid;
- else
- {
- static int my_gid = GUID_DEFAULT_CONST;
- if (my_gid < 0)
- my_gid = getgid ();
- savegid = my_gid;
- }
- }
- return savegid;
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Create a temporary file with a name resembling the original.
- * This is needed e.g. for local copies requested by extfs.
- * Some extfs scripts may look at the extension.
- * We also protect stupid scripts against dangerous names.
- */
- int
- vfs_mkstemps (vfs_path_t **pname_vpath, const char *prefix, const char *param_basename)
- {
- const char *p;
- GString *suffix;
- int shift;
- int fd;
- /* Strip directories */
- p = strrchr (param_basename, PATH_SEP);
- if (p == NULL)
- p = param_basename;
- else
- p++;
- /* Protection against very long names */
- shift = strlen (p) - (MC_MAXPATHLEN - 16);
- if (shift > 0)
- p += shift;
- suffix = g_string_sized_new (32);
- /* Protection against unusual characters */
- for (; *p != '\0' && *p != '#'; p++)
- if (strchr (".-_@", *p) != NULL || g_ascii_isalnum (*p))
- g_string_append_c (suffix, *p);
- fd = mc_mkstemps (pname_vpath, prefix, suffix->str);
- g_string_free (suffix, TRUE);
- return fd;
- }
- /* --------------------------------------------------------------------------------------------- */
- /** Extract the hostname and username from the path
- *
- * Format of the path is [user@]hostname:port/remote-dir, e.g.:
- *
- * ftp://sunsite.unc.edu/pub/linux
- * ftp://miguel@sphinx.nuclecu.unam.mx/c/nc
- * ftp://tsx-11.mit.edu:8192/
- * ftp://joe@foo.edu:11321/private
- * ftp://joe:password@foo.se
- *
- * @param path is an input string to be parsed
- * @param default_port is an input default port
- * @param flags are parsing modifier flags (@see vfs_url_flags_t)
- *
- * @return g_malloc()ed url info.
- * If the user is empty, e.g. ftp://@roxanne/private, and URL_USE_ANONYMOUS
- * is not set, then the current login name is supplied.
- * Return value is a g_malloc()ed structure with the pathname relative to the
- * host.
- */
- vfs_path_element_t *
- vfs_url_split (const char *path, int default_port, vfs_url_flags_t flags)
- {
- vfs_path_element_t *path_element;
- char *pcopy;
- size_t pcopy_len;
- const char *pend;
- char *colon, *at, *rest;
- path_element = g_new0 (vfs_path_element_t, 1);
- path_element->port = default_port;
- pcopy_len = strlen (path);
- pcopy = g_strndup (path, pcopy_len);
- pend = pcopy + pcopy_len;
- if ((flags & URL_NOSLASH) == 0)
- {
- char *dir;
- /* locate path component */
- dir = strchr (pcopy, PATH_SEP);
- if (dir == NULL)
- path_element->path = g_strdup (PATH_SEP_STR);
- else
- {
- path_element->path = g_strndup (dir, pcopy_len - (size_t) (dir - pcopy));
- *dir = '\0';
- }
- }
- /* search for any possible user */
- at = strrchr (pcopy, '@');
- /* We have a username */
- if (at == NULL)
- rest = pcopy;
- else
- {
- char *inner_colon;
- *at = '\0';
- inner_colon = strchr (pcopy, ':');
- if (inner_colon != NULL)
- {
- *inner_colon = '\0';
- inner_colon++;
- path_element->password = g_strdup (inner_colon);
- }
- if (*pcopy != '\0')
- path_element->user = g_strdup (pcopy);
- if (pend == at + 1)
- rest = at;
- else
- rest = at + 1;
- }
- if ((flags & URL_USE_ANONYMOUS) == 0)
- {
- g_free (path_element->user);
- path_element->user = vfs_get_local_username ();
- }
- /* Check if the host comes with a port spec, if so, chop it */
- if (*rest != '[')
- colon = strchr (rest, ':');
- else
- {
- colon = strchr (++rest, ']');
- if (colon != NULL)
- {
- colon[0] = '\0';
- colon[1] = '\0';
- colon++;
- }
- else
- {
- vfs_path_element_free (path_element);
- g_free (pcopy);
- return NULL;
- }
- }
- if (colon != NULL)
- {
- *colon = '\0';
- /* cppcheck-suppress invalidscanf */
- if (sscanf (colon + 1, "%d", &path_element->port) == 1)
- {
- if (path_element->port <= 0 || path_element->port >= 65536)
- path_element->port = default_port;
- }
- else
- while (*(++colon) != '\0')
- {
- switch (*colon)
- {
- case 'C':
- path_element->port = 1;
- break;
- case 'r':
- path_element->port = 2;
- break;
- default:
- break;
- }
- }
- }
- path_element->host = g_strdup (rest);
- g_free (pcopy);
- #ifdef HAVE_CHARSET
- path_element->dir.converter = INVALID_CONV;
- #endif
- return path_element;
- }
- /* --------------------------------------------------------------------------------------------- */
- void __attribute__((noreturn)) vfs_die (const char *m)
- {
- message (D_ERROR, _("Internal error:"), "%s", m);
- exit (EXIT_FAILURE);
- }
- /* --------------------------------------------------------------------------------------------- */
- char *
- vfs_get_password (const char *msg)
- {
- return input_dialog (msg, _("Password:"), MC_HISTORY_VFS_PASSWORD, INPUT_PASSWORD,
- INPUT_COMPLETE_NONE);
- }
- /* --------------------------------------------------------------------------------------------- */
- int
- vfs_utime (const char *path, mc_timesbuf_t *times)
- {
- #ifdef HAVE_UTIMENSAT
- return utimensat (AT_FDCWD, path, *times, AT_SYMLINK_NOFOLLOW);
- #else
- return utime (path, times);
- #endif
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- vfs_get_timespecs_from_timesbuf (mc_timesbuf_t *times, mc_timespec_t *atime, mc_timespec_t *mtime)
- {
- #ifdef HAVE_UTIMENSAT
- atime->tv_sec = (*times)[0].tv_sec;
- atime->tv_nsec = (*times)[0].tv_nsec;
- mtime->tv_sec = (*times)[1].tv_sec;
- mtime->tv_nsec = (*times)[1].tv_nsec;
- #else
- atime->tv_sec = times->actime;
- atime->tv_nsec = 0;
- mtime->tv_sec = times->modtime;
- mtime->tv_nsec = 0;
- #endif
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- vfs_get_timesbuf_from_stat (const struct stat *s, mc_timesbuf_t *times)
- {
- #ifdef HAVE_UTIMENSAT
- #ifdef HAVE_STRUCT_STAT_ST_MTIM
- /* POSIX IEEE Std 1003.1-2008 should be the preferred way
- *
- * AIX has internal type st_timespec_t conflicting with timespec, so assign per field, for details see:
- * https://github.com/libuv/libuv/pull/4404
- */
- (*times)[0].tv_sec = s->st_atim.tv_sec;
- (*times)[0].tv_nsec = s->st_atim.tv_nsec;
- (*times)[1].tv_sec = s->st_mtim.tv_sec;
- (*times)[1].tv_nsec = s->st_mtim.tv_nsec;
- #elif HAVE_STRUCT_STAT_ST_MTIMESPEC
- /* Modern BSD solution */
- (*times)[0] = s->st_atimespec;
- (*times)[1] = s->st_mtimespec;
- #elif HAVE_STRUCT_STAT_ST_MTIMENSEC
- /* Legacy BSD solution */
- (*times)[0].tv_sec = s->st_atime;
- (*times)[0].tv_nsec = s->st_atimensec;
- (*times)[1].tv_sec = s->st_mtime;
- (*times)[1].tv_nsec = s->st_mtimensec;
- #else
- #error "Found utimensat for nanosecond timestamps, but unsupported struct stat format!"
- #endif
- #else
- times->actime = s->st_atime;
- times->modtime = s->st_mtime;
- #endif
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- vfs_copy_stat_times (const struct stat *src, struct stat *dst)
- {
- dst->st_atime = src->st_atime;
- dst->st_mtime = src->st_mtime;
- dst->st_ctime = src->st_ctime;
- #ifdef HAVE_STRUCT_STAT_ST_MTIM
- dst->st_atim.tv_nsec = src->st_atim.tv_nsec;
- dst->st_mtim.tv_nsec = src->st_mtim.tv_nsec;
- dst->st_ctim.tv_nsec = src->st_ctim.tv_nsec;
- #elif HAVE_STRUCT_STAT_ST_MTIMESPEC
- dst->st_atimespec.tv_nsec = src->st_atimespec.tv_nsec;
- dst->st_mtimespec.tv_nsec = src->st_mtimespec.tv_nsec;
- dst->st_ctimespec.tv_nsec = src->st_ctimespec.tv_nsec;
- #elif HAVE_STRUCT_STAT_ST_MTIMENSEC
- dst->st_atimensec = src->st_atimensec;
- dst->st_mtimensec = src->st_mtimensec;
- dst->st_ctimensec = src->st_ctimensec;
- #endif
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- vfs_zero_stat_times (struct stat *s)
- {
- const struct stat empty = { 0 };
- vfs_copy_stat_times (&empty, s);
- }
- /* --------------------------------------------------------------------------------------------- */
|