| // © 2016 and later: Unicode, Inc. and others. |
| // License & terms of use: http://www.unicode.org/copyright.html |
| /****************************************************************************** |
| * Copyright (C) 2009-2013, International Business Machines |
| * Corporation and others. All Rights Reserved. |
| ******************************************************************************* |
| */ |
| |
| #include "unicode/platform.h" |
| #if U_PLATFORM == U_PF_MINGW |
| // *cough* - for struct stat |
| #ifdef __STRICT_ANSI__ |
| #undef __STRICT_ANSI__ |
| #endif |
| #endif |
| |
| #include "filetools.h" |
| #include "filestrm.h" |
| #include "charstr.h" |
| #include "cstring.h" |
| #include "unicode/putil.h" |
| #include "putilimp.h" |
| |
| #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; |
| |
| #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. |
| * Also can be given just one file to check against. Default value for isDir is FALSE. |
| */ |
| U_CAPI UBool U_EXPORT2 |
| isFileModTimeLater(const char *filePath, const char *checkAgainst, UBool isDir) { |
| UBool isLatest = TRUE; |
| |
| if (filePath == NULL || checkAgainst == NULL) { |
| return FALSE; |
| } |
| |
| if (isDir == TRUE) { |
| #if U_HAVE_DIRENT_H |
| 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) { |
| UErrorCode status = U_ZERO_ERROR; |
| icu::CharString newpath(checkAgainst, -1, status); |
| newpath.append(U_FILE_SEP_STRING, -1, status); |
| newpath.append(dirEntry->d_name, -1, status); |
| if (U_FAILURE(status)) { |
| fprintf(stderr, "%s:%d: %s\n", __FILE__, __LINE__, u_errorName(status)); |
| return FALSE; |
| } |
| |
| if ((subDirp = opendir(newpath.data())) != NULL) { |
| /* If this new path is a directory, make a recursive call with the newpath. */ |
| closedir(subDirp); |
| isLatest = isFileModTimeLater(filePath, newpath.data(), isDir); |
| if (!isLatest) { |
| break; |
| } |
| } else { |
| int32_t latest = whichFileModTimeIsLater(filePath, newpath.data()); |
| if (latest < 0 || latest == 2) { |
| isLatest = FALSE; |
| break; |
| } |
| } |
| |
| } |
| } |
| closedir(pDir); |
| } else { |
| fprintf(stderr, "Unable to open directory: %s\n", checkAgainst); |
| return FALSE; |
| } |
| #endif |
| } else { |
| if (T_FileStream_file_exists(checkAgainst)) { |
| int32_t latest = whichFileModTimeIsLater(filePath, checkAgainst); |
| if (latest < 0 || latest == 2) { |
| isLatest = FALSE; |
| } |
| } else { |
| isLatest = FALSE; |
| } |
| } |
| |
| 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; |
| struct stat stbuf1, stbuf2; |
| |
| if (stat(file1, &stbuf1) == 0 && stat(file2, &stbuf2) == 0) { |
| time_t modtime1, modtime2; |
| double diff; |
| |
| modtime1 = stbuf1.st_mtime; |
| modtime2 = stbuf2.st_mtime; |
| |
| 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; |
| } |
| |
| /* 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 = static_cast<int32_t>(uprv_strlen(filePath)); i < length; i++) { |
| filePath[i] = (filePath[i] == oldFileSepChar ) ? newFileSepChar : filePath[i]; |
| } |
| } |