You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
148 lines
3.1 KiB
148 lines
3.1 KiB
/* |
|
* This program can either change modification time of the given |
|
* file(s) or just print it. The program does not change atime or |
|
* ctime (their values are explicitly preserved). |
|
* |
|
* The mtime can be changed to an absolute value: |
|
* |
|
* test-tool chmtime =<seconds> file... |
|
* |
|
* Relative to the current time as returned by time(3): |
|
* |
|
* test-tool chmtime =+<seconds> (or =-<seconds>) file... |
|
* |
|
* Or relative to the current mtime of the file: |
|
* |
|
* test-tool chmtime <seconds> file... |
|
* test-tool chmtime +<seconds> (or -<seconds>) file... |
|
* |
|
* Examples: |
|
* |
|
* To print the mtime and the file name use --verbose and set |
|
* the file mtime offset to 0: |
|
* |
|
* test-tool chmtime -v +0 file |
|
* |
|
* To print only the mtime use --get: |
|
* |
|
* test-tool chmtime --get file |
|
* |
|
* To set the mtime to current time: |
|
* |
|
* test-tool chmtime =+0 file |
|
* |
|
* To set the file mtime offset to +1 and print the new value: |
|
* |
|
* test-tool chmtime --get +1 file |
|
* |
|
*/ |
|
#include "test-tool.h" |
|
#include "git-compat-util.h" |
|
#include <utime.h> |
|
|
|
static const char usage_str[] = |
|
"(-v|--verbose|-g|--get) (+|=|=+|=-|-)<seconds> <file>..."; |
|
|
|
static int timespec_arg(const char *arg, long int *set_time, int *set_eq) |
|
{ |
|
char *test; |
|
const char *timespec = arg; |
|
*set_eq = (*timespec == '=') ? 1 : 0; |
|
if (*set_eq) { |
|
timespec++; |
|
if (*timespec == '+') { |
|
*set_eq = 2; /* relative "in the future" */ |
|
timespec++; |
|
} |
|
} |
|
*set_time = strtol(timespec, &test, 10); |
|
if (*test) { |
|
return 0; |
|
} |
|
if ((*set_eq && *set_time < 0) || *set_eq == 2) { |
|
time_t now = time(NULL); |
|
*set_time += now; |
|
} |
|
return 1; |
|
} |
|
|
|
int cmd__chmtime(int argc, const char **argv) |
|
{ |
|
static int verbose; |
|
static int get; |
|
|
|
int i = 1; |
|
/* no mtime change by default */ |
|
int set_eq = 0; |
|
long int set_time = 0; |
|
|
|
if (argc < 3) |
|
goto usage; |
|
|
|
if (strcmp(argv[i], "--get") == 0 || strcmp(argv[i], "-g") == 0) { |
|
get = 1; |
|
++i; |
|
} else if (strcmp(argv[i], "--verbose") == 0 || strcmp(argv[i], "-v") == 0) { |
|
verbose = 1; |
|
++i; |
|
} |
|
|
|
if (i == argc) { |
|
goto usage; |
|
} |
|
|
|
if (timespec_arg(argv[i], &set_time, &set_eq)) { |
|
++i; |
|
} else { |
|
if (get == 0) { |
|
fprintf(stderr, "Not a base-10 integer: %s\n", argv[i] + 1); |
|
goto usage; |
|
} |
|
} |
|
|
|
if (i == argc) |
|
goto usage; |
|
|
|
for (; i < argc; i++) { |
|
struct stat sb; |
|
struct utimbuf utb; |
|
uintmax_t mtime; |
|
|
|
if (stat(argv[i], &sb) < 0) { |
|
fprintf(stderr, "Failed to stat %s: %s\n", |
|
argv[i], strerror(errno)); |
|
return 1; |
|
} |
|
|
|
#ifdef GIT_WINDOWS_NATIVE |
|
if (!(sb.st_mode & S_IWUSR) && |
|
chmod(argv[i], sb.st_mode | S_IWUSR)) { |
|
fprintf(stderr, "Could not make user-writable %s: %s", |
|
argv[i], strerror(errno)); |
|
return 1; |
|
} |
|
#endif |
|
|
|
utb.actime = sb.st_atime; |
|
utb.modtime = set_eq ? set_time : sb.st_mtime + set_time; |
|
|
|
mtime = utb.modtime < 0 ? 0: utb.modtime; |
|
if (get) { |
|
printf("%"PRIuMAX"\n", mtime); |
|
} else if (verbose) { |
|
printf("%"PRIuMAX"\t%s\n", mtime, argv[i]); |
|
} |
|
|
|
if (utb.modtime != sb.st_mtime && utime(argv[i], &utb) < 0) { |
|
fprintf(stderr, "Failed to modify time on %s: %s\n", |
|
argv[i], strerror(errno)); |
|
return 1; |
|
} |
|
} |
|
|
|
return 0; |
|
|
|
usage: |
|
fprintf(stderr, "usage: %s %s\n", argv[0], usage_str); |
|
return 1; |
|
}
|
|
|