2009-01-09 22:30:44 +00:00
|
|
|
/******************************************************************************
|
2013-09-04 06:59:04 +00:00
|
|
|
* Copyright (C) 2009-2013, International Business Machines
|
2009-01-09 22:30:44 +00:00
|
|
|
* Corporation and others. All Rights Reserved.
|
|
|
|
*******************************************************************************
|
|
|
|
*/
|
|
|
|
|
2013-09-04 06:59:04 +00:00
|
|
|
#if U_PLATFORM == U_PF_MINGW
|
|
|
|
// *cough* - for struct stat
|
|
|
|
#ifdef __STRICT_ANSI__
|
|
|
|
#undef __STRICT_ANSI__
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2009-01-12 23:02:36 +00:00
|
|
|
#include "filetools.h"
|
2009-01-12 20:52:01 +00:00
|
|
|
#include "filestrm.h"
|
2009-01-09 22:30:44 +00:00
|
|
|
#include "cstring.h"
|
|
|
|
#include "unicode/putil.h"
|
2011-07-27 20:55:09 +00:00
|
|
|
#include "putilimp.h"
|
2009-01-09 22:30:44 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#if U_HAVE_DIRENT_H
|
|
|
|
#include <dirent.h>
|
|
|
|
typedef struct dirent DIRENT;
|
|
|
|
|
2009-01-09 22:58:05 +00:00
|
|
|
#define MAX_PATH_SIZE 4096 /* Set the limit for the size of the path. */
|
2009-01-09 22:30:44 +00:00
|
|
|
|
|
|
|
#define SKIP1 "."
|
|
|
|
#define SKIP2 ".."
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int32_t whichFileModTimeIsLater(const char *file1, const char *file2);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Goes through the given directory recursive to compare each file's modification time with that of the file given.
|
2009-01-12 20:52:01 +00:00
|
|
|
* Also can be given just one file to check against. Default value for isDir is FALSE.
|
2009-01-09 22:30:44 +00:00
|
|
|
*/
|
|
|
|
U_CAPI UBool U_EXPORT2
|
2009-01-12 20:52:01 +00:00
|
|
|
isFileModTimeLater(const char *filePath, const char *checkAgainst, UBool isDir) {
|
2009-01-09 22:30:44 +00:00
|
|
|
UBool isLatest = TRUE;
|
|
|
|
|
2009-01-12 20:52:01 +00:00
|
|
|
if (filePath == NULL || checkAgainst == NULL) {
|
2009-01-09 22:30:44 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2009-01-12 20:52:01 +00:00
|
|
|
if (isDir == TRUE) {
|
2009-01-09 22:30:44 +00:00
|
|
|
#if U_HAVE_DIRENT_H
|
2009-01-12 20:52:01 +00:00
|
|
|
DIR *pDir = NULL;
|
|
|
|
if ((pDir= opendir(checkAgainst)) != NULL) {
|
|
|
|
DIR *subDirp = NULL;
|
|
|
|
DIRENT *dirEntry = NULL;
|
|
|
|
|
|
|
|
while ((dirEntry = readdir(pDir)) != NULL) {
|
|
|
|
if (uprv_strcmp(dirEntry->d_name, SKIP1) != 0 && uprv_strcmp(dirEntry->d_name, SKIP2) != 0) {
|
|
|
|
char newpath[MAX_PATH_SIZE] = "";
|
|
|
|
uprv_strcpy(newpath, checkAgainst);
|
|
|
|
uprv_strcat(newpath, U_FILE_SEP_STRING);
|
|
|
|
uprv_strcat(newpath, dirEntry->d_name);
|
|
|
|
|
|
|
|
if ((subDirp = opendir(newpath)) != NULL) {
|
|
|
|
/* If this new path is a directory, make a recursive call with the newpath. */
|
|
|
|
closedir(subDirp);
|
|
|
|
isLatest = isFileModTimeLater(filePath, newpath, isDir);
|
|
|
|
if (!isLatest) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int32_t latest = whichFileModTimeIsLater(filePath, newpath);
|
|
|
|
if (latest < 0 || latest == 2) {
|
|
|
|
isLatest = FALSE;
|
|
|
|
break;
|
|
|
|
}
|
2009-01-09 22:30:44 +00:00
|
|
|
}
|
|
|
|
|
2009-01-12 20:52:01 +00:00
|
|
|
}
|
2009-01-09 22:30:44 +00:00
|
|
|
}
|
2009-01-12 20:52:01 +00:00
|
|
|
closedir(pDir);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Unable to open directory: %s\n", checkAgainst);
|
|
|
|
return FALSE;
|
2009-01-09 22:30:44 +00:00
|
|
|
}
|
2009-01-12 20:52:01 +00:00
|
|
|
#endif
|
2009-01-09 22:30:44 +00:00
|
|
|
} else {
|
2009-01-12 20:52:01 +00:00
|
|
|
if (T_FileStream_file_exists(checkAgainst)) {
|
|
|
|
int32_t latest = whichFileModTimeIsLater(filePath, checkAgainst);
|
|
|
|
if (latest < 0 || latest == 2) {
|
|
|
|
isLatest = FALSE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
isLatest = FALSE;
|
|
|
|
}
|
2009-01-09 22:30:44 +00:00
|
|
|
}
|
2009-01-12 20:52:01 +00:00
|
|
|
|
2009-01-09 22:30:44 +00:00
|
|
|
return isLatest;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compares the mod time of both files returning a number indicating which one is later. -1 if error ocurs. */
|
|
|
|
static int32_t whichFileModTimeIsLater(const char *file1, const char *file2) {
|
|
|
|
int32_t result = 0;
|
2009-01-12 20:52:01 +00:00
|
|
|
struct stat stbuf1, stbuf2;
|
2009-01-09 22:30:44 +00:00
|
|
|
|
|
|
|
if (stat(file1, &stbuf1) == 0 && stat(file2, &stbuf2) == 0) {
|
|
|
|
time_t modtime1, modtime2;
|
|
|
|
double diff;
|
|
|
|
|
2009-01-09 23:08:04 +00:00
|
|
|
modtime1 = stbuf1.st_mtime;
|
|
|
|
modtime2 = stbuf2.st_mtime;
|
2009-01-09 22:30:44 +00:00
|
|
|
|
|
|
|
diff = difftime(modtime1, modtime2);
|
|
|
|
if (diff < 0.0) {
|
|
|
|
result = 2;
|
|
|
|
} else if (diff > 0.0) {
|
|
|
|
result = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Unable to get stats from file: %s or %s\n", file1, file2);
|
|
|
|
result = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2009-01-12 23:02:36 +00:00
|
|
|
|
|
|
|
/* Swap the file separater character given with the new one in the file path. */
|
|
|
|
U_CAPI void U_EXPORT2
|
|
|
|
swapFileSepChar(char *filePath, const char oldFileSepChar, const char newFileSepChar) {
|
|
|
|
for (int32_t i = 0, length = uprv_strlen(filePath); i < length; i++) {
|
|
|
|
filePath[i] = (filePath[i] == oldFileSepChar ) ? newFileSepChar : filePath[i];
|
|
|
|
}
|
|
|
|
}
|