forked from zproksi/bpatch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfileprocessing.h
265 lines (219 loc) · 8.46 KB
/
fileprocessing.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
#pragma once
#include <filesystem>
#include <limits>
#include <span>
#include <vector>
#include "flexiblecache.h"
namespace bpatch
{
//------------------------------------------------------
/// <summary>
/// interface for source of the data
/// </summary>
class Reader
{
Reader(const Reader&) = delete;
Reader& operator =(const Reader&) = delete;
Reader(Reader&&) = delete;
Reader& operator =(Reader&&) = delete;
public:
Reader() = default;
virtual ~Reader() = default;
/// <summary>
/// reads data to span. span should contain maximum ampount of data to read
/// </summary>
/// <param name="place">place where readed data to hold. and maximum data to read</param>
/// <returns>the span but with data amount readed</returns>
virtual std::span<char> ReadData(const std::span<char> place) = 0;
/// <summary>
/// Check if read everything
/// </summary>
/// <returns> true if we read all the data</returns>
virtual bool FileReaded()const noexcept = 0;
/// <summary>
/// how much data we have readed already
/// </summary>
/// <returns>amount of data already readed from file</returns>
virtual size_t Readed() const noexcept = 0;
};
//------------------------------------------------------
/// <summary>
/// interface for writing of the data
/// </summary>
class Writer
{
Writer(const Writer&)= delete;
Writer& operator =(const Writer&) = delete;
Writer(Writer&&) = delete;
Writer& operator =(Writer&&) = delete;
public:
Writer() = default;
virtual ~Writer() = default;
/// <summary>
/// Write character to somewhere unless aEod is true
/// </summary>
/// <param name="toProcess">character to write</param>
/// <param name="aEod">true if it is end of data and we should not write</param>
/// <returns>Actually written data. Could be 0 if we just accumulated character</returns>
virtual size_t WriteCharacter(const char toProcess, const bool aEod) = 0;
/// <summary>
/// how much data we have written already
/// </summary>
/// <returns>amount of data already written to file</returns>
virtual size_t Written() const noexcept = 0;
};
//------------------------------------------------------
/// <summary>
/// Class holds basic file stream.
/// closes it in the destructor
/// </summary>
class FileProcessing
{
FileProcessing(const FileProcessing&) = delete;
FileProcessing& operator=(const FileProcessing&) = delete;
FileProcessing(FileProcessing&&) = delete;
FileProcessing& operator=(FileProcessing&&) = delete;
public:
/// <summary>
/// opens file with requested mode
/// </summary>
/// <param name="fname">file name to open</param>
/// <param name="mode">mode of file to open. See std::fopen documentation</param>
FileProcessing(const char* fname, const char* mode);
/// <summary>
/// closes file here
/// </summary>
~FileProcessing();
protected:
/// <summary>
/// our stream for read or write
/// </summary>
std::FILE* stream_ = nullptr;
private:
/// <summary>
/// bufferization of io
/// </summary>
std::vector<char> buff_;
};
//------------------------------------------------------
/// <summary>
/// Read data from zero position.
/// </summary>
class ReadFileProcessing : virtual public FileProcessing, public Reader
{
public:
/// <summary>
/// open file for reading only
/// </summary>
/// <param name="fname">file name to open</param>
/// <param name="mode"> mode of file - rb</param>
ReadFileProcessing(const char* fname, const char* mode = "rb");
/// <summary>
/// Read Data from file and put it into span
/// total length of data in span can be 0
/// if end of file reached
/// </summary>
/// <param name="place">place where readed data to hold. and maximum data to read</param>
/// <returns>the span but with data amount readed</returns>
std::span<char> ReadData(const std::span<char> place) override;
/// <summary>
/// Check if read everything from file
/// </summary>
/// <returns> true if we read all the data from file</returns>
bool FileReaded()const noexcept override {return eof_;}
/// <summary>
/// how much data we have readed already
/// </summary>
/// <returns>amount of data already readed from file</returns>
size_t Readed() const noexcept override {return readedAmount_;};
protected:
bool eof_ = false; // if we reached end of file during reading
size_t readedAmount_ = 0; // how many bytes we have readed
};
//------------------------------------------------------
class FlexibleCache;
/// <summary>
/// Write data from zero position.
/// </summary>
class WriteFileProcessing : virtual public FileProcessing, public Writer
{
public:
/// <summary>
/// Creates/overwrites file for writing
/// </summary>
/// <param name="fname">file name to write to</param>
/// <param name="mode">writing mode - wb is default</param>
WriteFileProcessing(const char* fname, const char* mode = "wb");
size_t WriteCharacter(const char toProcess, const bool aEod) override;
size_t Written() const noexcept override; // the only way to get writeAt_
protected:
/// <summary>
/// !unlocked! write file inside. Throws If the written amount do not equal to the requested
/// </summary>
/// <param name="sv"> data to write</param>
/// <returns>written amount</returns>
size_t WriteAndThrowIfFail(const std::string_view sv); // here writeAt_ could grow
/// <summary>
/// write either full chunks only or everything if aEod was achieved
/// </summary>
/// <param name="aEod">end of data is here - write everything</param>
/// <returns>number of written bytes</returns>
size_t WriteEverythingOrFullChunks(const bool aEod);
// we will write to the file only by big chunks or if the data ends
// data will be accumulated here
std::unique_ptr<FlexibleCache> cache_;
private:
size_t writeAt_ = 0; // now we are writing at
};
///------------------------------------------------------
/// <summary>
/// Read data from zero position.
/// Write data from zero position but not further than you have readed
/// </summary>
class ReadWriteFileProcessing final : virtual public ReadFileProcessing, virtual public WriteFileProcessing
{
public:
/// <summary>
/// opens file for reading/writing
/// </summary>
/// <param name="fname">file name to read from/write to</param>
/// <param name="mode">writing mode - r+b is default.
/// read from start</param>
ReadWriteFileProcessing(const char* fname, const char* mode = "r+b");
/// <summary>
/// save readAt_ and continue reading from readAt_ always.
/// set readAt_ to maximum size_t when file has been readed
/// </summary>
/// <param name="place">place where readed data to hold. and maximum data to read</param>
/// <returns>the span but with data amount readed</returns>
/// throws in case of error</returns>
std::span<char> ReadData(const std::span<char> place) override;
/// <summary>
/// save writeAt_ and continue writing from writeAt_ always.
/// write not later than readAt_.
/// </summary>
/// <param name="toProcess">character to add to cache</param>
/// <param name="aEod">sign that no more data in the current session
/// to write everything what was cached</param>
/// <returns>how may bytes were written into the file (not chached)</returns>
size_t WriteCharacter(const char toProcess, const bool aEod) override;
protected:
/// <summary>
/// Set position in file for reading or for writing
/// </summary>
/// <param name="bReading"> true if we need to read</param>
/// <returns>throws if fail</returns>
void SeekSet(const bool bReading);
};
/// <summary>
/// Reads full file to provided vector
/// Initially, just check if the fname is file and has size.
/// opens the file if found it. If file cannot be found merges fname to
/// provided additionalPath and search again
/// </summary>
/// <param name="readTo">vector to read data to</param>
/// <param name="fname">the filename of the file to open</param>
/// <param name="additionalPath">path where search the file if it was not open in the current folder</param>
/// <returns>true if file exists and data readed, even if the file has 0 size</returns>
bool ReadFullFile(std::vector<char>& readTo, const char* const fname, const std::filesystem::path& additionalPath);
};// namespace bpatch