2011-10-15 14:56:47 +04:00
|
|
|
/*
|
2013-05-26 10:11:06 +04:00
|
|
|
Routines for parsing output from the 'ls' command.
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2024-01-01 09:46:17 +03:00
|
|
|
Copyright (C) 1988-2024
|
2014-02-12 10:33:10 +04:00
|
|
|
Free Software Foundation, Inc.
|
2011-10-15 14:56:47 +04:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
Copyright (C) 1995, 1996 Miguel de Icaza
|
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
Written by:
|
|
|
|
Miguel de Icaza, 1995, 1996
|
|
|
|
Slava Zanko <slavazanko@gmail.com>, 2011
|
|
|
|
|
|
|
|
This file is part of the Midnight Commander.
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
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,
|
2011-08-27 16:16:25 +04:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2011-10-15 14:56:47 +04:00
|
|
|
GNU General Public License for more details.
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* \brief Source: Utilities for VFS modules
|
|
|
|
* \author Miguel de Icaza
|
|
|
|
* \date 1995, 1996
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "lib/global.h"
|
2011-10-23 14:52:39 +04:00
|
|
|
#include "lib/unixcompat.h" /* makedev */
|
2011-08-27 16:16:25 +04:00
|
|
|
#include "lib/widget.h" /* message() */
|
|
|
|
|
|
|
|
#include "utilvfs.h"
|
|
|
|
|
|
|
|
/*** global variables ****************************************************************************/
|
|
|
|
|
|
|
|
/*** file scope macro definitions ****************************************************************/
|
|
|
|
|
2023-09-24 17:32:22 +03:00
|
|
|
/* Parsing code is used by ftpfs, shell and extfs */
|
2011-08-27 16:16:25 +04:00
|
|
|
#define MAXCOLS 30
|
|
|
|
|
|
|
|
/*** file scope type declarations ****************************************************************/
|
|
|
|
|
Update template for .c files.
Add section for forward declarations of local functions. This section is
located before file scope variables because functions can be used in
strucutres (see find.c for example):
/*** forward declarations (file scope functions) *************************************************/
/* button callbacks */
static int start_stop (WButton * button, int action);
static int find_do_view_file (WButton * button, int action);
static int find_do_edit_file (WButton * button, int action);
/*** file scope variables ************************************************************************/
static struct
{
...
bcback_fn callback;
} fbuts[] =
{
...
{ B_STOP, NORMAL_BUTTON, N_("S&uspend"), 0, 0, NULL, start_stop },
...
{ B_VIEW, NORMAL_BUTTON, N_("&View - F3"), 0, 0, NULL, find_do_view_file },
{ B_VIEW, NORMAL_BUTTON, N_("&Edit - F4"), 0, 0, NULL, find_do_edit_file }
};
Signed-off-by: Andrew Borodin <aborodin@vmail.ru>
2023-02-24 09:27:11 +03:00
|
|
|
/*** forward declarations (file scope functions) *************************************************/
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/*** file scope variables ************************************************************************/
|
|
|
|
|
|
|
|
static char *columns[MAXCOLS]; /* Points to the string in column n */
|
|
|
|
static int column_ptr[MAXCOLS]; /* Index from 0 to the starting positions of the columns */
|
2011-08-27 21:22:41 +04:00
|
|
|
static size_t vfs_parce_ls_final_num_spaces = 0;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2011-08-27 16:16:25 +04:00
|
|
|
/*** file scope functions ************************************************************************/
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_num (int idx)
|
|
|
|
{
|
|
|
|
char *column = columns[idx];
|
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
return (column != NULL && isdigit (column[0]));
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2020-02-08 20:28:25 +03:00
|
|
|
/* Return TRUE for MM-DD-YY and MM-DD-YYYY */
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_dos_date (const char *str)
|
|
|
|
{
|
2020-02-08 20:47:08 +03:00
|
|
|
size_t len;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (str == NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
len = strlen (str);
|
|
|
|
if (len != 8 && len != 10)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
if (str[2] != str[5])
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
return (strchr ("\\-/", (int) str[2]) != NULL);
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_week (const char *str, struct tm *tim)
|
|
|
|
{
|
|
|
|
static const char *week = "SunMonTueWedThuFriSat";
|
|
|
|
const char *pos;
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (str == NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
pos = strstr (week, str);
|
2020-02-08 21:10:23 +03:00
|
|
|
if (pos == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (tim != NULL)
|
|
|
|
tim->tm_wday = (pos - week) / 3;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
return TRUE;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Check for possible locale's abbreviated month name (Jan..Dec).
|
|
|
|
* Any 3 bytes long string without digit, control and punctuation characters.
|
|
|
|
* isalpha() is locale specific, so it cannot be used if current
|
|
|
|
* locale is "C" and ftp server use Cyrillic.
|
|
|
|
* NB: It is assumed there are no whitespaces in month.
|
|
|
|
*/
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_localized_month (const char *month)
|
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
int i;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (month == NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
for (i = 0;
|
|
|
|
i < 3 && *month != '\0' && !isdigit ((unsigned char) *month)
|
|
|
|
&& !iscntrl ((unsigned char) *month) && !ispunct ((unsigned char) *month); i++, month++)
|
|
|
|
;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
|
|
|
return (i == 3 && *month == '\0');
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_time (const char *str, struct tm *tim)
|
|
|
|
{
|
|
|
|
const char *p, *p2;
|
|
|
|
|
|
|
|
if (str == NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
p = strchr (str, ':');
|
2020-02-08 21:10:23 +03:00
|
|
|
if (p == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
p2 = strrchr (str, ':');
|
2020-02-08 21:10:23 +03:00
|
|
|
if (p2 == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (p != p2)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
if (sscanf (str, "%2d:%2d:%2d", &tim->tm_hour, &tim->tm_min, &tim->tm_sec) != 3)
|
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
else
|
2020-02-08 21:10:23 +03:00
|
|
|
{
|
|
|
|
if (sscanf (str, "%2d:%2d", &tim->tm_hour, &tim->tm_min) != 2)
|
|
|
|
return FALSE;
|
|
|
|
}
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
return TRUE;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
static gboolean
|
2011-08-27 16:16:25 +04:00
|
|
|
is_year (char *str, struct tm *tim)
|
|
|
|
{
|
|
|
|
long year;
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (str == NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (strchr (str, ':') != NULL)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
if (strlen (str) != 4)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2013-10-10 16:21:26 +04:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2011-08-27 16:16:25 +04:00
|
|
|
if (sscanf (str, "%ld", &year) != 1)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
if (year < 1900 || year > 3000)
|
2020-02-08 20:28:25 +03:00
|
|
|
return FALSE;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
tim->tm_year = (int) (year - 1900);
|
|
|
|
|
2020-02-08 20:28:25 +03:00
|
|
|
return TRUE;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/*** public functions ****************************************************************************/
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
vfs_parse_filetype (const char *s, size_t * ret_skipped, mode_t * ret_type)
|
|
|
|
{
|
|
|
|
mode_t type;
|
|
|
|
|
|
|
|
switch (*s)
|
|
|
|
{
|
|
|
|
case 'd':
|
|
|
|
type = S_IFDIR;
|
|
|
|
break;
|
|
|
|
case 'b':
|
|
|
|
type = S_IFBLK;
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
type = S_IFCHR;
|
|
|
|
break;
|
|
|
|
case 'l':
|
|
|
|
type = S_IFLNK;
|
|
|
|
break;
|
|
|
|
#ifdef S_IFSOCK
|
|
|
|
case 's':
|
|
|
|
type = S_IFSOCK;
|
|
|
|
break;
|
|
|
|
#else
|
|
|
|
case 's':
|
|
|
|
type = S_IFIFO;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#ifdef S_IFDOOR /* Solaris door */
|
|
|
|
case 'D':
|
|
|
|
type = S_IFDOOR;
|
|
|
|
break;
|
|
|
|
#else
|
|
|
|
case 'D':
|
|
|
|
type = S_IFIFO;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 'p':
|
|
|
|
type = S_IFIFO;
|
|
|
|
break;
|
|
|
|
#ifdef S_IFNAM /* Special named files */
|
|
|
|
case 'n':
|
|
|
|
type = S_IFNAM;
|
|
|
|
break;
|
|
|
|
#else
|
|
|
|
case 'n':
|
|
|
|
type = S_IFREG;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 'm': /* Don't know what these are :-) */
|
|
|
|
case '-':
|
|
|
|
case '?':
|
|
|
|
type = S_IFREG;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*ret_type = type;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2013-10-03 14:18:16 +04:00
|
|
|
if (ret_skipped != NULL)
|
|
|
|
*ret_skipped = 1;
|
2011-08-27 16:16:25 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
vfs_parse_fileperms (const char *s, size_t * ret_skipped, mode_t * ret_perms)
|
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
const char *p = s;
|
|
|
|
mode_t perms = 0;
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
perms |= S_IRUSR;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
perms |= S_IWUSR;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
perms |= S_ISUID;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
perms |= S_IXUSR | S_ISUID;
|
|
|
|
break;
|
|
|
|
case 'x':
|
|
|
|
perms |= S_IXUSR;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
perms |= S_IRGRP;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
perms |= S_IWGRP;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
perms |= S_ISGID;
|
|
|
|
break;
|
|
|
|
case 'l':
|
|
|
|
perms |= S_ISGID;
|
|
|
|
break; /* found on Solaris */
|
|
|
|
case 's':
|
|
|
|
perms |= S_IXGRP | S_ISGID;
|
|
|
|
break;
|
|
|
|
case 'x':
|
|
|
|
perms |= S_IXGRP;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
perms |= S_IROTH;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
perms |= S_IWOTH;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
switch (*p++)
|
|
|
|
{
|
|
|
|
case '-':
|
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
perms |= S_ISVTX;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
perms |= S_IXOTH | S_ISVTX;
|
|
|
|
break;
|
|
|
|
case 'x':
|
|
|
|
perms |= S_IXOTH;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
if (*p == '+')
|
2020-02-08 20:47:08 +03:00
|
|
|
/* ACLs on Solaris, HP-UX and others */
|
2011-08-27 16:16:25 +04:00
|
|
|
p++;
|
|
|
|
|
2013-10-03 14:18:16 +04:00
|
|
|
if (ret_skipped != NULL)
|
|
|
|
*ret_skipped = p - s;
|
2011-08-27 16:16:25 +04:00
|
|
|
*ret_perms = perms;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
vfs_parse_filemode (const char *s, size_t * ret_skipped, mode_t * ret_mode)
|
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
const char *p = s;
|
2011-08-27 16:16:25 +04:00
|
|
|
mode_t type, perms;
|
|
|
|
size_t skipped;
|
|
|
|
|
|
|
|
if (!vfs_parse_filetype (p, &skipped, &type))
|
|
|
|
return FALSE;
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
p += skipped;
|
2011-08-27 16:16:25 +04:00
|
|
|
if (!vfs_parse_fileperms (p, &skipped, &perms))
|
|
|
|
return FALSE;
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
p += skipped;
|
2011-08-27 16:16:25 +04:00
|
|
|
*ret_skipped = p - s;
|
|
|
|
*ret_mode = type | perms;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
vfs_parse_raw_filemode (const char *s, size_t * ret_skipped, mode_t * ret_mode)
|
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
const char *p = s;
|
2011-08-27 16:16:25 +04:00
|
|
|
mode_t remote_type = 0, local_type, perms = 0;
|
|
|
|
|
|
|
|
/* isoctal */
|
2020-02-08 21:10:23 +03:00
|
|
|
for (; *p >= '0' && *p <= '7'; p++)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
|
|
|
perms *= 010;
|
|
|
|
perms += (*p - '0');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*p++ != ' ')
|
|
|
|
return FALSE;
|
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
for (; *p >= '0' && *p <= '7'; p++)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
|
|
|
remote_type *= 010;
|
|
|
|
remote_type += (*p - '0');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*p++ != ' ')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* generated with:
|
|
|
|
$ perl -e 'use Fcntl ":mode";
|
|
|
|
my @modes = (S_IFDIR, S_IFBLK, S_IFCHR, S_IFLNK, S_IFREG);
|
|
|
|
foreach $t (@modes) { printf ("%o\n", $t); };'
|
|
|
|
TODO: S_IFDOOR, S_IFIFO, S_IFSOCK (if supported by os)
|
|
|
|
(see vfs_parse_filetype)
|
|
|
|
*/
|
|
|
|
|
|
|
|
switch (remote_type)
|
|
|
|
{
|
|
|
|
case 020000:
|
|
|
|
local_type = S_IFCHR;
|
|
|
|
break;
|
|
|
|
case 040000:
|
|
|
|
local_type = S_IFDIR;
|
|
|
|
break;
|
|
|
|
case 060000:
|
|
|
|
local_type = S_IFBLK;
|
|
|
|
break;
|
|
|
|
case 0120000:
|
|
|
|
local_type = S_IFLNK;
|
|
|
|
break;
|
|
|
|
case 0100000:
|
|
|
|
default: /* don't know what is it */
|
|
|
|
local_type = S_IFREG;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
*ret_skipped = p - s;
|
|
|
|
*ret_mode = local_type | perms;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-10-03 14:18:16 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
vfs_parse_month (const char *str, struct tm * tim)
|
|
|
|
{
|
|
|
|
static const char *month = "JanFebMarAprMayJunJulAugSepOctNovDec";
|
|
|
|
const char *pos;
|
|
|
|
|
|
|
|
if (str == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
pos = strstr (month, str);
|
|
|
|
if (pos == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (tim != NULL)
|
|
|
|
tim->tm_mon = (pos - month) / 3;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/** This function parses from idx in the columns[] array */
|
|
|
|
|
|
|
|
int
|
|
|
|
vfs_parse_filedate (int idx, time_t * t)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
struct tm tim;
|
|
|
|
int d[3];
|
2020-02-08 20:28:25 +03:00
|
|
|
gboolean got_year = FALSE;
|
|
|
|
gboolean l10n = FALSE; /* Locale's abbreviated month name */
|
2011-08-27 16:16:25 +04:00
|
|
|
time_t current_time;
|
|
|
|
struct tm *local_time;
|
|
|
|
|
|
|
|
/* Let's setup default time values */
|
|
|
|
current_time = time (NULL);
|
|
|
|
local_time = localtime (¤t_time);
|
|
|
|
tim.tm_mday = local_time->tm_mday;
|
|
|
|
tim.tm_mon = local_time->tm_mon;
|
|
|
|
tim.tm_year = local_time->tm_year;
|
|
|
|
|
|
|
|
tim.tm_hour = 0;
|
|
|
|
tim.tm_min = 0;
|
|
|
|
tim.tm_sec = 0;
|
|
|
|
tim.tm_isdst = -1; /* Let mktime() try to guess correct dst offset */
|
|
|
|
|
|
|
|
p = columns[idx++];
|
|
|
|
|
|
|
|
/* We eat weekday name in case of extfs */
|
|
|
|
if (is_week (p, &tim))
|
|
|
|
p = columns[idx++];
|
|
|
|
|
2016-09-27 22:40:15 +03:00
|
|
|
/*
|
|
|
|
ALLOWED DATE FORMATS
|
|
|
|
|
|
|
|
We expect 3 fields max or we'll see oddities with certain file names.
|
|
|
|
|
|
|
|
Formats that contain either year or time (the default 'ls' formats):
|
|
|
|
|
|
|
|
* Mon DD hh:mm[:ss]
|
|
|
|
* Mon DD YYYY
|
|
|
|
|
|
|
|
Formats that contain both year and time, to make it easier to write
|
|
|
|
extfs scripts:
|
|
|
|
|
|
|
|
* MM-DD-YYYY hh:mm[:ss]
|
|
|
|
* MM-DD-YY hh:mm[:ss]
|
|
|
|
|
|
|
|
('/' and '\' can be used instead of '-'.)
|
|
|
|
|
|
|
|
where Mon is Jan-Dec, DD, MM, YY two digit day, month, year,
|
|
|
|
YYYY four digit year, hh, mm, ss two digit hour, minute or second.
|
|
|
|
|
|
|
|
(As for the "3 fields max" restriction: this prevents, for example, a
|
|
|
|
file name "13:48" from being considered part of a "Sep 19 2016" date
|
|
|
|
string preceding it.)
|
|
|
|
*/
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/* Month name */
|
2013-10-03 14:18:16 +04:00
|
|
|
if (vfs_parse_month (p, &tim))
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
|
|
|
/* And we expect, it followed by day number */
|
2020-02-08 21:10:23 +03:00
|
|
|
if (!is_num (idx))
|
2011-08-27 16:16:25 +04:00
|
|
|
return 0; /* No day */
|
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
tim.tm_mday = (int) atol (columns[idx++]);
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
else if (is_dos_date (p))
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2016-09-27 22:40:15 +03:00
|
|
|
/* Case with MM-DD-YY or MM-DD-YYYY */
|
2020-02-08 20:47:08 +03:00
|
|
|
p[2] = p[5] = '-';
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2020-02-08 21:10:23 +03:00
|
|
|
if (sscanf (p, "%2d-%2d-%d", &d[0], &d[1], &d[2]) != 3)
|
2020-02-08 20:47:08 +03:00
|
|
|
return 0; /* sscanf failed */
|
2020-02-08 21:10:23 +03:00
|
|
|
|
|
|
|
/* Months are zero based */
|
|
|
|
if (d[0] > 0)
|
|
|
|
d[0]--;
|
|
|
|
|
|
|
|
if (d[2] > 1900)
|
|
|
|
d[2] -= 1900;
|
|
|
|
else if (d[2] < 70)
|
|
|
|
/* Y2K madness */
|
|
|
|
d[2] += 100;
|
|
|
|
|
|
|
|
tim.tm_mon = d[0];
|
|
|
|
tim.tm_mday = d[1];
|
|
|
|
tim.tm_year = d[2];
|
|
|
|
got_year = TRUE;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
else if (is_localized_month (p) && is_num (idx++))
|
|
|
|
/* Locale's abbreviated month name followed by day number */
|
|
|
|
l10n = TRUE;
|
|
|
|
else
|
|
|
|
return 0; /* unsupported format */
|
2011-08-27 16:16:25 +04:00
|
|
|
|
|
|
|
/* Here we expect to find time or year */
|
2020-02-08 21:10:23 +03:00
|
|
|
if (!is_num (idx)
|
|
|
|
|| !(is_time (columns[idx], &tim) || (got_year = is_year (columns[idx], &tim))))
|
2011-08-27 16:16:25 +04:00
|
|
|
return 0; /* Neither time nor date */
|
|
|
|
|
2020-02-08 21:10:23 +03:00
|
|
|
idx++;
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/*
|
|
|
|
* If the date is less than 6 months in the past, it is shown without year
|
|
|
|
* other dates in the past or future are shown with year but without time
|
|
|
|
* This does not check for years before 1900 ... I don't know, how
|
|
|
|
* to represent them at all
|
|
|
|
*/
|
2020-02-08 21:10:23 +03:00
|
|
|
if (!got_year && local_time->tm_mon < 6 && local_time->tm_mon < tim.tm_mon
|
|
|
|
&& tim.tm_mon - local_time->tm_mon >= 6)
|
2011-08-27 16:16:25 +04:00
|
|
|
tim.tm_year--;
|
|
|
|
|
|
|
|
*t = mktime (&tim);
|
|
|
|
if (l10n || (*t < 0))
|
|
|
|
*t = 0;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
int
|
|
|
|
vfs_split_text (char *p)
|
|
|
|
{
|
|
|
|
char *original = p;
|
|
|
|
int numcols;
|
|
|
|
|
|
|
|
memset (columns, 0, sizeof (columns));
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
for (numcols = 0; *p != '\0' && numcols < MAXCOLS; numcols++)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2020-02-08 21:10:23 +03:00
|
|
|
for (; *p == ' ' || *p == '\r' || *p == '\n'; p++)
|
2020-02-08 20:47:08 +03:00
|
|
|
*p = '\0';
|
2020-02-08 21:10:23 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
columns[numcols] = p;
|
|
|
|
column_ptr[numcols] = p - original;
|
2020-02-08 21:10:23 +03:00
|
|
|
|
|
|
|
for (; *p != '\0' && *p != ' ' && *p != '\r' && *p != '\n'; p++)
|
|
|
|
;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
return numcols;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2011-08-27 21:22:41 +04:00
|
|
|
void
|
|
|
|
vfs_parse_ls_lga_init (void)
|
|
|
|
{
|
2011-10-20 15:28:39 +04:00
|
|
|
vfs_parce_ls_final_num_spaces = 1;
|
2011-08-27 21:22:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
size_t
|
|
|
|
vfs_parse_ls_lga_get_final_spaces (void)
|
|
|
|
{
|
|
|
|
return vfs_parce_ls_final_num_spaces;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
gboolean
|
|
|
|
vfs_parse_ls_lga (const char *p, struct stat * s, char **filename, char **linkname,
|
2011-08-27 21:22:41 +04:00
|
|
|
size_t * num_spaces)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
|
|
|
int idx, idx2, num_cols;
|
|
|
|
int i;
|
|
|
|
char *p_copy = NULL;
|
|
|
|
char *t = NULL;
|
|
|
|
const char *line = p;
|
|
|
|
size_t skipped;
|
|
|
|
|
|
|
|
if (strncmp (p, "total", 5) == 0)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!vfs_parse_filetype (p, &skipped, &s->st_mode))
|
|
|
|
goto error;
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
p += skipped;
|
2011-08-27 16:16:25 +04:00
|
|
|
if (*p == ' ') /* Notwell 4 */
|
|
|
|
p++;
|
|
|
|
if (*p == '[')
|
|
|
|
{
|
|
|
|
if (strlen (p) <= 8 || p[8] != ']')
|
|
|
|
goto error;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/* Should parse here the Notwell permissions :) */
|
|
|
|
if (S_ISDIR (s->st_mode))
|
|
|
|
s->st_mode |= (S_IRUSR | S_IRGRP | S_IROTH | S_IWUSR | S_IXUSR | S_IXGRP | S_IXOTH);
|
|
|
|
else
|
|
|
|
s->st_mode |= (S_IRUSR | S_IRGRP | S_IROTH | S_IWUSR);
|
|
|
|
p += 9;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
size_t lc_skipped;
|
|
|
|
mode_t perms;
|
|
|
|
|
|
|
|
if (!vfs_parse_fileperms (p, &lc_skipped, &perms))
|
|
|
|
goto error;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
p += lc_skipped;
|
|
|
|
s->st_mode |= perms;
|
|
|
|
}
|
|
|
|
|
|
|
|
p_copy = g_strdup (p);
|
|
|
|
num_cols = vfs_split_text (p_copy);
|
|
|
|
|
|
|
|
s->st_nlink = atol (columns[0]);
|
|
|
|
if (s->st_nlink <= 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (!is_num (1))
|
|
|
|
s->st_uid = vfs_finduid (columns[1]);
|
|
|
|
else
|
|
|
|
s->st_uid = (uid_t) atol (columns[1]);
|
|
|
|
|
|
|
|
/* Mhm, the ls -lg did not produce a group field */
|
|
|
|
for (idx = 3; idx <= 5; idx++)
|
2013-10-03 14:18:16 +04:00
|
|
|
if (vfs_parse_month (columns[idx], NULL) || is_week (columns[idx], NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
|| is_dos_date (columns[idx]) || is_localized_month (columns[idx]))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (idx == 6 || (idx == 5 && !S_ISCHR (s->st_mode) && !S_ISBLK (s->st_mode)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
/* We don't have gid */
|
|
|
|
if (idx == 3 || (idx == 4 && (S_ISCHR (s->st_mode) || S_ISBLK (s->st_mode))))
|
|
|
|
idx2 = 2;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* We have gid field */
|
|
|
|
if (is_num (2))
|
|
|
|
s->st_gid = (gid_t) atol (columns[2]);
|
|
|
|
else
|
|
|
|
s->st_gid = vfs_findgid (columns[2]);
|
|
|
|
idx2 = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is device */
|
|
|
|
if (S_ISCHR (s->st_mode) || S_ISBLK (s->st_mode))
|
|
|
|
{
|
|
|
|
int maj, min;
|
|
|
|
|
|
|
|
/* Corner case: there is no whitespace(s) between maj & min */
|
|
|
|
if (!is_num (idx2) && idx2 == 2)
|
|
|
|
{
|
2013-10-10 16:21:26 +04:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2011-08-27 16:16:25 +04:00
|
|
|
if (!is_num (++idx2) || sscanf (columns[idx2], " %d,%d", &maj, &min) != 2)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-10-10 16:21:26 +04:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2011-08-27 16:16:25 +04:00
|
|
|
if (!is_num (idx2) || sscanf (columns[idx2], " %d,", &maj) != 1)
|
|
|
|
goto error;
|
|
|
|
|
2013-10-10 16:21:26 +04:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2011-08-27 16:16:25 +04:00
|
|
|
if (!is_num (++idx2) || sscanf (columns[idx2], " %d", &min) != 1)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
#ifdef HAVE_STRUCT_STAT_ST_RDEV
|
|
|
|
s->st_rdev = makedev (maj, min);
|
|
|
|
#endif
|
|
|
|
s->st_size = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Common file size */
|
|
|
|
if (!is_num (idx2))
|
|
|
|
goto error;
|
|
|
|
|
2012-12-07 14:32:46 +04:00
|
|
|
s->st_size = (off_t) g_ascii_strtoll (columns[idx2], NULL, 10);
|
2011-08-27 16:16:25 +04:00
|
|
|
#ifdef HAVE_STRUCT_STAT_ST_RDEV
|
|
|
|
s->st_rdev = 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = vfs_parse_filedate (idx, &s->st_mtime);
|
2020-02-08 20:47:08 +03:00
|
|
|
if (idx == 0)
|
2011-08-27 16:16:25 +04:00
|
|
|
goto error;
|
2020-02-08 20:47:08 +03:00
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/* Use resulting time value */
|
|
|
|
s->st_atime = s->st_ctime = s->st_mtime;
|
2017-05-06 21:57:59 +03:00
|
|
|
#ifdef HAVE_STRUCT_STAT_ST_MTIM
|
|
|
|
s->st_atim.tv_nsec = s->st_mtim.tv_nsec = s->st_ctim.tv_nsec = 0;
|
|
|
|
#endif
|
|
|
|
|
2011-08-27 16:16:25 +04:00
|
|
|
/* s->st_dev and s->st_ino must be initialized by vfs_s_new_inode () */
|
|
|
|
#ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
|
|
|
|
s->st_blksize = 512;
|
|
|
|
#endif
|
2017-01-04 11:27:19 +03:00
|
|
|
vfs_adjust_stat (s);
|
2011-08-27 16:16:25 +04:00
|
|
|
|
2011-08-27 21:22:41 +04:00
|
|
|
if (num_spaces != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2011-08-27 21:22:41 +04:00
|
|
|
*num_spaces = column_ptr[idx] - column_ptr[idx - 1] - strlen (columns[idx - 1]);
|
2013-07-13 21:02:34 +04:00
|
|
|
if (DIR_IS_DOTDOT (columns[idx]))
|
2011-08-27 21:22:41 +04:00
|
|
|
vfs_parce_ls_final_num_spaces = *num_spaces;
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = idx + 1, idx2 = 0; i < num_cols; i++)
|
|
|
|
if (strcmp (columns[i], "->") == 0)
|
|
|
|
{
|
|
|
|
idx2 = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((S_ISLNK (s->st_mode) || (num_cols == idx + 3 && s->st_nlink > 1))) /* Maybe a hardlink? (in extfs) */
|
2020-02-08 20:47:08 +03:00
|
|
|
&& idx2 != 0)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2020-02-08 20:47:08 +03:00
|
|
|
if (filename != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
*filename = g_strndup (p + column_ptr[idx], column_ptr[idx2] - column_ptr[idx] - 1);
|
2020-02-08 20:47:08 +03:00
|
|
|
|
|
|
|
if (linkname != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
|
|
|
t = g_strdup (p + column_ptr[idx2 + 1]);
|
|
|
|
*linkname = t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Extract the filename from the string copy, not from the columns
|
|
|
|
* this way we have a chance of entering hidden directories like ". ."
|
|
|
|
*/
|
2020-02-08 20:47:08 +03:00
|
|
|
if (filename != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2020-02-08 20:47:08 +03:00
|
|
|
/* filename = g_strdup (columns [idx++]); */
|
2011-08-27 16:16:25 +04:00
|
|
|
t = g_strdup (p + column_ptr[idx]);
|
|
|
|
*filename = t;
|
|
|
|
}
|
2020-02-08 20:47:08 +03:00
|
|
|
|
|
|
|
if (linkname != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
*linkname = NULL;
|
|
|
|
}
|
|
|
|
|
2020-02-08 20:47:08 +03:00
|
|
|
if (t != NULL)
|
2011-08-27 16:16:25 +04:00
|
|
|
{
|
2020-02-08 20:47:08 +03:00
|
|
|
size_t p2;
|
|
|
|
|
|
|
|
p2 = strlen (t);
|
|
|
|
if (--p2 > 0 && (t[p2] == '\r' || t[p2] == '\n'))
|
|
|
|
t[p2] = '\0';
|
|
|
|
if (--p2 > 0 && (t[p2] == '\r' || t[p2] == '\n'))
|
|
|
|
t[p2] = '\0';
|
2011-08-27 16:16:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
g_free (p_copy);
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
error:
|
|
|
|
{
|
|
|
|
static int errorcount = 0;
|
|
|
|
|
|
|
|
if (++errorcount < 5)
|
2020-02-08 20:47:08 +03:00
|
|
|
message (D_ERROR, _("Cannot parse:"), "%s",
|
|
|
|
(p_copy != NULL && *p_copy != '\0') ? p_copy : line);
|
2011-08-27 16:16:25 +04:00
|
|
|
else if (errorcount == 5)
|
|
|
|
message (D_ERROR, MSG_ERROR, _("More parsing errors will be ignored."));
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (p_copy);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|