TDME2 1.9.121
archive-main.cpp
Go to the documentation of this file.
1#include <cassert>
2#include <fstream>
3#include <string>
4#include <vector>
5
6#include <ext/zlib/zlib.h>
7
8#include <tdme/tdme.h>
10#include <tdme/engine/Version.h>
18
19using std::ofstream;
20using std::string;
21using std::to_string;
22using std::vector;
23
33
34namespace tdme {
35namespace tools {
36namespace cli {
37namespace archive {
39 string name;
40 uint64_t bytes;
41 uint8_t compressed;
43 uint64_t offset;
45 };
46};
47};
48};
49};
50
52
53void scanPath(const string& path, vector<string>& totalFiles) {
54 class ListFilter : public virtual FileNameFilter {
55 public:
56 virtual ~ListFilter() {}
57
58 bool accept(const string& pathName, const string& fileName) override {
59 if (fileName == ".") return false;
60 if (fileName == "..") return false;
61 if (FileSystem::getInstance()->isPath(pathName + "/" + fileName) == true) return true;
62 auto fileNameLowerCase = StringTools::toLowerCase(fileName);
63 // audio
64 if (StringTools::endsWith(fileNameLowerCase, ".ogg") == true) return true;
65 // fonts
66 if (StringTools::endsWith(fileNameLowerCase, ".fnt") == true) return true;
67 // images
68 if (StringTools::endsWith(fileNameLowerCase, ".ico") == true) return true;
69 if (StringTools::endsWith(fileNameLowerCase, ".png") == true) return true;
70 // models
71 if (StringTools::endsWith(fileNameLowerCase, ".dae") == true) return true;
72 if (StringTools::endsWith(fileNameLowerCase, ".fbx") == true) return true;
73 if (StringTools::endsWith(fileNameLowerCase, ".glb") == true) return true;
74 if (StringTools::endsWith(fileNameLowerCase, ".tm") == true) return true;
75 // property files
76 if (StringTools::endsWith(fileNameLowerCase, ".properties") == true) return true;
77 // shader
78 if (StringTools::endsWith(fileNameLowerCase, ".cl") == true) return true;
79 if (StringTools::endsWith(fileNameLowerCase, ".frag") == true) return true;
80 if (StringTools::endsWith(fileNameLowerCase, ".glsl") == true) return true;
81 if (StringTools::endsWith(fileNameLowerCase, ".vert") == true) return true;
82 // tdme model
83 if (StringTools::endsWith(fileNameLowerCase, ".tmodel") == true) return true;
84 // tdme scene
85 if (StringTools::endsWith(fileNameLowerCase, ".tscene") == true) return true;
86 // tdme particle system
87 if (StringTools::endsWith(fileNameLowerCase, ".tparticle") == true) return true;
88 // tdme terrain
89 if (StringTools::endsWith(fileNameLowerCase, ".tterrain") == true) return true;
90 // tdme script
91 if (StringTools::endsWith(fileNameLowerCase, ".tscript") == true) return true;
92 // xml
93 if (StringTools::endsWith(fileNameLowerCase, ".xml") == true) return true;
94 // files without ending
95 if (fileName.rfind(".") == string::npos ||
96 (fileName.rfind("/") != string::npos &&
97 fileName.rfind(".") < fileName.rfind("/"))) {
98 return true;
99 }
100 //
101 return false;
102 }
103 };
104
105 ListFilter listFilter;
106 vector<string> files;
107
108 FileSystem::getInstance()->list(path, files, &listFilter);
109
110 for (auto fileName: files) {
111 if (FileSystem::getInstance()->isPath(path + "/" + fileName) == false) {
112 totalFiles.push_back(path + "/" + fileName);
113 } else {
114 scanPath(path + "/" + fileName, totalFiles);
115 }
116 }
117}
118
119void processFile(const string& fileName, vector<FileInformation>& fileInformations) {
120 Console::print("Processing file: " + fileName);
121
122 // read content
123 vector<uint8_t> content;
124 FileSystem::getInstance()->getContent(
125 FileSystem::getInstance()->getPathName(fileName),
126 FileSystem::getInstance()->getFileName(fileName),
127 content
128 );
129
130 // append to archive
131 ofstream ofs("archive.ta", ofstream::binary | ofstream::app);
132 ofs.seekp(0, ofstream::end);
133 uint64_t fileOffset = ofs.tellp();
134
135 //
136 uint64_t bytesCompressed = 0;
137 uint8_t compressed = 1;
138
139 // always use compression for now
140 if (compressed == 1) {
141 // see: https://www.zlib.net/zpipe.c
142
143 #define CHUNK 16384
144
145 int ret;
146 int flush;
147 size_t have;
148 z_stream strm;
149 unsigned char in[CHUNK];
150 unsigned char out[CHUNK];
151
152 /* allocate deflate state */
153 strm.zalloc = Z_NULL;
154 strm.zfree = Z_NULL;
155 strm.opaque = Z_NULL;
156 ret = deflateInit(&strm, Z_DEFAULT_COMPRESSION);
157 if (ret != Z_OK) {
158 Console::println("processFile(): Error compressing file: Aborting");
159 return;
160 }
161
162 // compress until end of file
163 size_t inPosition = 0;
164 size_t inBytes = content.size();
165 do {
166 auto inStartPosition = inPosition;
167 for (size_t i = 0; i < CHUNK; i++) {
168 if (inPosition == inBytes) break;
169 in[i] = content[inPosition];
170 inPosition++;
171 }
172 strm.avail_in = inPosition - inStartPosition;
173 flush = inPosition == inBytes?Z_FINISH:Z_NO_FLUSH;
174 strm.next_in = in;
175
176 // run deflate() on input until output buffer not full, finish compression if all of source has been read in
177 do {
178 strm.avail_out = CHUNK;
179 strm.next_out = out;
180 ret = deflate(&strm, flush); // no bad return value
181 assert(ret != Z_STREAM_ERROR); // state not clobbered
182 have = CHUNK - strm.avail_out;
183 ofs.write((char*)out, have);
184 bytesCompressed+= have;
185 } while (strm.avail_out == 0);
186 assert(strm.avail_in == 0); // all input will be used
187
188 // done when last data in file processed
189 } while (flush != Z_FINISH);
190 assert(ret == Z_STREAM_END); // stream will be complete
191
192 // clean up and return
193 (void) deflateEnd(&strm);
194 } else {
195 ofs.write((char*)content.data(), content.size());
196 }
197 ofs.close();
198
199 // store file information
200 FileInformation fileInformation;
201 fileInformation.name = fileName;
202 fileInformation.bytes = content.size();
203 fileInformation.compressed = compressed;
204 fileInformation.bytesCompressed = bytesCompressed;
205 fileInformation.offset = fileOffset;
206 fileInformation.executable = false;
207 fileInformations.push_back(fileInformation);
208
209 // done
210 Console::println(", processed " + to_string(content.size()) + " bytes" + (compressed == 1?", " + to_string(bytesCompressed) + " bytes compressed":""));
211}
212
213int main(int argc, char** argv)
214{
215 Console::println(string("archive ") + Version::getVersion());
216 Console::println(Version::getCopyright());
217 Console::println();
218
219 if (argc != 1) {
220 Console::println("Usage: archive");
221 Application::exit(1);
222 }
223
224 // scan files
225 Console::println("Scanning files");
226 vector<string> files;
227 scanPath("resources", files);
228 scanPath("shader", files);
229
230 // processing
231 Console::println("Processing files");
232
233 // reset archive
234 {
235 ofstream ofs("archive.ta", ofstream::binary | ofstream::trunc);
236 ofs.close();
237 }
238
239 // add files to archive
240 vector<FileInformation> fileInformations;
241 for (auto fileName: files) {
242 processFile(fileName, fileInformations);
243 }
244
245 // add file informations
246 {
247 ofstream ofs("archive.ta", ofstream::binary | ofstream::app);
248 ofs.seekp(0, ofstream::end);
249 uint32_t fileInformationOffsetEnd = 0LL;
250 uint64_t fileInformationOffset = ofs.tellp();
251 for (auto& fileInformation: fileInformations) {
252 uint32_t nameSize = fileInformation.name.size();
253 ofs.write((char*)&nameSize, sizeof(nameSize));
254 for (auto i = 0; i < nameSize; i++) ofs.write(&fileInformation.name[i], 1);
255 ofs.write((char*)&fileInformation.bytes, sizeof(fileInformation.bytes));
256 ofs.write((char*)&fileInformation.compressed, sizeof(fileInformation.compressed));
257 ofs.write((char*)&fileInformation.bytesCompressed, sizeof(fileInformation.bytesCompressed));
258 ofs.write((char*)&fileInformation.offset, sizeof(fileInformation.offset));
259 ofs.write((char*)&fileInformation.executable, sizeof(fileInformation.executable));
260 }
261 ofs.write((char*)&fileInformationOffsetEnd, sizeof(fileInformationOffsetEnd));
262 ofs.write((char*)&fileInformationOffset, sizeof(fileInformationOffset));
263 ofs.close();
264 }
265}
void scanPath(const string &path, vector< string > &totalFiles)
#define CHUNK
int main(int argc, char **argv)
void processFile(const string &fileName, vector< FileInformation > &fileInformations)
Application base class, please make sure to allocate application on heap to have correct application ...
Definition: Application.h:37
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
String tokenizer class.
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19
Definition: fwd-tdme.h:4
File system file name filter interface.
virtual bool accept(const string &path, const string &file)=0
Accept a file.