generated from zhangwei/Train_Identify
			
		
			
	
	
		
			526 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
		
		
			
		
	
	
			526 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
|  | /*
 | ||
|  |  * Copyright (c) 2020.Huawei Technologies Co., Ltd. All rights reserved. | ||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||
|  |  * you may not use this file except in compliance with the License. | ||
|  |  * You may obtain a copy of the License at | ||
|  |  * | ||
|  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||
|  |  * | ||
|  |  * Unless required by applicable law or agreed to in writing, software | ||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
|  |  * See the License for the specific language governing permissions and | ||
|  |  * limitations under the License. | ||
|  |  */ | ||
|  | 
 | ||
|  | #include "FileManager.h"
 | ||
|  | #include <ctime>
 | ||
|  | #include <string>
 | ||
|  | 
 | ||
|  | namespace { | ||
|  |     const int BUFFER_SIZE = 2048; | ||
|  | #ifndef _WIN32
 | ||
|  |     const mode_t DEFAULT_FILE_PERMISSION = 0077; | ||
|  | #endif
 | ||
|  | } | ||
|  | 
 | ||
|  | #ifndef _WIN32
 | ||
|  | mode_t SetFileDefaultUmask() | ||
|  | { | ||
|  |     return umask(DEFAULT_FILE_PERMISSION); | ||
|  | } | ||
|  | 
 | ||
|  | mode_t SetFileUmask(mode_t newUmask) | ||
|  | { | ||
|  |     return umask(newUmask); | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Check whether the file exists. | ||
|  |  * | ||
|  |  * @param filePath the file path we want to check | ||
|  |  * @return APP_ERR_OK if file exists, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR ExistFile(const std::string &filePath) | ||
|  | { | ||
|  |     std::string resolvedPath; | ||
|  |     APP_ERROR ret = GetRealPath(filePath, resolvedPath); | ||
|  |     if (ret != APP_ERR_OK) { | ||
|  |         return ret; | ||
|  |     } | ||
|  | #ifndef _WIN32
 | ||
|  |     struct stat fileStat = {0}; | ||
|  |     if (stat(resolvedPath.c_str(), &fileStat) == 0 && S_ISREG(fileStat.st_mode)) { | ||
|  | #else
 | ||
|  |     DWORD fileStat = GetFileAttributes((LPCSTR)resolvedPath.c_str()); | ||
|  |     if ((fileStat == FILE_ATTRIBUTE_ARCHIVE) || (fileStat == FILE_ATTRIBUTE_NORMAL) || | ||
|  |         (fileStat == FILE_ATTRIBUTE_DIRECTORY)) { | ||
|  | #endif
 | ||
|  |         return APP_ERR_OK; | ||
|  |     } | ||
|  |     return APP_ERR_COMM_FAILURE; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Split the input path string with delimiter | ||
|  |  * | ||
|  |  * @param str path string | ||
|  |  * @param delimiters a set of delimiter | ||
|  |  * @return the vector of splitted path | ||
|  |  */ | ||
|  | std::vector<std::string> SplitPath(const std::string &str, const std::set<char> delimiters) | ||
|  | { | ||
|  |     std::vector<std::string> result; | ||
|  |     std::string temp = str; | ||
|  |     auto start = temp.begin(); | ||
|  |     auto it = temp.begin(); | ||
|  |     for (; it != temp.end(); ++it) { | ||
|  |         if (delimiters.find(*it) != delimiters.end()) { | ||
|  |             if (it != start) { | ||
|  |                 result.emplace_back(start, it); | ||
|  |             } else { | ||
|  |                 result.emplace_back(""); | ||
|  |             } | ||
|  |             start = it + 1; | ||
|  |         } | ||
|  |     } | ||
|  |     result.emplace_back(start, it); | ||
|  |     return result; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Create a directory | ||
|  |  * | ||
|  |  * @param dirPath the directory we want to create | ||
|  |  * @return APP_ERR_OK if create success, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR CreateDir(const std::string &dirPath) | ||
|  | { | ||
|  | #ifndef _WIN32
 | ||
|  |     SetFileDefaultUmask(); | ||
|  |     if (dirPath.length() > PATH_MAX) { | ||
|  |         LogError << dirPath << "is larger than " << std::to_string(PATH_MAX) << "."; | ||
|  |         return APP_ERR_COMM_NO_EXIST; | ||
|  |     } | ||
|  | #else
 | ||
|  |     if (dirPath.length() > MAX_PATH) { | ||
|  |         LogError << dirPath << "is larger than " << std::to_string(MAX_PATH) << "."; | ||
|  |     } | ||
|  |     return APP_ERR_COMM_NO_EXIST; | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     // Check the write authority of directory, if not exist, create it
 | ||
|  |     int dirExist = access(dirPath.c_str(), W_OK); | ||
|  |     if (-1 == dirExist) { | ||
|  | #ifdef _WIN32
 | ||
|  |         if (_mkdir(dirPath.c_str()) == -1) { | ||
|  | #else
 | ||
|  |         if (mkdir(dirPath.c_str(), S_IRUSR | S_IWUSR | S_IXUSR) == -1) { | ||
|  | #endif
 | ||
|  |             return APP_ERR_COMM_NO_EXIST; | ||
|  |         } | ||
|  |     } | ||
|  |     return APP_ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Create directory recursively | ||
|  |  * | ||
|  |  * @param file target directory to create | ||
|  |  */ | ||
|  | void CreateDirRecursively(const std::string &file) | ||
|  | { | ||
|  |     CreateDirRecursivelyByFile(file); | ||
|  |     if (access(file.c_str(), 0) != 0) { | ||
|  | #ifndef _WIN32
 | ||
|  |         int result = mkdir(file.c_str(), S_IRUSR | S_IWUSR | S_IXUSR); // for linux
 | ||
|  | #else
 | ||
|  |         int result = _mkdir(file.c_str()); | ||
|  | #endif
 | ||
|  |         if (result < 0) { | ||
|  |             LogError << "mkdir logs file " << file << " fail."; | ||
|  |             return; | ||
|  |         } | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Create directory recursively by file | ||
|  |  * | ||
|  |  * @param file target file to create | ||
|  |  */ | ||
|  | void CreateDirRecursivelyByFile(const std::string &file) | ||
|  | { | ||
|  |     size_t pos = file.rfind('/'); // for linux
 | ||
|  |     std::string filePath = file.substr(0, pos); | ||
|  |     if (access(filePath.c_str(), 0) != 0) { | ||
|  |         CreateDirRecursivelyByFile(filePath); | ||
|  | #ifndef _WIN32
 | ||
|  |         int result = mkdir(filePath.c_str(), S_IRUSR | S_IWUSR | S_IXUSR); // for linux
 | ||
|  | #else
 | ||
|  |         int result = _mkdir(filePath.c_str()); | ||
|  | #endif
 | ||
|  |         if (result < 0) { | ||
|  |             LogError << "mkdir logs file " << filePath << " fail."; | ||
|  |             return; | ||
|  |         } | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Read a file, store it into the RawData structure | ||
|  |  * | ||
|  |  * @param filePath file to read to | ||
|  |  * @param fileData RawData structure to store in | ||
|  |  * @return APP_ERR_OK if create success, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR ReadFile(const std::string &filePath, RawData &fileData) | ||
|  | { | ||
|  |     std::string resolvedPath; | ||
|  |     APP_ERROR ret = GetRealPath(filePath, resolvedPath); | ||
|  |     if (ret != APP_ERR_OK) { | ||
|  |         return ret; | ||
|  |     } | ||
|  |     // Open file with reading mode
 | ||
|  |     FILE *fp = fopen(resolvedPath.c_str(), "rb"); | ||
|  |     if (fp == nullptr) { | ||
|  |         LogError << "Failed to open file"; | ||
|  |         return APP_ERR_COMM_OPEN_FAIL; | ||
|  |     } | ||
|  |     // Get the length of input file
 | ||
|  |     fseek(fp, 0, SEEK_END); | ||
|  |     long fileSize = ftell(fp); | ||
|  |     fseek(fp, 0, SEEK_SET); | ||
|  |     // If file not empty, read it into FileInfo and return it
 | ||
|  |     if (fileSize > 0) { | ||
|  |         fileData.lenOfByte = fileSize; | ||
|  |         fileData.data = std::make_shared<uint8_t>(); | ||
|  |         fileData.data.reset(new uint8_t[fileSize], std::default_delete<uint8_t[]>()); | ||
|  |         size_t readRet = fread(fileData.data.get(), 1, fileSize, fp); | ||
|  |         if (readRet <= 0) { | ||
|  |             fclose(fp); | ||
|  |             return APP_ERR_COMM_READ_FAIL; | ||
|  |         } | ||
|  |         fclose(fp); | ||
|  |         return APP_ERR_OK; | ||
|  |     } | ||
|  |     fclose(fp); | ||
|  |     return APP_ERR_COMM_FAILURE; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Read a binary file, store the data into a uint8_t array | ||
|  |  * | ||
|  |  * @param fileName the file for reading | ||
|  |  * @param buffShared a shared pointer to a uint8_t array for storing file | ||
|  |  * @param buffLength the length of the array | ||
|  |  * @return APP_ERR_OK if create success, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR ReadBinaryFile(const std::string &fileName, std::shared_ptr<uint8_t> &buffShared, int &buffLength) | ||
|  | { | ||
|  |     // read binary file
 | ||
|  |     std::ifstream inFile(fileName, std::ios::in | std::ios::binary); | ||
|  |     if (!inFile) { | ||
|  |         LogError << "FaceFeatureLib: read file " << fileName << " fail."; | ||
|  |         return APP_ERR_COMM_READ_FAIL; | ||
|  |     } | ||
|  | 
 | ||
|  |     // get length of file:
 | ||
|  |     inFile.seekg(0, inFile.end); | ||
|  |     buffLength = inFile.tellg(); | ||
|  |     inFile.seekg(0, inFile.beg); | ||
|  | 
 | ||
|  |     auto tempShared = std::make_shared<uint8_t>(); | ||
|  |     tempShared.reset(new uint8_t[buffLength], std::default_delete<uint8_t[]>()); | ||
|  |     inFile.read((char *)tempShared.get(), buffLength); | ||
|  |     inFile.close(); | ||
|  |     buffShared = tempShared; | ||
|  | 
 | ||
|  |     LogDebug << "read file: fileName=" << fileName << ", size=" << buffLength << "."; | ||
|  | 
 | ||
|  |     return APP_ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Read a file with specified offset | ||
|  |  * Only used in Jpegd | ||
|  |  * | ||
|  |  * @param fileName the file for reading | ||
|  |  * @param fileData RawData structure to store in | ||
|  |  * @param offset offset for file | ||
|  |  * @return APP_ERR_OK if create success, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR ReadFileWithOffset(const std::string &fileName, RawData &fileData, const uint32_t offset) | ||
|  | { | ||
|  |     std::string resolvedPath; | ||
|  |     APP_ERROR ret = GetRealPath(fileName, resolvedPath); | ||
|  |     if (ret != APP_ERR_OK) { | ||
|  |         return ret; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Open file with reading mode
 | ||
|  |     FILE *fp = fopen(resolvedPath.c_str(), "rb"); | ||
|  |     if (fp == nullptr) { | ||
|  |         LogError << "Failed to open file"; | ||
|  |         return APP_ERR_COMM_OPEN_FAIL; | ||
|  |     } | ||
|  |     // Get the length of input file
 | ||
|  |     fseek(fp, 0, SEEK_END); | ||
|  |     long fileSize = ftell(fp); | ||
|  |     fseek(fp, 0, SEEK_SET); | ||
|  |     // If file not empty, read it into FileInfo and return it
 | ||
|  |     if (fileSize > 0) { | ||
|  |         fileData.lenOfByte = fileSize; | ||
|  |         fileData.data = std::make_shared<uint8_t>(); | ||
|  |         fileData.data.reset(new uint8_t[fileSize + offset], std::default_delete<uint8_t[]>()); | ||
|  |         size_t readRet = fread(fileData.data.get(), 1, fileSize, fp); | ||
|  |         if (readRet <= 0) { | ||
|  |             fclose(fp); | ||
|  |             return APP_ERR_COMM_READ_FAIL; | ||
|  |         } | ||
|  |         fclose(fp); | ||
|  |         return APP_ERR_OK; | ||
|  |     } | ||
|  |     fclose(fp); | ||
|  |     return APP_ERR_COMM_FAILURE; | ||
|  | } | ||
|  | 
 | ||
|  | APP_ERROR GetRealPath(const std::string &srcPath, std::string &resolvedPath) | ||
|  | { | ||
|  |     // Get the absolute path of input file
 | ||
|  | #ifndef _WIN32
 | ||
|  |     char path[PATH_MAX + 1] = {0}; | ||
|  |     if ((strlen(srcPath.c_str()) > PATH_MAX) || (realpath(srcPath.c_str(), path) == nullptr)) { | ||
|  | #else
 | ||
|  |     #pragma comment(lib, "Shlwapi.lib")
 | ||
|  |     char path[MAX_PATH + 1] = {0}; | ||
|  |     if ((strlen(srcPath.c_str()) > MAX_PATH) || (_fullpath(path, srcPath.c_str(), MAX_PATH) == nullptr)) { | ||
|  | #endif
 | ||
|  |         LogError << "Failed to get canonicalize path----" << srcPath; | ||
|  |         return APP_ERR_COMM_NO_EXIST; | ||
|  |     } | ||
|  |     resolvedPath = path; | ||
|  |     return APP_ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Get the extension name of input file | ||
|  |  * | ||
|  |  * @param filePath the file for reading extension name | ||
|  |  * @return extension name of the file | ||
|  |  */ | ||
|  | std::string GetExtension(const std::string &filePath) | ||
|  | { | ||
|  |     std::set<char> delims { '.' }; | ||
|  |     std::vector<std::string> path = SplitPath(filePath, delims); | ||
|  |     return path[path.size() - 1]; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Get file canonical name | ||
|  |  * | ||
|  |  * @param filePath absolute path of the target file | ||
|  |  * @return filename of the file | ||
|  |  */ | ||
|  | std::string GetName(const std::string &filePath) | ||
|  | { | ||
|  |     std::set<char> delims { '/' }; | ||
|  |     std::vector<std::string> path = SplitPath(filePath, delims); | ||
|  |     return (path.size() < 1) ? "" : path[path.size() - 1]; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Get the Parent of input file | ||
|  |  * | ||
|  |  * @param filePath file for looking for parent | ||
|  |  * @return parent of the file | ||
|  |  */ | ||
|  | std::string GetParent(const std::string &filePath) | ||
|  | { | ||
|  |     std::set<char> delims { '/' }; | ||
|  |     std::vector<std::string> path = SplitPath(filePath, delims); | ||
|  |     return (path.size() < TWO) ? "" : path[path.size() - TWO]; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Change the current directory | ||
|  |  * | ||
|  |  * @param dir target directory to change to | ||
|  |  * @return APP_ERR_OK if create success, error code otherwise | ||
|  |  */ | ||
|  | APP_ERROR ChangeDir(const std::string &dir) | ||
|  | { | ||
|  |     std::string resolvedPath; | ||
|  |     APP_ERROR ret = GetRealPath(dir, resolvedPath); | ||
|  |     if (ret != APP_ERR_OK) { | ||
|  |         return ret; | ||
|  |     } | ||
|  | #ifndef _WIN32
 | ||
|  |     char path[PATH_MAX + 1] = {0}; | ||
|  |     resolvedPath.copy(path, resolvedPath.length()); | ||
|  |     char *dName = dirname(path); | ||
|  |     if (dName == nullptr) { | ||
|  |         return APP_ERR_COMM_NO_EXIST; | ||
|  |     } | ||
|  | #else
 | ||
|  |     char path[MAX_PATH + 1] = {0}; | ||
|  |     resolvedPath.copy(path, resolvedPath.length()); | ||
|  |     if (!PathRemoveFileSpecA(path)) { | ||
|  |         return APP_ERR_COMM_NO_EXIST; | ||
|  |     } | ||
|  |     char *dName = path; | ||
|  | #endif
 | ||
|  |     if (chdir(dName) != 0) { | ||
|  |         return APP_ERR_COMM_NO_EXIST; | ||
|  |     } | ||
|  |     return APP_ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Append stream to file | ||
|  |  * | ||
|  |  * @param fileName to append to | ||
|  |  * @param stream content of string | ||
|  |  * @param streamLength length of string | ||
|  |  */ | ||
|  | void SaveFileAppend(const std::string &fileName, const std::string &stream, const int streamLength) | ||
|  | { | ||
|  |     LogDebug << "saving binary file by app: fileName=" << fileName << ", streamLength=" << streamLength; | ||
|  |     std::ofstream outfile(fileName, std::ios::app | std::ofstream::binary); | ||
|  |     outfile.write(stream.c_str(), streamLength); | ||
|  |     outfile.close(); | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Overwrite a file with stream | ||
|  |  * | ||
|  |  * @param fileName to overwrite to | ||
|  |  * @param stream content of string | ||
|  |  * @param streamLength length of string | ||
|  |  */ | ||
|  | void SaveFileOverwrite(const std::string &fileName, const std::string &stream, const int streamLength) | ||
|  | { | ||
|  |     LogDebug << "Saving binary file by over write: fileName=" << fileName << ", streamLength=" << streamLength; | ||
|  |     std::ofstream outfile(fileName, std::ios::out | std::ofstream::binary); | ||
|  |     outfile.write(stream.c_str(), streamLength); | ||
|  |     outfile.close(); | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Copy file | ||
|  |  * | ||
|  |  * @param srcFile from source | ||
|  |  * @param destFile to destination | ||
|  |  */ | ||
|  | void CopyFile(const std::string &srcFile, const std::string &destFile) | ||
|  | { | ||
|  |     std::ifstream in(srcFile, std::ios::binary); | ||
|  |     if (!in) { | ||
|  |         LogError << "Failed to get source file, it may be not exists. srcFile=" << srcFile; | ||
|  |         return; | ||
|  |     } | ||
|  |     std::ofstream out(destFile, std::ios::binary); | ||
|  |     if (!out) { | ||
|  |         LogError << "Failed to save destination file. destFile=" << destFile; | ||
|  |         in.close(); | ||
|  |         return; | ||
|  |     } | ||
|  |     char flush[BUFFER_SIZE]; | ||
|  |     while (!in.eof()) { | ||
|  |         in.read(flush, BUFFER_SIZE); | ||
|  |         out.write(flush, in.gcount()); | ||
|  |     } | ||
|  |     out.close(); | ||
|  |     in.close(); | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Save file with timestamp under specified folder | ||
|  |  * | ||
|  |  * @param dataBuffer  buffer of file | ||
|  |  * @param bufferSize   buffer size | ||
|  |  * @param folderName   specified folder will be created if it not existed | ||
|  |  * @param fileName     file name without suffix, the finally name will append time stamp to it | ||
|  |  * @param fileSuffix   suffix name of file | ||
|  |  */ | ||
|  | APP_ERROR SaveFileWithTimeStamp(std::shared_ptr<void> dataBuffer, uint32_t bufferSize, std::string folderName, | ||
|  |                                 std::string fileName, std::string fileSuffix) | ||
|  | { | ||
|  | #ifndef _WIN32
 | ||
|  |     SetFileDefaultUmask(); | ||
|  | #endif
 | ||
|  |     APP_ERROR ret; | ||
|  |     if (folderName.length() != 0) { | ||
|  |         ret = CreateDir(folderName); | ||
|  |         if (ret != APP_ERR_OK) { | ||
|  |             return ret; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     // Result file name use the time stamp as a suffix
 | ||
|  |     std::string timeString; | ||
|  |     GetCurTimeString(timeString); | ||
|  | 
 | ||
|  |     // Create file under folderName directory
 | ||
|  |     std::stringstream resultPathName; | ||
|  |     if (folderName.length() == 0) { | ||
|  |         resultPathName << "./" << fileName << "_" << timeString << fileSuffix; | ||
|  |     } else { | ||
|  |         resultPathName << folderName << "/" << fileName << "_" << timeString << fileSuffix; | ||
|  |     } | ||
|  |     std::string resolvedPath; | ||
|  |     ret = GetRealPath(resultPathName.str(), resolvedPath); | ||
|  |     if (ret != APP_ERR_OK) { | ||
|  |         return ret; | ||
|  |     } | ||
|  | 
 | ||
|  |     FILE *fp = fopen(resolvedPath.c_str(), "wb"); | ||
|  |     if (fp == nullptr) { | ||
|  |         LogError << "Failed to open file"; | ||
|  |         return APP_ERR_COMM_OPEN_FAIL; | ||
|  |     } | ||
|  |     uint32_t result = fwrite(dataBuffer.get(), 1, bufferSize, fp); | ||
|  |     if (result != bufferSize) { | ||
|  |         LogError << "Failed to write file"; | ||
|  |         fclose(fp); | ||
|  |         return APP_ERR_COMM_WRITE_FAIL; | ||
|  |     } | ||
|  |     LogInfo << "Write result to file successfully"; | ||
|  |     uint32_t ff = fflush(fp); | ||
|  |     if (ff != 0) { | ||
|  |         LogError << "Failed to fflush file"; | ||
|  |         fclose(fp); | ||
|  |         return APP_ERR_COMM_DESTORY_FAIL; | ||
|  |     } | ||
|  |     uint32_t fc = fclose(fp); | ||
|  |     if (fc != 0) { | ||
|  |         LogError << "Failed to fclose file"; | ||
|  |         return APP_ERR_COMM_DESTORY_FAIL; | ||
|  |     } | ||
|  |     return APP_ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * Convert the current time to the format "%Y%m%d%H%M%S" | ||
|  |  * | ||
|  |  * @param timeString buffer to save the time string with format "%Y%m%d%H%M%S" | ||
|  |  */ | ||
|  | void GetCurTimeString(std::string &timeString) | ||
|  | { | ||
|  |     // Result file name use the time stamp as a suffix
 | ||
|  |     const int timeZoneDiff = 28800; // 8 hour time difference
 | ||
|  |     const int timeStringSize = 32; | ||
|  |     char timeStr[timeStringSize] = {0}; | ||
|  |     time_t tmValue = time(nullptr) + timeZoneDiff; | ||
|  |     struct tm tmStruct = {0}; | ||
|  | #ifdef _WIN32
 | ||
|  |     if (0 == gmtime_s(&tmStruct, &tmValue)) { | ||
|  | #else
 | ||
|  |     if (nullptr != gmtime_r(&tmValue, &tmStruct)) { | ||
|  | #endif
 | ||
|  |         strftime(timeStr, sizeof(timeStr), "%Y%m%d%H%M%S", &tmStruct); | ||
|  |     } | ||
|  |     timeString = timeStr; | ||
|  |     return; | ||
|  | } |