file.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  1. const fs = require('fs');
  2. const path = require('path');
  3. const {logger} = require("./logger");
  4. const {rustConfig} = require("../config");
  5. const http = require("./http");
  6. const readline = require('readline');
  7. // 检查目录是否存在,不存在新建
  8. function checkPathSync(filePath) {
  9. // 将相对路径转换为绝对路径
  10. const directoryPath = path.resolve(filePath);
  11. // 同步检查目录是否存在
  12. if (!fs.existsSync(directoryPath)) {
  13. // 目录不存在,需要创建它
  14. logger.info('目录不存在,正在创建...');
  15. try {
  16. // 同步创建目录
  17. fs.mkdirSync(directoryPath, {recursive: true});
  18. logger.info('目录已成功创建');
  19. } catch (error) {
  20. // 创建目录失败
  21. logger.error('创建目录失败:', error);
  22. return false
  23. }
  24. } else {
  25. // 目录已存在
  26. // logger.info('目录已存在');
  27. }
  28. return true
  29. }
  30. // 检查文件是否存在
  31. function checkFilePath(filePath) {
  32. // 同步检查文件是否存在
  33. const directoryPath = path.resolve(filePath);
  34. if (fs.existsSync(directoryPath)) {
  35. logger.info('文件存在:' + filePath);
  36. return true
  37. } else {
  38. logger.error('文件不存在:' + filePath);
  39. return false
  40. }
  41. }
  42. // 创建文件
  43. function writeFile(filePath, text, breakFun) {
  44. fs.writeFile(filePath, text, (err) => {
  45. if (err) {
  46. logger.error('写入文件时发生错误:', err);
  47. breakFun(err)
  48. } else {
  49. logger.log('文件已成功写入!');
  50. breakFun(null, true)
  51. }
  52. });
  53. }
  54. // 同步复制
  55. function copyFileSync(source, destination) {
  56. try {
  57. // 同步复制文件
  58. fs.copyFileSync(source, destination);
  59. // logger.info("成功");
  60. return true;
  61. } catch (err) {
  62. return false;
  63. // logger.info("失败");
  64. }
  65. }
  66. // // 读取日志(倒叙)指定行数
  67. // function readLastNLines(filePath, n) {
  68. // const content = fs.readFileSync(filePath, 'utf8');
  69. // const lines = content.trim().split('\n');
  70. // const lastNLines = lines.slice(-n).join('\n');
  71. //
  72. // // 此处处理或输出最后n行的内容
  73. // // console.log(lastNLines);
  74. // return lastNLines.split("\n").reverse();
  75. // }
  76. // 下载文件
  77. function dowFile(url, fileName, dowPath, headers) {
  78. return http.dowFile(url, fileName, dowPath, headers)
  79. }
  80. // 删除文件
  81. function delFile() {
  82. }
  83. /***********************************/
  84. function getLastFile(dirPath, number, callback) {
  85. fs.readdir(dirPath, async (err, files) => {
  86. // logger.info(`---------------------4`)
  87. if (err) {
  88. logger.error(`无法列出目录。`, err);
  89. // process.exit(1);
  90. } else {
  91. let fileList = [];
  92. //提示日志默认是根据日志生成的,拿到的files 也是 按照生成日志排序的,所以
  93. // 按照倒叙获取,如果一个文件内容满足num,就不在继续查询,如果不够,继续获取文件
  94. for (file of files) {
  95. const filePath = path.join(dirPath, file);
  96. // logger.info("所有文件:",filePath)
  97. const stats = await fs.promises.stat(filePath);
  98. if (stats.isFile()) {
  99. fileList.push({name: file, time: stats.birthtime.getTime()});
  100. logger.info("----------------所有文件:" + filePath + "时间:" + stats.birthtime.getTime())
  101. }
  102. }
  103. fileList.sort((a, b) => b.time - a.time);
  104. for (a of fileList) {
  105. const filePath = path.join(dirPath, a.name);
  106. logger.info("排序之后的顺序:" + filePath + "\ttime:" + a.time)
  107. }
  108. let lastFileList = fileList.slice(0, number != -1 ? number : fileList.length);
  109. let lastFileNameList = lastFileList.map((item) => item.name);
  110. callback(lastFileNameList.reverse(), lastFileList.reverse());
  111. }
  112. });
  113. }
  114. // 读取日志(倒叙)指定行数
  115. async function readLastNLines(dirPath, filePathList, n) {
  116. try {
  117. const fileList = [];
  118. var count = 0;
  119. for (var i = filePathList.length - 1; i >= 0; i--) {
  120. var item = filePathList[i];
  121. const filepath = `${dirPath}/${item}`;
  122. logger.info("日志文件:" + filepath)
  123. //老版本 加载整个文件
  124. // const content = fs.readFileSync(`${filepath}`, "utf8");
  125. // let lines = content.trim().split("\n");
  126. //新版本:流式读取和逐行处理
  127. const lines = await readLinesFromEnd(filepath, n - count);
  128. // logger.info("日:" + lines)
  129. fileList.push({
  130. filePath: item,
  131. lines: lines,
  132. })
  133. count += lines.length;
  134. if (count >= n) {
  135. break
  136. }
  137. }
  138. fileList.reverse();
  139. const allFile = [].concat(...fileList.map((item) => item.lines));
  140. const lastNLines = allFile.slice(-n).reverse();
  141. return lastNLines;
  142. } catch (e) {
  143. logger.info('读取文件发生异常咯~~2', e);
  144. }
  145. }
  146. async function getRecentLogs(dirPath, logFiles, requiredLogs = 100) {
  147. let logs = [];
  148. try {
  149. let logsSize = requiredLogs
  150. for (const file of logFiles) {
  151. let log = [];
  152. // 创建文件流
  153. const filepath = `${dirPath}/${file}`;
  154. const fileStream = fs.createReadStream(filepath);
  155. const rl = readline.createInterface({
  156. input: fileStream,
  157. crlfDelay: Infinity
  158. });
  159. for await (const line of rl) {
  160. log.push(line);
  161. }
  162. for (const l of log.reverse()) {
  163. logs.push(l);
  164. if (logs.length >= requiredLogs) {
  165. break
  166. }
  167. }
  168. }
  169. } catch (e) {
  170. logger.info('读取文件发生异常咯~~~', e.message);
  171. }
  172. return logs; // 如果日志不够500条,返回现有日志
  173. }
  174. async function getLatestLogEntries(dirPath, logFiles, requiredLogs = 100) {
  175. const MAX_LOGS = requiredLogs;
  176. let logs = [];
  177. try {
  178. for (const file of logFiles) {
  179. // 创建文件流
  180. const filepath = `${dirPath}/${file}`;
  181. logger.info('文件~~~', filepath);
  182. // const readStream = fs.createReadStream(filepath, { highWaterMark: 2 * 1024 * 1024 }); // 每次读取10MB
  183. // let buffer = '';
  184. //
  185. // logs = await new Promise((resolve,reject) =>{
  186. // // 逐块读取并处理数据
  187. // readStream.on('data', (chunk) => {
  188. // buffer += chunk.toString(); // 将块数据转成字符串
  189. // const lines = buffer.split('\n'); // 按行拆分
  190. //
  191. // // 处理新行
  192. // for (let i = 0; i < lines.length - 1; i++) {
  193. // logs.unshift(lines[i]); // 添加到数组前面以实现倒序
  194. // // logs.push(lines[i]); // 添加到数组前面以实现倒序
  195. // logger.info( lines[i]);
  196. //
  197. // if (logs.length >= MAX_LOGS) break; // 达到所需条数则停止
  198. // }
  199. //
  200. // // 保留未处理的最后一行(如果是半行,继续留在缓冲区)
  201. // buffer = lines[lines.length - 1];
  202. // });
  203. //
  204. // // 读取结束后处理剩余缓冲区
  205. // readStream.on('end', () => {
  206. // if (buffer) {
  207. // logs.unshift(buffer); // 加入最后的未处理行
  208. // }
  209. // resolve(logs); // 完成时解析 Promise
  210. // });
  211. //
  212. // readStream.on('error', (err) => {
  213. // reject(err); // 如果出错,拒绝 Promise
  214. // });
  215. //
  216. // // 确保在获取到足够记录时停止读取
  217. // readStream.on('close', () => {
  218. // if (logs.length >= MAX_LOGS) {
  219. // readStream.destroy(); // 如果达到了最大记录数,停止读取流
  220. // }else{
  221. // logger.info('----继续读取~~',logs.length);
  222. // }
  223. // });
  224. // })
  225. // // logger.info( logs);
  226. // // 如果已达到所需行数,停止继续读取文件
  227. // if (logs.length >= MAX_LOGS) {
  228. // break; // 退出循环
  229. // }
  230. // 文件信息
  231. const stats = fs.statSync(filepath); // 获取文件信息
  232. let position = stats.size; // 从文件末尾开始
  233. const chunkSize = 1 * 1024 * 1024; // 每次读取1MB
  234. // 文件读取方式
  235. while (position > 0 && logs.length < MAX_LOGS) {
  236. const readSize = Math.min(chunkSize, position); // 确保不读取超过文件的大小
  237. const readStream = fs.createReadStream(filepath, {
  238. start: position - readSize, // 从当前位置向上读取
  239. end: position - 1, // 读取到当前位置
  240. highWaterMark: readSize // 一次性读取指定大小
  241. });
  242. // 缓存
  243. let buffer = '';
  244. await new Promise((resolve, reject) => {
  245. readStream.on('data', (chunk) => {
  246. buffer = chunk.toString() + buffer; // 新数据放在前面
  247. });
  248. readStream.on('end', () => {
  249. // 按行处理数据
  250. const lines = buffer.split('\n');
  251. while (lines.length > 0) {
  252. const line = lines.pop(); // 取出最新的一行
  253. if (line) {
  254. logs.unshift(line); // 添加到日志数组开头
  255. // logger.info(line);
  256. if (logs.length >= requiredLogs) break; // 达到要求的条数
  257. }
  258. }
  259. resolve();
  260. });
  261. readStream.on('error', (err) => {
  262. console.error('读取文件发生错误:', err);
  263. reject(err); // 确保不会因错误而卡住
  264. });
  265. readStream.on('close', () => {
  266. if (logs.length >= MAX_LOGS) {
  267. readStream.destroy(); // 如果达到了最大记录数,停止读取流
  268. } else {
  269. logger.info('----继续读取~~', logs.length);
  270. }
  271. });
  272. });
  273. position -= readSize; // 递减位置
  274. }
  275. // 如果已达到所需行数,停止继续读取文件
  276. if (logs.length >= MAX_LOGS) {
  277. break; // 退出循环
  278. }
  279. }
  280. } catch (e) {
  281. logger.info('读取文件发生异常咯~~~', e.message);
  282. }
  283. // logger.info( logs);
  284. return logs.reverse(); // 如果日志不够500条,返回现有日志
  285. }
  286. async function readLinesFromEnd(filePath, maxLines) {
  287. const lines = [];
  288. const fileSize = fs.statSync(filePath).size;
  289. const chunkSize = 1024 * 10 * 5; // 每次读取的块大小
  290. try {
  291. if (fileSize < chunkSize) {
  292. const fileStream = fs.createReadStream(filePath, {
  293. start: Math.max(0, 0),
  294. end: fileSize
  295. });
  296. let buffer = '';
  297. for await (const chunk of fileStream) {
  298. buffer = chunk + buffer;
  299. let lineEndIndex = buffer.length;
  300. while (lineEndIndex >= 0 && lines.length < maxLines) {
  301. const lineStartIndex = buffer.lastIndexOf('\n', lineEndIndex - 1);
  302. if (lineStartIndex === -1) {
  303. break;
  304. }
  305. const line = buffer.substring(lineStartIndex + 1, lineEndIndex).trim();
  306. if (line) {
  307. lines.push(line);
  308. }
  309. lineEndIndex = lineStartIndex;
  310. }
  311. if (lines.length >= maxLines) {
  312. break;
  313. }
  314. }
  315. } else {
  316. let forNum = (fileSize / chunkSize) + 1;
  317. for (let i = 1; i <= forNum; i++) {
  318. let position = fileSize - chunkSize * i;
  319. let z = position + chunkSize
  320. const fileStream = fs.createReadStream(filePath, {
  321. start: Math.max(position, 0),
  322. end: z
  323. });
  324. let buffer = '';
  325. for await (const chunk of fileStream) {
  326. buffer = chunk + buffer;
  327. let lineEndIndex = buffer.length;
  328. while (lineEndIndex >= 0 && lines.length < maxLines) {
  329. const lineStartIndex = buffer.lastIndexOf('\n', lineEndIndex - 1);
  330. if (lineStartIndex === -1) {
  331. break;
  332. }
  333. const line = buffer.substring(lineStartIndex + 1, lineEndIndex).trim();
  334. if (line) {
  335. lines.push(line);
  336. }
  337. lineEndIndex = lineStartIndex;
  338. }
  339. if (lines.length >= maxLines) {
  340. break;
  341. }
  342. position -= chunkSize;
  343. if (position < 0) {
  344. position = 0;
  345. }
  346. }
  347. }
  348. }
  349. } catch (e) {
  350. logger.info('读取文件发生异常咯~~~1', e.message);
  351. }
  352. return lines.reverse();
  353. }
  354. async function readLines(filePath, maxLines) {
  355. const fileStream = fs.createReadStream(filePath);
  356. const rl = readline.createInterface({
  357. input: fileStream,
  358. crlfDelay: Infinity
  359. });
  360. const lines = [];
  361. for await (const line of rl) {
  362. lines.push(line);
  363. if (lines.length >= maxLines) {
  364. break;
  365. }
  366. }
  367. rl.close();
  368. fileStream.close();
  369. return lines;
  370. }
  371. module.exports = {
  372. dowFile,
  373. checkFilePath,
  374. checkPathSync,
  375. writeFile,
  376. copyFileSync,
  377. readLastNLines,
  378. getLatestLogEntries,
  379. getRecentLogs,
  380. getLastFile
  381. }