@coralogix/rum-cli 1.1.26 → 1.1.28
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +86 -75
- package/api/client/request-builder.ts +40 -57
- package/api/dsym.api.ts +16 -13
- package/api/proguard.api.ts +64 -0
- package/api/react-native.api.ts +8 -14
- package/api/source-maps.api.ts +7 -13
- package/cli/commands/proguard/index.ts +5 -0
- package/cli/commands/proguard/upload-proguard-command.ts +36 -0
- package/cli/rum-cli.ts +5 -0
- package/config/config.ts +2 -0
- package/consts/consts.ts +0 -1
- package/dist/api/client/client-factory.d.ts +40 -0
- package/dist/api/client/request-builder.d.ts +27 -11
- package/dist/api/client/request-builder.js +15 -21
- package/dist/api/dsym.api.d.ts +1 -1
- package/dist/api/dsym.api.js +9 -7
- package/dist/api/proguard.api.d.ts +4 -0
- package/dist/api/proguard.api.js +52 -0
- package/dist/api/react-native.api.js +5 -4
- package/dist/api/source-maps.api.js +5 -4
- package/dist/cli/commands/proguard/index.d.ts +4 -0
- package/dist/cli/commands/proguard/index.js +6 -0
- package/dist/cli/commands/proguard/upload-proguard-command.d.ts +2 -0
- package/dist/cli/commands/proguard/upload-proguard-command.js +45 -0
- package/dist/cli/rum-cli.js +4 -1
- package/dist/config/config.d.ts +1 -0
- package/dist/config/config.js +2 -1
- package/dist/consts/consts.js +2 -3
- package/dist/models/proguard-commands.model.d.ts +11 -0
- package/dist/models/proguard-commands.model.js +3 -0
- package/dist/proto-models/com/coralogix/blobset/v2/entity_metadata.d.ts +18 -0
- package/dist/proto-models/com/coralogix/blobset/v2/entity_metadata.js +139 -5
- package/dist/proto-models/com/coralogix/blobset/v2/entity_type.d.ts +2 -1
- package/dist/proto-models/com/coralogix/blobset/v2/entity_type.js +7 -1
- package/dist/proto-models/com/coralogix/rum/v2/release_entity_metadata.d.ts +18 -0
- package/dist/proto-models/com/coralogix/rum/v2/release_entity_metadata.js +139 -5
- package/dist/proto-models/com/coralogix/rum/v2/release_entity_type.d.ts +2 -1
- package/dist/proto-models/com/coralogix/rum/v2/release_entity_type.js +7 -1
- package/dist/proto-models/com/coralogix/rum/v2/rum_sdk_versions.d.ts +1 -0
- package/dist/proto-models/com/coralogix/rum/v2/rum_sdk_versions.js +17 -3
- package/dist/proto-models/com/coralogix/rum/v2/rum_service.d.ts +93 -0
- package/dist/proto-models/com/coralogix/rum/v2/rum_service.js +300 -4
- package/dist/proto-models/com/coralogix/rum/v2/rum_settings_service.d.ts +9 -8
- package/dist/proto-models/com/coralogix/rum/v2/rum_settings_service.js +22 -7
- package/dist/services/dsym.service.d.ts +1 -1
- package/dist/services/dsym.service.js +3 -3
- package/dist/services/proguard.service.d.ts +3 -0
- package/dist/services/proguard.service.js +36 -0
- package/dist/services/react-native.service.js +2 -2
- package/dist/services/source-maps.service.js +2 -2
- package/dist/utils/file-processor.utils.d.ts +5 -8
- package/dist/utils/file-processor.utils.js +222 -60
- package/mapping.txt +160886 -0
- package/models/proguard-commands.model.ts +11 -0
- package/package.json +2 -3
- package/proto-models/com/coralogix/blobset/v2/entity_metadata.ts +160 -4
- package/proto-models/com/coralogix/blobset/v2/entity_type.ts +6 -0
- package/proto-models/com/coralogix/rum/v2/release_entity_metadata.ts +156 -0
- package/proto-models/com/coralogix/rum/v2/release_entity_type.ts +6 -0
- package/proto-models/com/coralogix/rum/v2/rum_sdk_versions.ts +17 -1
- package/proto-models/com/coralogix/rum/v2/rum_service.ts +347 -1
- package/proto-models/com/coralogix/rum/v2/rum_settings_service.ts +23 -6
- package/protofetch.lock +4 -4
- package/protofetch.toml +3 -3
- package/protos/com/coralogix/blobset/v2/entity_metadata.proto +9 -0
- package/protos/com/coralogix/blobset/v2/entity_type.proto +1 -0
- package/protos/com/coralogix/rum/v2/release_entity_metadata.proto +9 -0
- package/protos/com/coralogix/rum/v2/release_entity_type.proto +1 -0
- package/protos/com/coralogix/rum/v2/rum_sdk_versions.proto +1 -0
- package/protos/com/coralogix/rum/v2/rum_service.proto +23 -0
- package/protos/com/coralogix/rum/v2/rum_settings_service.proto +3 -1
- package/protoset.bin +0 -0
- package/services/dsym.service.ts +2 -2
- package/services/proguard.service.ts +33 -0
- package/services/react-native.service.ts +1 -1
- package/services/source-maps.service.ts +1 -1
- package/utils/file-processor.utils.ts +224 -70
|
@@ -1,4 +1,180 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
// import * as fs from 'fs';
|
|
3
|
+
// import { Dirent, promises as fsPromises } from 'fs';
|
|
4
|
+
// import * as path from 'path';
|
|
5
|
+
// import { consoleError } from './shared.utils';
|
|
6
|
+
// import { UploadBlobRequest } from '../proto-models/com/coralogix/blobset/v2/blobset_service';
|
|
7
|
+
// import { UploadSourceMapsRequest } from '../proto-models/com/coralogix/rum/v2/rum_source_map_service';
|
|
8
|
+
// import config from '../config';
|
|
9
|
+
// import pako from 'pako';
|
|
10
|
+
// import { FileMetadata } from '../proto-models/com/coralogix/rum/v2/file';
|
|
11
|
+
// import { RED_COLOR } from '../consts/consts';
|
|
12
|
+
// import * as zlib from 'node:zlib';
|
|
13
|
+
//
|
|
14
|
+
// const { readdir, stat, readFile } = fsPromises;
|
|
15
|
+
//
|
|
16
|
+
// export class FileProcessor {
|
|
17
|
+
// static async traverseDsymFolder(folderPath: string, request: UploadBlobRequest): Promise<void> {
|
|
18
|
+
// try {
|
|
19
|
+
// const processDsymFolder = async (dsymFolderPath: string) => {
|
|
20
|
+
// const dwarfPath = await FileProcessor.getDwarfPath(dsymFolderPath);
|
|
21
|
+
// const content = await readFile(dwarfPath);
|
|
22
|
+
// const compressedContent = zlib.gzipSync(content);
|
|
23
|
+
// request.data = new Uint8Array(compressedContent);
|
|
24
|
+
// };
|
|
25
|
+
//
|
|
26
|
+
// // Check if the folder itself is a .dSYM folder
|
|
27
|
+
// if (folderPath.endsWith('.dSYM')) {
|
|
28
|
+
// await processDsymFolder(folderPath);
|
|
29
|
+
// return;
|
|
30
|
+
// }
|
|
31
|
+
//
|
|
32
|
+
// const files: Dirent[] = await readdir(folderPath, { withFileTypes: true });
|
|
33
|
+
//
|
|
34
|
+
// for (const file of files) {
|
|
35
|
+
// // Skip .DS_Store and hidden files
|
|
36
|
+
// if (file.name === '.DS_Store' || file.name.startsWith('.')) {
|
|
37
|
+
// continue;
|
|
38
|
+
// }
|
|
39
|
+
//
|
|
40
|
+
// const filePath: string = path.join(folderPath, file.name);
|
|
41
|
+
//
|
|
42
|
+
// if (file.isDirectory()) {
|
|
43
|
+
// if (file.name.endsWith('.dSYM')) {
|
|
44
|
+
// await processDsymFolder(filePath);
|
|
45
|
+
// return;
|
|
46
|
+
// } else {
|
|
47
|
+
// await FileProcessor.traverseDsymFolder(filePath, request);
|
|
48
|
+
// }
|
|
49
|
+
// }
|
|
50
|
+
// }
|
|
51
|
+
// } catch (error) {
|
|
52
|
+
// consoleError(`Error traversing directory: ${JSON.stringify(error)}`);
|
|
53
|
+
// }
|
|
54
|
+
// }
|
|
55
|
+
//
|
|
56
|
+
// static async traverseSourceMapsFolder(
|
|
57
|
+
// folderPath: string,
|
|
58
|
+
// fileExtension: string,
|
|
59
|
+
// request: UploadBlobRequest,
|
|
60
|
+
// ): Promise<number | undefined> {
|
|
61
|
+
// try {
|
|
62
|
+
// const stats = await stat(folderPath);
|
|
63
|
+
//
|
|
64
|
+
// if (stats.isFile() && folderPath.endsWith(fileExtension)) {
|
|
65
|
+
// return await this.processSourceMapSingleFile(folderPath, fileExtension, request);
|
|
66
|
+
// } else if (stats.isDirectory()) {
|
|
67
|
+
// return await this.processSourceMapDirectory(folderPath, fileExtension, request);
|
|
68
|
+
// }
|
|
69
|
+
// } catch (error) {
|
|
70
|
+
// consoleError(`Error traversing directory: ${error}`);
|
|
71
|
+
// }
|
|
72
|
+
// }
|
|
73
|
+
//
|
|
74
|
+
// private static async processSourceMapSingleFile(filePath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {
|
|
75
|
+
// try {
|
|
76
|
+
// const content = await readFile(filePath);
|
|
77
|
+
//
|
|
78
|
+
// const { name } = path.parse(filePath);
|
|
79
|
+
//
|
|
80
|
+
// const fileData = { name, content: content.toString('base64') };
|
|
81
|
+
//
|
|
82
|
+
// const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify([fileData])));
|
|
83
|
+
//
|
|
84
|
+
// request.data = new Uint8Array(compressedContent);
|
|
85
|
+
//
|
|
86
|
+
// return 1;
|
|
87
|
+
// } catch (error) {
|
|
88
|
+
// consoleError(`Error processing file: ${error}`);
|
|
89
|
+
// throw error;
|
|
90
|
+
// }
|
|
91
|
+
// }
|
|
92
|
+
//
|
|
93
|
+
// private static async processSourceMapDirectory(folderPath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {
|
|
94
|
+
// const files = await readdir(folderPath, { withFileTypes: true });
|
|
95
|
+
// const fileDataList: { name: string; content: string }[] = [];
|
|
96
|
+
//
|
|
97
|
+
// for (const dirent of files) {
|
|
98
|
+
// const filePath = path.join(folderPath, dirent.name);
|
|
99
|
+
//
|
|
100
|
+
// if (dirent.isFile() && dirent.name.endsWith(fileExtension)) {
|
|
101
|
+
// const content = await readFile(filePath);
|
|
102
|
+
// const { name } = path.parse(filePath);
|
|
103
|
+
// fileDataList.push({ name, content: content.toString('base64') });
|
|
104
|
+
// } else if (dirent.isDirectory()) {
|
|
105
|
+
// await this.traverseSourceMapsFolder(filePath, fileExtension, request);
|
|
106
|
+
// }
|
|
107
|
+
// }
|
|
108
|
+
//
|
|
109
|
+
// const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify(fileDataList)));
|
|
110
|
+
//
|
|
111
|
+
// request.data = new Uint8Array(compressedContent);
|
|
112
|
+
//
|
|
113
|
+
// return fileDataList.length;
|
|
114
|
+
// }
|
|
115
|
+
//
|
|
116
|
+
// private static async getDwarfPath(dsymPath: string): Promise<string> {
|
|
117
|
+
// const dwarfPath: string = 'Contents/Resources/DWARF';
|
|
118
|
+
// const targetPath: string = path.join(dsymPath, dwarfPath);
|
|
119
|
+
// const files = fs.readdirSync(targetPath);
|
|
120
|
+
// if (files.length === 1) {
|
|
121
|
+
// return path.join(dsymPath, dwarfPath, files[0]);
|
|
122
|
+
// } else {
|
|
123
|
+
// consoleError(`Expected exactly one file in the directory, but found: ${files.length}`);
|
|
124
|
+
// return '';
|
|
125
|
+
// }
|
|
126
|
+
// }
|
|
127
|
+
//
|
|
128
|
+
// static async traverseSourceMapsFolderOld(folderPath: string, request: UploadSourceMapsRequest): Promise<void> {
|
|
129
|
+
// const stack: string[] = [folderPath];
|
|
130
|
+
//
|
|
131
|
+
// while (stack.length) {
|
|
132
|
+
// const currentPath = stack.pop();
|
|
133
|
+
// const entries: Dirent[] = await readdir(currentPath!, { withFileTypes: true });
|
|
134
|
+
//
|
|
135
|
+
// await Promise.all(
|
|
136
|
+
// entries.map(async (file: Dirent) => {
|
|
137
|
+
// const { name } = file;
|
|
138
|
+
// const entryPath = path.join(currentPath as string, name);
|
|
139
|
+
// const isCorrectSuffix = file.isFile() && name.endsWith('.js.map');
|
|
140
|
+
//
|
|
141
|
+
// if (file.isDirectory()) stack.push(entryPath);
|
|
142
|
+
// else if (isCorrectSuffix) request.files.push(await FileProcessor.getFileMetadata(entryPath));
|
|
143
|
+
// }),
|
|
144
|
+
// );
|
|
145
|
+
// }
|
|
146
|
+
// }
|
|
147
|
+
//
|
|
148
|
+
// static compressFileContentChunks(fileContent: Uint8Array): Uint8Array {
|
|
149
|
+
// const { chunkSize } = config.rumApi; // 512 KB chunk size
|
|
150
|
+
// const deflate = new pako.Deflate();
|
|
151
|
+
// const totalChunks = Math.ceil(fileContent.length / chunkSize);
|
|
152
|
+
//
|
|
153
|
+
// for (let i = 0; i < totalChunks; i++) {
|
|
154
|
+
// const start = i * chunkSize;
|
|
155
|
+
// const end = start + chunkSize;
|
|
156
|
+
// const chunk = fileContent.subarray(start, end);
|
|
157
|
+
//
|
|
158
|
+
// const isLastChunk = i === totalChunks - 1;
|
|
159
|
+
//
|
|
160
|
+
// deflate.push(chunk, isLastChunk);
|
|
161
|
+
// }
|
|
162
|
+
//
|
|
163
|
+
// const { err, msg, result } = deflate;
|
|
164
|
+
//
|
|
165
|
+
// if (err) console.error(RED_COLOR, msg);
|
|
166
|
+
//
|
|
167
|
+
// return result;
|
|
168
|
+
// }
|
|
169
|
+
//
|
|
170
|
+
// private static async getFileMetadata(filePath: string): Promise<FileMetadata> {
|
|
171
|
+
// const { size } = await stat(filePath);
|
|
172
|
+
// const { name } = path.parse(filePath);
|
|
173
|
+
// const content = await readFile(filePath);
|
|
174
|
+
//
|
|
175
|
+
// return { chunkName: name, size, content };
|
|
176
|
+
// }
|
|
177
|
+
// }
|
|
2
178
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
179
|
if (k2 === undefined) k2 = k;
|
|
4
180
|
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
@@ -31,22 +207,46 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
|
|
|
31
207
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
32
208
|
});
|
|
33
209
|
};
|
|
34
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
35
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
36
|
-
};
|
|
37
210
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
38
211
|
exports.FileProcessor = void 0;
|
|
39
212
|
const fs = __importStar(require("fs"));
|
|
40
213
|
const fs_1 = require("fs");
|
|
41
214
|
const path = __importStar(require("path"));
|
|
42
215
|
const shared_utils_1 = require("./shared.utils");
|
|
43
|
-
const config_1 = __importDefault(require("../config"));
|
|
44
|
-
const pako_1 = __importDefault(require("pako"));
|
|
45
|
-
const consts_1 = require("../consts/consts");
|
|
46
216
|
const zlib = __importStar(require("node:zlib"));
|
|
47
217
|
const { readdir, stat, readFile } = fs_1.promises;
|
|
48
218
|
class FileProcessor {
|
|
49
|
-
static
|
|
219
|
+
static traverseProguardFolder(proguardPath, request) {
|
|
220
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
221
|
+
try {
|
|
222
|
+
const processTxtFile = (proguardFile) => __awaiter(this, void 0, void 0, function* () {
|
|
223
|
+
const content = yield readFile(proguardFile);
|
|
224
|
+
const compressedContent = zlib.gzipSync(content);
|
|
225
|
+
request.data = new Uint8Array(compressedContent);
|
|
226
|
+
});
|
|
227
|
+
// Check if the folder itself is a .txt file
|
|
228
|
+
if (proguardPath.endsWith('.txt')) {
|
|
229
|
+
yield processTxtFile(proguardPath);
|
|
230
|
+
return;
|
|
231
|
+
}
|
|
232
|
+
const files = yield readdir(proguardPath, { withFileTypes: true });
|
|
233
|
+
for (const file of files) {
|
|
234
|
+
const filePath = path.join(proguardPath, file.name);
|
|
235
|
+
if (file.name.endsWith('.txt')) {
|
|
236
|
+
yield processTxtFile(filePath);
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
else {
|
|
240
|
+
yield FileProcessor.traverseProguardFolder(filePath, request);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
catch (error) {
|
|
245
|
+
(0, shared_utils_1.consoleError)(`Error traversing directory: ${JSON.stringify(error)}`);
|
|
246
|
+
}
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
static traverseDsymFolder(dsymPath, request) {
|
|
50
250
|
return __awaiter(this, void 0, void 0, function* () {
|
|
51
251
|
try {
|
|
52
252
|
const processDsymFolder = (dsymFolderPath) => __awaiter(this, void 0, void 0, function* () {
|
|
@@ -56,17 +256,17 @@ class FileProcessor {
|
|
|
56
256
|
request.data = new Uint8Array(compressedContent);
|
|
57
257
|
});
|
|
58
258
|
// Check if the folder itself is a .dSYM folder
|
|
59
|
-
if (
|
|
60
|
-
yield processDsymFolder(
|
|
259
|
+
if (dsymPath.endsWith('.dSYM')) {
|
|
260
|
+
yield processDsymFolder(dsymPath);
|
|
61
261
|
return;
|
|
62
262
|
}
|
|
63
|
-
const files = yield readdir(
|
|
263
|
+
const files = yield readdir(dsymPath, { withFileTypes: true });
|
|
64
264
|
for (const file of files) {
|
|
65
265
|
// Skip .DS_Store and hidden files
|
|
66
266
|
if (file.name === '.DS_Store' || file.name.startsWith('.')) {
|
|
67
267
|
continue;
|
|
68
268
|
}
|
|
69
|
-
const filePath = path.join(
|
|
269
|
+
const filePath = path.join(dsymPath, file.name);
|
|
70
270
|
if (file.isDirectory()) {
|
|
71
271
|
if (file.name.endsWith('.dSYM')) {
|
|
72
272
|
yield processDsymFolder(filePath);
|
|
@@ -83,15 +283,15 @@ class FileProcessor {
|
|
|
83
283
|
}
|
|
84
284
|
});
|
|
85
285
|
}
|
|
86
|
-
static
|
|
286
|
+
static traverse(folderPath, fileExtension, request) {
|
|
87
287
|
return __awaiter(this, void 0, void 0, function* () {
|
|
88
288
|
try {
|
|
89
289
|
const stats = yield stat(folderPath);
|
|
90
290
|
if (stats.isFile() && folderPath.endsWith(fileExtension)) {
|
|
91
|
-
return yield this.
|
|
291
|
+
return yield this.processSingleFile(folderPath, fileExtension, request);
|
|
92
292
|
}
|
|
93
293
|
else if (stats.isDirectory()) {
|
|
94
|
-
return yield this.
|
|
294
|
+
return yield this.processDirectory(folderPath, fileExtension, request);
|
|
95
295
|
}
|
|
96
296
|
}
|
|
97
297
|
catch (error) {
|
|
@@ -99,9 +299,13 @@ class FileProcessor {
|
|
|
99
299
|
}
|
|
100
300
|
});
|
|
101
301
|
}
|
|
102
|
-
static
|
|
302
|
+
static processSingleFile(filePath, fileExtension, request) {
|
|
103
303
|
return __awaiter(this, void 0, void 0, function* () {
|
|
104
304
|
try {
|
|
305
|
+
if (!filePath.endsWith(fileExtension)) {
|
|
306
|
+
(0, shared_utils_1.consoleError)(`File ${filePath} does not have the correct extension: ${fileExtension}`);
|
|
307
|
+
return 0;
|
|
308
|
+
}
|
|
105
309
|
const content = yield readFile(filePath);
|
|
106
310
|
const { name } = path.parse(filePath);
|
|
107
311
|
const fileData = { name, content: content.toString('base64') };
|
|
@@ -115,7 +319,7 @@ class FileProcessor {
|
|
|
115
319
|
}
|
|
116
320
|
});
|
|
117
321
|
}
|
|
118
|
-
static
|
|
322
|
+
static processDirectory(folderPath, fileExtension, request) {
|
|
119
323
|
return __awaiter(this, void 0, void 0, function* () {
|
|
120
324
|
const files = yield readdir(folderPath, { withFileTypes: true });
|
|
121
325
|
const fileDataList = [];
|
|
@@ -127,7 +331,7 @@ class FileProcessor {
|
|
|
127
331
|
fileDataList.push({ name, content: content.toString('base64') });
|
|
128
332
|
}
|
|
129
333
|
else if (dirent.isDirectory()) {
|
|
130
|
-
yield this.
|
|
334
|
+
yield this.traverse(filePath, fileExtension, request);
|
|
131
335
|
}
|
|
132
336
|
}
|
|
133
337
|
const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify(fileDataList)));
|
|
@@ -149,48 +353,6 @@ class FileProcessor {
|
|
|
149
353
|
}
|
|
150
354
|
});
|
|
151
355
|
}
|
|
152
|
-
static traverseSourceMapsFolderOld(folderPath, request) {
|
|
153
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
154
|
-
const stack = [folderPath];
|
|
155
|
-
while (stack.length) {
|
|
156
|
-
const currentPath = stack.pop();
|
|
157
|
-
const entries = yield readdir(currentPath, { withFileTypes: true });
|
|
158
|
-
yield Promise.all(entries.map((file) => __awaiter(this, void 0, void 0, function* () {
|
|
159
|
-
const { name } = file;
|
|
160
|
-
const entryPath = path.join(currentPath, name);
|
|
161
|
-
const isCorrectSuffix = file.isFile() && name.endsWith('.js.map');
|
|
162
|
-
if (file.isDirectory())
|
|
163
|
-
stack.push(entryPath);
|
|
164
|
-
else if (isCorrectSuffix)
|
|
165
|
-
request.files.push(yield FileProcessor.getFileMetadata(entryPath));
|
|
166
|
-
})));
|
|
167
|
-
}
|
|
168
|
-
});
|
|
169
|
-
}
|
|
170
|
-
static compressFileContentChunks(fileContent) {
|
|
171
|
-
const { chunkSize } = config_1.default.rumApi; // 512 KB chunk size
|
|
172
|
-
const deflate = new pako_1.default.Deflate();
|
|
173
|
-
const totalChunks = Math.ceil(fileContent.length / chunkSize);
|
|
174
|
-
for (let i = 0; i < totalChunks; i++) {
|
|
175
|
-
const start = i * chunkSize;
|
|
176
|
-
const end = start + chunkSize;
|
|
177
|
-
const chunk = fileContent.subarray(start, end);
|
|
178
|
-
const isLastChunk = i === totalChunks - 1;
|
|
179
|
-
deflate.push(chunk, isLastChunk);
|
|
180
|
-
}
|
|
181
|
-
const { err, msg, result } = deflate;
|
|
182
|
-
if (err)
|
|
183
|
-
console.error(consts_1.RED_COLOR, msg);
|
|
184
|
-
return result;
|
|
185
|
-
}
|
|
186
|
-
static getFileMetadata(filePath) {
|
|
187
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
188
|
-
const { size } = yield stat(filePath);
|
|
189
|
-
const { name } = path.parse(filePath);
|
|
190
|
-
const content = yield readFile(filePath);
|
|
191
|
-
return { chunkName: name, size, content };
|
|
192
|
-
});
|
|
193
|
-
}
|
|
194
356
|
}
|
|
195
357
|
exports.FileProcessor = FileProcessor;
|
|
196
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-processor.utils.js","sourceRoot":"","sources":["../../utils/file-processor.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport { Dirent, promises as fsPromises } from 'fs';\nimport * as path from 'path';\nimport { consoleError } from './shared.utils';\nimport { UploadBlobRequest } from '../proto-models/com/coralogix/blobset/v2/blobset_service';\nimport { UploadSourceMapsRequest } from '../proto-models/com/coralogix/rum/v2/rum_source_map_service';\nimport config from '../config';\nimport pako from 'pako';\nimport { FileMetadata } from '../proto-models/com/coralogix/rum/v2/file';\nimport { RED_COLOR } from '../consts/consts';\nimport * as zlib from 'node:zlib';\n\nconst { readdir, stat, readFile } = fsPromises;\n\nexport class FileProcessor {\n  static async traverseDsymFolder(folderPath: string, request: UploadBlobRequest): Promise<void> {\n    try {\n      const processDsymFolder = async (dsymFolderPath: string) => {\n        const dwarfPath = await FileProcessor.getDwarfPath(dsymFolderPath);\n        const content = await readFile(dwarfPath);\n        const compressedContent = zlib.gzipSync(content);\n        request.data = new Uint8Array(compressedContent);\n      };\n\n      // Check if the folder itself is a .dSYM folder\n      if (folderPath.endsWith('.dSYM')) {\n        await processDsymFolder(folderPath);\n        return;\n      }\n\n      const files: Dirent[] = await readdir(folderPath, { withFileTypes: true });\n\n      for (const file of files) {\n        // Skip .DS_Store and hidden files\n        if (file.name === '.DS_Store' || file.name.startsWith('.')) {\n          continue;\n        }\n\n        const filePath: string = path.join(folderPath, file.name);\n\n        if (file.isDirectory()) {\n          if (file.name.endsWith('.dSYM')) {\n            await processDsymFolder(filePath);\n            return;\n          } else {\n            await FileProcessor.traverseDsymFolder(filePath, request);\n          }\n        }\n      }\n    } catch (error) {\n      consoleError(`Error traversing directory: ${JSON.stringify(error)}`);\n    }\n  }\n\n  static async traverseSourceMapsFolder(\n    folderPath: string,\n    fileExtension: string,\n    request: UploadBlobRequest,\n  ): Promise<number | undefined> {\n    try {\n      const stats = await stat(folderPath);\n\n      if (stats.isFile() && folderPath.endsWith(fileExtension)) {\n        return await this.processSourceMapSingleFile(folderPath, fileExtension, request);\n      } else if (stats.isDirectory()) {\n        return await this.processSourceMapDirectory(folderPath, fileExtension, request);\n      }\n    } catch (error) {\n      consoleError(`Error traversing directory: ${error}`);\n    }\n  }\n\n  private static async processSourceMapSingleFile(filePath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n    try {\n      const content = await readFile(filePath);\n\n      const { name } = path.parse(filePath);\n\n      const fileData = { name, content: content.toString('base64') };\n\n      const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify([fileData])));\n\n      request.data = new Uint8Array(compressedContent);\n\n      return 1;\n    } catch (error) {\n      consoleError(`Error processing file: ${error}`);\n      throw error;\n    }\n  }\n\n  private static async processSourceMapDirectory(folderPath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n    const files = await readdir(folderPath, { withFileTypes: true });\n    const fileDataList: { name: string; content: string }[] = [];\n\n    for (const dirent of files) {\n      const filePath = path.join(folderPath, dirent.name);\n\n      if (dirent.isFile() && dirent.name.endsWith(fileExtension)) {\n        const content = await readFile(filePath);\n        const { name } = path.parse(filePath);\n        fileDataList.push({ name, content: content.toString('base64') });\n      } else if (dirent.isDirectory()) {\n        await this.traverseSourceMapsFolder(filePath, fileExtension, request);\n      }\n    }\n\n    const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify(fileDataList)));\n\n    request.data = new Uint8Array(compressedContent);\n\n    return fileDataList.length;\n  }\n\n  private static async getDwarfPath(dsymPath: string): Promise<string> {\n    const dwarfPath: string = 'Contents/Resources/DWARF';\n    const targetPath: string = path.join(dsymPath, dwarfPath);\n    const files = fs.readdirSync(targetPath);\n    if (files.length === 1) {\n      return path.join(dsymPath, dwarfPath, files[0]);\n    } else {\n      consoleError(`Expected exactly one file in the directory, but found: ${files.length}`);\n      return '';\n    }\n  }\n\n  static async traverseSourceMapsFolderOld(folderPath: string, request: UploadSourceMapsRequest): Promise<void> {\n    const stack: string[] = [folderPath];\n\n    while (stack.length) {\n      const currentPath = stack.pop();\n      const entries: Dirent[] = await readdir(currentPath!, { withFileTypes: true });\n\n      await Promise.all(\n        entries.map(async (file: Dirent) => {\n          const { name } = file;\n          const entryPath = path.join(currentPath as string, name);\n          const isCorrectSuffix = file.isFile() && name.endsWith('.js.map');\n\n          if (file.isDirectory()) stack.push(entryPath);\n          else if (isCorrectSuffix) request.files.push(await FileProcessor.getFileMetadata(entryPath));\n        }),\n      );\n    }\n  }\n\n  static compressFileContentChunks(fileContent: Uint8Array): Uint8Array {\n    const { chunkSize } = config.rumApi; // 512 KB chunk size\n    const deflate = new pako.Deflate();\n    const totalChunks = Math.ceil(fileContent.length / chunkSize);\n\n    for (let i = 0; i < totalChunks; i++) {\n      const start = i * chunkSize;\n      const end = start + chunkSize;\n      const chunk = fileContent.subarray(start, end);\n\n      const isLastChunk = i === totalChunks - 1;\n\n      deflate.push(chunk, isLastChunk);\n    }\n\n    const { err, msg, result } = deflate;\n\n    if (err) console.error(RED_COLOR, msg);\n\n    return result;\n  }\n\n  private static async getFileMetadata(filePath: string): Promise<FileMetadata> {\n    const { size } = await stat(filePath);\n    const { name } = path.parse(filePath);\n    const content = await readFile(filePath);\n\n    return { chunkName: name, size, content };\n  }\n}\n"]}
|
|
358
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-processor.utils.js","sourceRoot":"","sources":["../../utils/file-processor.utils.ts"],"names":[],"mappings":"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","sourcesContent":["// import * as fs from 'fs';\n// import { Dirent, promises as fsPromises } from 'fs';\n// import * as path from 'path';\n// import { consoleError } from './shared.utils';\n// import { UploadBlobRequest } from '../proto-models/com/coralogix/blobset/v2/blobset_service';\n// import { UploadSourceMapsRequest } from '../proto-models/com/coralogix/rum/v2/rum_source_map_service';\n// import config from '../config';\n// import pako from 'pako';\n// import { FileMetadata } from '../proto-models/com/coralogix/rum/v2/file';\n// import { RED_COLOR } from '../consts/consts';\n// import * as zlib from 'node:zlib';\n//\n// const { readdir, stat, readFile } = fsPromises;\n//\n// export class FileProcessor {\n//   static async traverseDsymFolder(folderPath: string, request: UploadBlobRequest): Promise<void> {\n//     try {\n//       const processDsymFolder = async (dsymFolderPath: string) => {\n//         const dwarfPath = await FileProcessor.getDwarfPath(dsymFolderPath);\n//         const content = await readFile(dwarfPath);\n//         const compressedContent = zlib.gzipSync(content);\n//         request.data = new Uint8Array(compressedContent);\n//       };\n//\n//       // Check if the folder itself is a .dSYM folder\n//       if (folderPath.endsWith('.dSYM')) {\n//         await processDsymFolder(folderPath);\n//         return;\n//       }\n//\n//       const files: Dirent[] = await readdir(folderPath, { withFileTypes: true });\n//\n//       for (const file of files) {\n//         // Skip .DS_Store and hidden files\n//         if (file.name === '.DS_Store' || file.name.startsWith('.')) {\n//           continue;\n//         }\n//\n//         const filePath: string = path.join(folderPath, file.name);\n//\n//         if (file.isDirectory()) {\n//           if (file.name.endsWith('.dSYM')) {\n//             await processDsymFolder(filePath);\n//             return;\n//           } else {\n//             await FileProcessor.traverseDsymFolder(filePath, request);\n//           }\n//         }\n//       }\n//     } catch (error) {\n//       consoleError(`Error traversing directory: ${JSON.stringify(error)}`);\n//     }\n//   }\n//\n//   static async traverseSourceMapsFolder(\n//     folderPath: string,\n//     fileExtension: string,\n//     request: UploadBlobRequest,\n//   ): Promise<number | undefined> {\n//     try {\n//       const stats = await stat(folderPath);\n//\n//       if (stats.isFile() && folderPath.endsWith(fileExtension)) {\n//         return await this.processSourceMapSingleFile(folderPath, fileExtension, request);\n//       } else if (stats.isDirectory()) {\n//         return await this.processSourceMapDirectory(folderPath, fileExtension, request);\n//       }\n//     } catch (error) {\n//       consoleError(`Error traversing directory: ${error}`);\n//     }\n//   }\n//\n//   private static async processSourceMapSingleFile(filePath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n//     try {\n//       const content = await readFile(filePath);\n//\n//       const { name } = path.parse(filePath);\n//\n//       const fileData = { name, content: content.toString('base64') };\n//\n//       const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify([fileData])));\n//\n//       request.data = new Uint8Array(compressedContent);\n//\n//       return 1;\n//     } catch (error) {\n//       consoleError(`Error processing file: ${error}`);\n//       throw error;\n//     }\n//   }\n//\n//   private static async processSourceMapDirectory(folderPath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n//     const files = await readdir(folderPath, { withFileTypes: true });\n//     const fileDataList: { name: string; content: string }[] = [];\n//\n//     for (const dirent of files) {\n//       const filePath = path.join(folderPath, dirent.name);\n//\n//       if (dirent.isFile() && dirent.name.endsWith(fileExtension)) {\n//         const content = await readFile(filePath);\n//         const { name } = path.parse(filePath);\n//         fileDataList.push({ name, content: content.toString('base64') });\n//       } else if (dirent.isDirectory()) {\n//         await this.traverseSourceMapsFolder(filePath, fileExtension, request);\n//       }\n//     }\n//\n//     const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify(fileDataList)));\n//\n//     request.data = new Uint8Array(compressedContent);\n//\n//     return fileDataList.length;\n//   }\n//\n//   private static async getDwarfPath(dsymPath: string): Promise<string> {\n//     const dwarfPath: string = 'Contents/Resources/DWARF';\n//     const targetPath: string = path.join(dsymPath, dwarfPath);\n//     const files = fs.readdirSync(targetPath);\n//     if (files.length === 1) {\n//       return path.join(dsymPath, dwarfPath, files[0]);\n//     } else {\n//       consoleError(`Expected exactly one file in the directory, but found: ${files.length}`);\n//       return '';\n//     }\n//   }\n//\n//   static async traverseSourceMapsFolderOld(folderPath: string, request: UploadSourceMapsRequest): Promise<void> {\n//     const stack: string[] = [folderPath];\n//\n//     while (stack.length) {\n//       const currentPath = stack.pop();\n//       const entries: Dirent[] = await readdir(currentPath!, { withFileTypes: true });\n//\n//       await Promise.all(\n//         entries.map(async (file: Dirent) => {\n//           const { name } = file;\n//           const entryPath = path.join(currentPath as string, name);\n//           const isCorrectSuffix = file.isFile() && name.endsWith('.js.map');\n//\n//           if (file.isDirectory()) stack.push(entryPath);\n//           else if (isCorrectSuffix) request.files.push(await FileProcessor.getFileMetadata(entryPath));\n//         }),\n//       );\n//     }\n//   }\n//\n//   static compressFileContentChunks(fileContent: Uint8Array): Uint8Array {\n//     const { chunkSize } = config.rumApi; // 512 KB chunk size\n//     const deflate = new pako.Deflate();\n//     const totalChunks = Math.ceil(fileContent.length / chunkSize);\n//\n//     for (let i = 0; i < totalChunks; i++) {\n//       const start = i * chunkSize;\n//       const end = start + chunkSize;\n//       const chunk = fileContent.subarray(start, end);\n//\n//       const isLastChunk = i === totalChunks - 1;\n//\n//       deflate.push(chunk, isLastChunk);\n//     }\n//\n//     const { err, msg, result } = deflate;\n//\n//     if (err) console.error(RED_COLOR, msg);\n//\n//     return result;\n//   }\n//\n//   private static async getFileMetadata(filePath: string): Promise<FileMetadata> {\n//     const { size } = await stat(filePath);\n//     const { name } = path.parse(filePath);\n//     const content = await readFile(filePath);\n//\n//     return { chunkName: name, size, content };\n//   }\n// }\n\nimport * as fs from 'fs';\nimport { Dirent, promises as fsPromises } from 'fs';\nimport * as path from 'path';\nimport { consoleError } from './shared.utils';\nimport { UploadBlobRequest } from '../proto-models/com/coralogix/blobset/v2/blobset_service';\nimport * as zlib from 'node:zlib';\n\nconst { readdir, stat, readFile } = fsPromises;\n\nexport class FileProcessor {\n  static async traverseProguardFolder(proguardPath: string, request: UploadBlobRequest): Promise<void> {\n    try {\n      const processTxtFile = async (proguardFile: string) => {\n        const content = await readFile(proguardFile);\n        const compressedContent = zlib.gzipSync(content);\n        request.data = new Uint8Array(compressedContent);\n      };\n\n      // Check if the folder itself is a .txt file\n      if (proguardPath.endsWith('.txt')) {\n        await processTxtFile(proguardPath);\n        return;\n      }\n\n      const files: Dirent[] = await readdir(proguardPath, { withFileTypes: true });\n\n      for (const file of files) {\n        const filePath: string = path.join(proguardPath, file.name);\n\n        if (file.name.endsWith('.txt')) {\n          await processTxtFile(filePath);\n          return;\n        } else {\n          await FileProcessor.traverseProguardFolder(filePath, request);\n        }\n      }\n    } catch (error) {\n      consoleError(`Error traversing directory: ${JSON.stringify(error)}`);\n    }\n  }\n\n  static async traverseDsymFolder(dsymPath: string, request: UploadBlobRequest): Promise<void> {\n    try {\n      const processDsymFolder = async (dsymFolderPath: string) => {\n        const dwarfPath = await FileProcessor.getDwarfPath(dsymFolderPath);\n        const content = await readFile(dwarfPath);\n        const compressedContent = zlib.gzipSync(content);\n        request.data = new Uint8Array(compressedContent);\n      };\n\n      // Check if the folder itself is a .dSYM folder\n      if (dsymPath.endsWith('.dSYM')) {\n        await processDsymFolder(dsymPath);\n        return;\n      }\n\n      const files: Dirent[] = await readdir(dsymPath, { withFileTypes: true });\n\n      for (const file of files) {\n        // Skip .DS_Store and hidden files\n        if (file.name === '.DS_Store' || file.name.startsWith('.')) {\n          continue;\n        }\n\n        const filePath: string = path.join(dsymPath, file.name);\n\n        if (file.isDirectory()) {\n          if (file.name.endsWith('.dSYM')) {\n            await processDsymFolder(filePath);\n            return;\n          } else {\n            await FileProcessor.traverseDsymFolder(filePath, request);\n          }\n        }\n      }\n    } catch (error) {\n      consoleError(`Error traversing directory: ${JSON.stringify(error)}`);\n    }\n  }\n\n  static async traverse(folderPath: string, fileExtension: string, request: UploadBlobRequest): Promise<number | undefined> {\n    try {\n      const stats = await stat(folderPath);\n\n      if (stats.isFile() && folderPath.endsWith(fileExtension)) {\n        return await this.processSingleFile(folderPath, fileExtension, request);\n      } else if (stats.isDirectory()) {\n        return await this.processDirectory(folderPath, fileExtension, request);\n      }\n    } catch (error) {\n      consoleError(`Error traversing directory: ${error}`);\n    }\n  }\n\n  private static async processSingleFile(filePath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n    try {\n      if (!filePath.endsWith(fileExtension)) {\n        consoleError(`File ${filePath} does not have the correct extension: ${fileExtension}`);\n        return 0;\n      }\n\n      const content = await readFile(filePath);\n\n      const { name } = path.parse(filePath);\n\n      const fileData = { name, content: content.toString('base64') };\n\n      const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify([fileData])));\n\n      request.data = new Uint8Array(compressedContent);\n\n      return 1;\n    } catch (error) {\n      consoleError(`Error processing file: ${error}`);\n      throw error;\n    }\n  }\n\n  private static async processDirectory(folderPath: string, fileExtension: string, request: UploadBlobRequest): Promise<number> {\n    const files = await readdir(folderPath, { withFileTypes: true });\n    const fileDataList: { name: string; content: string }[] = [];\n\n    for (const dirent of files) {\n      const filePath = path.join(folderPath, dirent.name);\n\n      if (dirent.isFile() && dirent.name.endsWith(fileExtension)) {\n        const content = await readFile(filePath);\n        const { name } = path.parse(filePath);\n        fileDataList.push({ name, content: content.toString('base64') });\n      } else if (dirent.isDirectory()) {\n        await this.traverse(filePath, fileExtension, request);\n      }\n    }\n\n    const compressedContent = zlib.gzipSync(Buffer.from(JSON.stringify(fileDataList)));\n\n    request.data = new Uint8Array(compressedContent);\n\n    return fileDataList.length;\n  }\n\n  private static async getDwarfPath(dsymPath: string): Promise<string> {\n    const dwarfPath: string = 'Contents/Resources/DWARF';\n    const targetPath: string = path.join(dsymPath, dwarfPath);\n    const files = fs.readdirSync(targetPath);\n    if (files.length === 1) {\n      return path.join(dsymPath, dwarfPath, files[0]);\n    } else {\n      consoleError(`Expected exactly one file in the directory, but found: ${files.length}`);\n      return '';\n    }\n  }\n}\n"]}
|