Agrarsense
LogFile.cpp
Go to the documentation of this file.
1// Copyright (c) 2024 FrostBit Software Lab at the Lapland University of Applied Sciences
2//
3// This work is licensed under the terms of the MIT license.
4// For a copy, see <https://opensource.org/licenses/MIT>.
5
6#include "LogFile.h"
9
10#include "Misc/Paths.h"
11
12#include <filesystem>
13#include <iostream>
14#include <iomanip>
15#include <ctime>
16
17ULogFile* ULogFile::CreateLogFile(const FString& FileNameWithoutExtension, FLogFileSettings Settings)
18{
19 ULogFile* LogFile = NewObject<ULogFile>(ULogFile::StaticClass());
20 LogFile->AddToRoot();
21 if (LogFile)
22 {
23 LogFile->Create(FileNameWithoutExtension, Settings);
24 }
25
26 return LogFile;
27}
28
30{
31
32}
33
35{
37 Close();
38}
39
40void ULogFile::Create(const FString& FileNameWithoutExtension, FLogFileSettings Settings)
41{
42 CurrentSettings = Settings;
43
44 FString ProjectPath = UAgrarsensePaths::GetDataFolder();
45 FString Folder = FString("/Logs/");
46 FString LogDirPath = FPaths::Combine(*ProjectPath, *Folder);
47
48 if (!std::filesystem::exists(TCHAR_TO_UTF8(*LogDirPath)))
49 {
50 if (!std::filesystem::create_directory(TCHAR_TO_UTF8(*LogDirPath)))
51 {
52 // Failed to create the directory, return
53 return;
54 }
55 }
56
57 FString LogFilePath;
58
59 if (Settings.OverrideFilePath && !Settings.FilePath.IsEmpty())
60 {
61 if (!std::filesystem::exists(TCHAR_TO_UTF8(*Settings.FilePath)))
62 {
63 bool created = std::filesystem::create_directories(TCHAR_TO_UTF8(*Settings.FilePath));
64 }
65 LogFilePath = FPaths::Combine(*Settings.FilePath, *FileNameWithoutExtension) + TEXT(".txt");
66 }
67 else
68 {
69 LogFilePath = FPaths::Combine(*LogDirPath, *FileNameWithoutExtension) + TEXT(".txt");
70 }
71
72 FilePath = LogFilePath;
73
75 {
77 // Clear file if it exists
78 Clear();
79 break;
80
82 // Append to existing file if it exists
83 break;
84
86 {
87 // Create unique file by adding number to the end until we find unique filename
88 int suffix = 0;
89 FString UniqueFileName = FileNameWithoutExtension + TEXT("_") + FString::FromInt(suffix++);
90 FString UniqueLogFilePath = FPaths::Combine(*LogDirPath, *UniqueFileName) + TEXT(".txt");
91
92 while (std::filesystem::exists(TCHAR_TO_UTF8(*UniqueLogFilePath)))
93 {
94 UniqueFileName = FileNameWithoutExtension + TEXT("_") + FString::FromInt(suffix++) + TEXT(".txt");
95 UniqueLogFilePath = FPaths::Combine(*LogDirPath, *UniqueFileName);
96 }
97
98 LogFilePath = UniqueLogFilePath;
99 break;
100 }
101
102 default:
103 break;
104 }
105
106 // Try to create the .txt file
107 std::ofstream newFile(TCHAR_TO_UTF8(*LogFilePath), std::ios::app);
108 if (newFile.is_open())
109 {
110 newFile.close();
111 }
112 else
113 {
114 std::ifstream file(TCHAR_TO_UTF8(*LogFilePath));
115 if (file.good())
116 {
117 // The file already exists, return
118 file.close();
119 }
120 }
121}
122
123void ULogFile::Write(const FString& Text)
124{
125 if (FilePath.IsEmpty() || !IsValid(this))
126 {
127 return;
128 }
129
131 {
133 {
135 }
136 else
137 {
138 // Add message to queue (with timestamp if CurrentSettings.Timestamp) and return
139 FString MessageToAdd = Text;
141 {
142 std::time_t now = std::time(nullptr);
143 std::tm tm_time = *std::localtime(&now);
144 char timestamp[12];
145 std::strftime(timestamp, sizeof(timestamp), "[%H:%M:%S] ", &tm_time);
146 MessageToAdd = FString(timestamp) + Text;
147 }
148 MessageQueue.Add(MessageToAdd);
149 }
150
151 return;
152 }
153
154 WriteToFile(Text, true);
155}
156
158{
159 if (LogFileStream.is_open())
160 {
161 LogFileStream.close();
162 }
163}
164
166{
167 if (std::filesystem::exists(TCHAR_TO_UTF8(*FilePath)))
168 {
169 std::ofstream clearFile(TCHAR_TO_UTF8(*FilePath), std::ofstream::out | std::ofstream::trunc);
170 clearFile.close();
171 }
172}
173
175{
176 if (IsValid(this))
177 {
179 Close();
180 ConditionalBeginDestroy();
181 }
182}
183
185{
187}
188
190{
191 if (MessageQueue.IsEmpty() || FilePath.IsEmpty())
192 {
193 return;
194 }
195
196 if (!LogFileStream.is_open())
197 {
198 std::string filePathString(TCHAR_TO_UTF8(*FilePath));
199 LogFileStream.open(filePathString, std::ios::app);
200 if (!LogFileStream.is_open())
201 {
202 // Failed to open the file, return
203 return;
204 }
205 }
206
207 for (const FString& Message : MessageQueue)
208 {
209 WriteToFile(Message, false);
210 }
211
212 MessageQueue.Empty();
213
215 {
216 LogFileStream.close();
217 }
218}
219
220void ULogFile::WriteToFile(const FString& Text, bool UseTimestamp)
221{
222 if (FilePath.IsEmpty() || !IsValid(this))
223 {
224 return;
225 }
226
227 if (!LogFileStream.is_open())
228 {
229 std::string filePathString(TCHAR_TO_UTF8(*FilePath));
230 LogFileStream.open(filePathString, std::ios::app);
231 if (!LogFileStream.is_open())
232 {
233 // Failed to open the file, return
234 return;
235 }
236 }
237
238 if (UseTimestamp && CurrentSettings.Timestamp)
239 {
240 std::time_t now = std::time(nullptr);
241 std::tm tm_time = *std::localtime(&now);
242 char timestamp[12];
243 std::strftime(timestamp, sizeof(timestamp), "[%H:%M:%S] ", &tm_time);
244
245 LogFileStream << timestamp << TCHAR_TO_UTF8(*Text) << std::endl;
246 }
247 else
248 {
249 LogFileStream << TCHAR_TO_UTF8(*Text) << std::endl;
250 }
251
253 {
254 LogFileStream.close();
255 }
256}
static FString GetDataFolder()
static void OpenFileExplorer(FString Path)
ULogFile()
Definition: LogFile.cpp:29
void Clear()
Definition: LogFile.cpp:165
void Destroy()
Definition: LogFile.cpp:174
void WriteQueuedMessages()
Definition: LogFile.cpp:189
FLogFileSettings CurrentSettings
Definition: LogFile.h:145
void OpenFilePath()
Definition: LogFile.cpp:184
void Write(const FString &Text)
Definition: LogFile.cpp:123
std::ofstream LogFileStream
Definition: LogFile.h:149
void Close()
Definition: LogFile.cpp:157
void WriteToFile(const FString &Text, bool UseTimestamp)
Definition: LogFile.cpp:220
static ULogFile * CreateLogFile(const FString &FileNameWithoutExtension, FLogFileSettings Settings=FLogFileSettings())
Definition: LogFile.cpp:17
void Create(const FString &FileNameWithoutExtension, FLogFileSettings Settings)
Definition: LogFile.cpp:40
FString FilePath
Definition: LogFile.h:147
~ULogFile()
Definition: LogFile.cpp:34
TArray< FString > MessageQueue
Definition: LogFile.h:143
bool KeepFileOpen
Definition: LogFile.h:42
bool Timestamp
Definition: LogFile.h:39
FString FilePath
Definition: LogFile.h:54
FFileWriteOptions FileWriteOptions
Definition: LogFile.h:45
int32 QueueLength
Definition: LogFile.h:48
bool OverrideFilePath
Definition: LogFile.h:51
FFileCreationOptions FileCreationOptions
Definition: LogFile.h:36