GitCodeStats.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. const API = require("../../lib/API")
  2. const AccessControl = require("../../lib/AccessControl")
  3. const { BaseStdResponse } = require("../../BaseStdResponse")
  4. const db = require("../../plugin/DataBase/db")
  5. const simpleGit = require('simple-git')
  6. const fs = require("fs")
  7. const path = require("path")
  8. class gitCodeStats extends API {
  9. constructor() {
  10. super()
  11. this.setMethod("GET")
  12. this.setPath("/Repos/CodeStats")
  13. }
  14. // 定义支持的语言后缀
  15. languageExtensions = {
  16. "JavaScript": [".js", ".jsx", ".mjs", ".cjs"],
  17. "TypeScript": [".ts", ".tsx"],
  18. "Python": [".py"],
  19. "Java": [".java"],
  20. "C": [".c"],
  21. "C++": [".cpp", ".cc", ".cxx", ".h", ".hpp"],
  22. "C#": [".cs"],
  23. "Go": [".go"],
  24. "PHP": [".php"],
  25. "HTML": [".html", ".htm"],
  26. "CSS": [".css"],
  27. "SCSS": [".scss", ".sass"],
  28. "JSON": [".json"],
  29. "YAML": [".yml", ".yaml"],
  30. "Markdown": [".md"],
  31. "Shell": [".sh"],
  32. "Bash": [".bash"],
  33. "PowerShell": [".ps1"],
  34. "Ruby": [".rb"],
  35. "Vue": [".vue"], // Vue文件
  36. "LESS": [".less"], // LESS样式文件
  37. "Sass": [".sass"], // Sass文件
  38. "Haml": [".haml"], // Haml文件
  39. "Elixir": [".ex", ".exs"], // Elixir语言文件
  40. "Erlang": [".erl", ".hrl"], // Erlang语言文件
  41. "Swift": [".swift"], // Swift语言文件
  42. "Kotlin": [".kt", ".kts"], // Kotlin语言文件
  43. "Rust": [".rs"], // Rust语言文件
  44. "Dart": [".dart"], // Dart语言文件
  45. "Lua": [".lua"], // Lua语言文件
  46. "R": [".r"], // R语言文件
  47. "Perl": [".pl"], // Perl语言文件
  48. "SQL": [".sql"], // SQL文件
  49. "Dockerfile": [".dockerfile", "Dockerfile"], // Dockerfile
  50. "GraphQL": [".graphql", ".gql"], // GraphQL文件
  51. "JSON5": [".json5"], // JSON5文件
  52. "TOML": [".toml"], // TOML文件
  53. "Text": [".txt", ".text"], // 文本文件
  54. "Other": [".bin", ".dat", ".exe", ".dll", ".obj", ".so", ".class"] // 二进制及无法识别的文件
  55. }
  56. // 判断文件是否为二进制文件
  57. isBinaryFile(filePath) {
  58. const binaryExtensions = ['.exe', '.dll', '.obj', '.so', '.bin', '.dat']
  59. const ext = path.extname(filePath).toLowerCase()
  60. return binaryExtensions.includes(ext)
  61. }
  62. // 统计单个文件的行数
  63. countLines(filePath) {
  64. return new Promise((resolve) => {
  65. if (this.isBinaryFile(filePath)) {
  66. return resolve(0) // 二进制文件直接跳过
  67. }
  68. let lineCount = 0;
  69. const stream = fs.createReadStream(filePath, 'utf8')
  70. stream.on("data", (chunk) => {
  71. lineCount += chunk.toString().split("\n").length
  72. });
  73. stream.on("end", () => resolve(lineCount))
  74. stream.on("error", () => resolve(0)) // 读取错误默认返回 0
  75. })
  76. }
  77. async onRequest(req, res) {
  78. let { uuid, session, id } = req.query
  79. if ([uuid, session, id].some(value => value === '' || value === null || value === undefined))
  80. return res.json({
  81. ...BaseStdResponse.MISSING_PARAMETER
  82. })
  83. // 检查 session
  84. if (!await AccessControl.checkSession(uuid, session))
  85. return res.status(401).json({
  86. ...BaseStdResponse.ACCESS_DENIED
  87. })
  88. let sql = 'SELECT state, path FROM repos WHERE create_user = ? AND id = ?'
  89. let r = await db.query(sql, [uuid, id])
  90. if (!r || r.length === 0)
  91. return res.json({
  92. ...BaseStdResponse.ERR,
  93. msg: '未找到仓库'
  94. })
  95. if (r[0].state !== 1 || !r[0].path)
  96. return res.json({
  97. ...BaseStdResponse.ERR,
  98. msg: '仓库未成功克隆!'
  99. })
  100. let totalLines = 0
  101. let languageStats = {}
  102. try {
  103. const git = simpleGit(r[0].path)
  104. const files = await git.raw(["ls-files"]) // 获取所有 Git 跟踪的文件
  105. const fileList = files.split("\n").map(f => f.trim()).filter(f => f)
  106. for (const file of fileList) {
  107. const fullPath = path.join(r[0].path, file)
  108. const ext = path.extname(file)
  109. const language = Object.keys(this.languageExtensions).find(lang =>
  110. this.languageExtensions[lang].includes(ext)
  111. )
  112. if (language) {
  113. const lines = await this.countLines(fullPath)
  114. totalLines += lines
  115. languageStats[language] = (languageStats[language] || 0) + lines
  116. } else {
  117. // 如果文件不属于任何已知语言,归类为其他
  118. const lines = await this.countLines(fullPath)
  119. totalLines += lines
  120. languageStats["Other"] = (languageStats["Other"] || 0) + lines
  121. }
  122. }
  123. } catch (error) {
  124. this.logger.error('获取仓库代码信息失败!' + error.stack)
  125. return res.json({
  126. ...BaseStdResponse.ERR,
  127. msg: '获取仓库代码信息失败!'
  128. })
  129. }
  130. res.json({
  131. ...BaseStdResponse.OK,
  132. data: {
  133. totalLines,
  134. languageStats
  135. }
  136. })
  137. }
  138. }
  139. module.exports.gitCodeStats = gitCodeStats