/* Utilities for VFS modules. Copyright (C) 1988-2025 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 . */ /** * \file * \brief Source: Utilities for VFS modules * \author Miguel de Icaza * \date 1995, 1996 */ #include #include #include #include #include #include #include #if !defined (HAVE_UTIMENSAT) && defined (HAVE_UTIME_H) #include #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); } /* --------------------------------------------------------------------------------------------- */