autosnippet 3.0.0 → 3.0.2

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.
Files changed (290) hide show
  1. package/README.md +230 -324
  2. package/bin/api-server.js +1 -1
  3. package/bin/cli.js +204 -244
  4. package/bin/mcp-server.js +5 -3
  5. package/config/knowledge-base.config.js +132 -132
  6. package/dashboard/dist/assets/{icons-CEfgGaZi.js → icons-Cdq22n2i.js} +95 -100
  7. package/dashboard/dist/assets/index-ClkyPkDX.js +133 -0
  8. package/dashboard/dist/assets/index-t4QrJwv1.css +1 -0
  9. package/dashboard/dist/index.html +3 -3
  10. package/lib/bootstrap.js +8 -8
  11. package/lib/cli/AiScanService.js +86 -40
  12. package/lib/cli/KnowledgeSyncService.js +113 -74
  13. package/lib/cli/SetupService.js +439 -277
  14. package/lib/cli/UpgradeService.js +63 -100
  15. package/lib/core/AstAnalyzer.js +276 -597
  16. package/lib/core/ast/ProjectGraph.js +101 -40
  17. package/lib/core/ast/ensure-grammars.js +232 -0
  18. package/lib/core/ast/index.js +115 -0
  19. package/lib/core/ast/lang-dart.js +661 -0
  20. package/lib/core/ast/lang-go.js +530 -0
  21. package/lib/core/ast/lang-java.js +435 -0
  22. package/lib/core/ast/lang-javascript.js +272 -0
  23. package/lib/core/ast/lang-kotlin.js +423 -0
  24. package/lib/core/ast/lang-objc.js +388 -0
  25. package/lib/core/ast/lang-python.js +371 -0
  26. package/lib/core/ast/lang-swift.js +337 -0
  27. package/lib/core/ast/lang-typescript.js +503 -0
  28. package/lib/core/capability/CapabilityProbe.js +18 -9
  29. package/lib/core/constitution/Constitution.js +2 -3
  30. package/lib/core/constitution/ConstitutionValidator.js +65 -24
  31. package/lib/core/discovery/DartDiscoverer.js +534 -0
  32. package/lib/core/discovery/DiscovererRegistry.js +83 -0
  33. package/lib/core/discovery/GenericDiscoverer.js +225 -0
  34. package/lib/core/discovery/GoDiscoverer.js +541 -0
  35. package/lib/core/discovery/JvmDiscoverer.js +506 -0
  36. package/lib/core/discovery/NodeDiscoverer.js +466 -0
  37. package/lib/core/discovery/ProjectDiscoverer.js +93 -0
  38. package/lib/core/discovery/PythonDiscoverer.js +338 -0
  39. package/lib/core/discovery/SpmDiscoverer.js +5 -0
  40. package/lib/core/discovery/index.js +53 -0
  41. package/lib/core/enhancement/EnhancementPack.js +71 -0
  42. package/lib/core/enhancement/EnhancementRegistry.js +47 -0
  43. package/lib/core/enhancement/android-enhancement.js +102 -0
  44. package/lib/core/enhancement/django-enhancement.js +70 -0
  45. package/lib/core/enhancement/fastapi-enhancement.js +63 -0
  46. package/lib/core/enhancement/go-grpc-enhancement.js +152 -0
  47. package/lib/core/enhancement/go-web-enhancement.js +201 -0
  48. package/lib/core/enhancement/index.js +65 -0
  49. package/lib/core/enhancement/node-server-enhancement.js +88 -0
  50. package/lib/core/enhancement/react-enhancement.js +86 -0
  51. package/lib/core/enhancement/spring-enhancement.js +112 -0
  52. package/lib/core/enhancement/vue-enhancement.js +96 -0
  53. package/lib/core/gateway/Gateway.js +8 -9
  54. package/lib/core/gateway/GatewayActionRegistry.js +1 -1
  55. package/lib/core/permission/PermissionManager.js +12 -8
  56. package/lib/domain/index.js +13 -9
  57. package/lib/domain/knowledge/KnowledgeEntry.js +111 -101
  58. package/lib/domain/knowledge/KnowledgeRepository.js +0 -1
  59. package/lib/domain/knowledge/Lifecycle.js +22 -22
  60. package/lib/domain/knowledge/index.js +9 -12
  61. package/lib/domain/knowledge/values/Constraints.js +31 -21
  62. package/lib/domain/knowledge/values/Content.js +21 -13
  63. package/lib/domain/knowledge/values/Quality.js +31 -18
  64. package/lib/domain/knowledge/values/Reasoning.js +20 -12
  65. package/lib/domain/knowledge/values/Relations.js +37 -25
  66. package/lib/domain/knowledge/values/Stats.js +18 -12
  67. package/lib/domain/knowledge/values/index.js +4 -3
  68. package/lib/domain/snippet/Snippet.js +35 -10
  69. package/lib/external/ai/AiFactory.js +48 -16
  70. package/lib/external/ai/AiProvider.js +184 -90
  71. package/lib/external/ai/providers/ClaudeProvider.js +25 -12
  72. package/lib/external/ai/providers/GoogleGeminiProvider.js +59 -30
  73. package/lib/external/ai/providers/MockProvider.js +9 -3
  74. package/lib/external/ai/providers/OpenAiProvider.js +51 -29
  75. package/lib/external/mcp/McpServer.js +66 -36
  76. package/lib/external/mcp/errorHandler.js +23 -11
  77. package/lib/external/mcp/handlers/LanguageExtensions.js +138 -53
  78. package/lib/external/mcp/handlers/TargetClassifier.js +52 -16
  79. package/lib/external/mcp/handlers/bootstrap/pipeline/BootstrapSnapshot.js +81 -20
  80. package/lib/external/mcp/handlers/bootstrap/pipeline/EpisodicMemory.js +71 -42
  81. package/lib/external/mcp/handlers/bootstrap/pipeline/IncrementalBootstrap.js +9 -17
  82. package/lib/external/mcp/handlers/bootstrap/pipeline/ToolResultCache.js +14 -9
  83. package/lib/external/mcp/handlers/bootstrap/pipeline/dimension-context.js +15 -7
  84. package/lib/external/mcp/handlers/bootstrap/pipeline/orchestrator.js +352 -153
  85. package/lib/external/mcp/handlers/bootstrap/pipeline/tier-scheduler.js +52 -12
  86. package/lib/external/mcp/handlers/bootstrap/skills.js +143 -39
  87. package/lib/external/mcp/handlers/bootstrap.js +691 -168
  88. package/lib/external/mcp/handlers/browse.js +66 -22
  89. package/lib/external/mcp/handlers/candidate.js +118 -35
  90. package/lib/external/mcp/handlers/consolidated.js +49 -17
  91. package/lib/external/mcp/handlers/guard.js +104 -39
  92. package/lib/external/mcp/handlers/knowledge.js +60 -36
  93. package/lib/external/mcp/handlers/search.js +43 -14
  94. package/lib/external/mcp/handlers/skill.js +120 -45
  95. package/lib/external/mcp/handlers/structure.js +240 -86
  96. package/lib/external/mcp/handlers/system.js +42 -12
  97. package/lib/external/mcp/handlers/wiki.js +58 -33
  98. package/lib/external/mcp/tools.js +306 -123
  99. package/lib/http/HttpServer.js +72 -47
  100. package/lib/http/middleware/RateLimiter.js +5 -3
  101. package/lib/http/middleware/errorHandler.js +6 -1
  102. package/lib/http/middleware/requestLogger.js +14 -3
  103. package/lib/http/middleware/roleResolver.js +30 -23
  104. package/lib/http/routes/ai.js +387 -265
  105. package/lib/http/routes/auth.js +81 -61
  106. package/lib/http/routes/candidates.js +430 -320
  107. package/lib/http/routes/commands.js +289 -189
  108. package/lib/http/routes/extract.js +158 -125
  109. package/lib/http/routes/guardRules.js +309 -217
  110. package/lib/http/routes/knowledge.js +213 -154
  111. package/lib/http/routes/modules.js +578 -0
  112. package/lib/http/routes/monitoring.js +6 -6
  113. package/lib/http/routes/recipes.js +104 -93
  114. package/lib/http/routes/search.js +361 -305
  115. package/lib/http/routes/skills.js +145 -98
  116. package/lib/http/routes/snippets.js +42 -30
  117. package/lib/http/routes/spm.js +3 -405
  118. package/lib/http/routes/violations.js +113 -93
  119. package/lib/http/routes/wiki.js +211 -170
  120. package/lib/http/utils/routeHelpers.js +3 -1
  121. package/lib/http/utils/sse-sessions.js +16 -6
  122. package/lib/http/utils/sse.js +15 -5
  123. package/lib/infrastructure/audit/AuditLogger.js +5 -2
  124. package/lib/infrastructure/audit/AuditStore.js +10 -7
  125. package/lib/infrastructure/cache/CacheService.js +3 -1
  126. package/lib/infrastructure/cache/GraphCache.js +8 -4
  127. package/lib/infrastructure/cache/UnifiedCacheAdapter.js +1 -1
  128. package/lib/infrastructure/config/ConfigLoader.js +9 -5
  129. package/lib/infrastructure/config/Defaults.js +30 -10
  130. package/lib/infrastructure/config/Paths.js +28 -8
  131. package/lib/infrastructure/config/TriggerSymbol.js +22 -10
  132. package/lib/infrastructure/database/DatabaseConnection.js +15 -10
  133. package/lib/infrastructure/database/migrations/001_initial_schema.js +0 -1
  134. package/lib/infrastructure/external/ClipboardManager.js +6 -2
  135. package/lib/infrastructure/external/NativeUi.js +50 -43
  136. package/lib/infrastructure/external/OpenBrowser.js +14 -17
  137. package/lib/infrastructure/external/XcodeAutomation.js +14 -258
  138. package/lib/infrastructure/logging/Logger.js +46 -30
  139. package/lib/infrastructure/monitoring/ErrorTracker.js +7 -5
  140. package/lib/infrastructure/monitoring/PerformanceMonitor.js +12 -4
  141. package/lib/infrastructure/paths/HeaderResolver.js +25 -9
  142. package/lib/infrastructure/paths/PathFinder.js +34 -12
  143. package/lib/infrastructure/plugin/PluginManager.js +26 -8
  144. package/lib/infrastructure/realtime/RealtimeService.js +2 -2
  145. package/lib/infrastructure/vector/Chunker.js +22 -7
  146. package/lib/infrastructure/vector/IndexingPipeline.js +46 -22
  147. package/lib/infrastructure/vector/JsonVectorAdapter.js +90 -53
  148. package/lib/infrastructure/vector/VectorStore.js +28 -10
  149. package/lib/injection/ServiceContainer.js +247 -93
  150. package/lib/platform/ios/index.js +63 -0
  151. package/lib/platform/ios/routes/spm.js +437 -0
  152. package/lib/platform/ios/snippet/PlaceholderConverter.js +55 -0
  153. package/lib/platform/ios/snippet/XcodeCodec.js +112 -0
  154. package/lib/{service → platform/ios}/spm/DependencyGraph.js +41 -17
  155. package/lib/{service → platform/ios}/spm/PackageSwiftParser.js +41 -14
  156. package/lib/{service → platform/ios}/spm/PolicyEngine.js +9 -4
  157. package/lib/platform/ios/spm/SpmDiscoverer.js +122 -0
  158. package/lib/{service → platform/ios}/spm/SpmService.js +385 -127
  159. package/lib/{service/automation → platform/ios/xcode}/SaveEventFilter.js +8 -7
  160. package/lib/platform/ios/xcode/XcodeAutomation.js +350 -0
  161. package/lib/{service/automation → platform/ios/xcode}/XcodeIntegration.js +325 -145
  162. package/lib/repository/base/BaseRepository.js +7 -9
  163. package/lib/repository/knowledge/KnowledgeRepository.impl.js +98 -75
  164. package/lib/repository/token/TokenUsageStore.js +4 -2
  165. package/lib/service/automation/ActionPipeline.js +1 -1
  166. package/lib/service/automation/AutomationOrchestrator.js +8 -4
  167. package/lib/service/automation/ContextCollector.js +7 -5
  168. package/lib/service/automation/DirectiveDetector.js +23 -16
  169. package/lib/service/automation/FileWatcher.js +112 -56
  170. package/lib/service/automation/TriggerResolver.js +6 -4
  171. package/lib/service/automation/handlers/AlinkHandler.js +24 -12
  172. package/lib/service/automation/handlers/CreateHandler.js +19 -20
  173. package/lib/service/automation/handlers/DraftHandler.js +14 -8
  174. package/lib/service/automation/handlers/GuardHandler.js +93 -63
  175. package/lib/service/automation/handlers/HeaderHandler.js +1 -6
  176. package/lib/service/automation/handlers/SearchHandler.js +155 -88
  177. package/lib/service/bootstrap/BootstrapTaskManager.js +77 -35
  178. package/lib/service/candidate/SimilarityService.js +25 -9
  179. package/lib/service/chat/AnalystAgent.js +50 -24
  180. package/lib/service/chat/CandidateGuardrail.js +143 -17
  181. package/lib/service/chat/ChatAgent.js +759 -243
  182. package/lib/service/chat/ContextWindow.js +116 -71
  183. package/lib/service/chat/ConversationStore.js +77 -36
  184. package/lib/service/chat/EpisodicConsolidator.js +47 -23
  185. package/lib/service/chat/HandoffProtocol.js +98 -22
  186. package/lib/service/chat/Memory.js +34 -14
  187. package/lib/service/chat/ProducerAgent.js +40 -20
  188. package/lib/service/chat/ProjectSemanticMemory.js +109 -78
  189. package/lib/service/chat/ReasoningLayer.js +148 -70
  190. package/lib/service/chat/ReasoningTrace.js +44 -32
  191. package/lib/service/chat/TaskPipeline.js +39 -19
  192. package/lib/service/chat/ToolRegistry.js +48 -29
  193. package/lib/service/chat/WorkingMemory.js +44 -18
  194. package/lib/service/chat/tools.js +1096 -494
  195. package/lib/service/context/RecipeExtractor.js +132 -51
  196. package/lib/service/cursor/CursorDeliveryPipeline.js +82 -37
  197. package/lib/service/cursor/KnowledgeCompressor.js +25 -22
  198. package/lib/service/cursor/RulesGenerator.js +13 -7
  199. package/lib/service/cursor/SkillsSyncer.js +77 -27
  200. package/lib/service/cursor/TokenBudget.js +2 -2
  201. package/lib/service/cursor/TopicClassifier.js +54 -20
  202. package/lib/service/guard/ComplianceReporter.js +55 -43
  203. package/lib/service/guard/ExclusionManager.js +67 -29
  204. package/lib/service/guard/GuardCheckEngine.js +381 -86
  205. package/lib/service/guard/GuardFeedbackLoop.js +22 -10
  206. package/lib/service/guard/GuardService.js +29 -19
  207. package/lib/service/guard/RuleLearner.js +55 -23
  208. package/lib/service/guard/SourceFileCollector.js +27 -20
  209. package/lib/service/guard/ViolationsStore.js +43 -38
  210. package/lib/service/knowledge/CodeEntityGraph.js +147 -82
  211. package/lib/service/knowledge/ConfidenceRouter.js +12 -10
  212. package/lib/service/knowledge/KnowledgeFileWriter.js +147 -56
  213. package/lib/service/knowledge/KnowledgeGraphService.js +81 -34
  214. package/lib/service/knowledge/KnowledgeService.js +222 -112
  215. package/lib/service/module/ModuleService.js +969 -0
  216. package/lib/service/quality/FeedbackCollector.js +27 -15
  217. package/lib/service/quality/QualityScorer.js +78 -24
  218. package/lib/service/recipe/RecipeCandidateValidator.js +110 -44
  219. package/lib/service/recipe/RecipeParser.js +78 -45
  220. package/lib/service/search/CoarseRanker.js +43 -28
  221. package/lib/service/search/CrossEncoderReranker.js +32 -21
  222. package/lib/service/search/InvertedIndex.js +21 -7
  223. package/lib/service/search/MultiSignalRanker.js +90 -28
  224. package/lib/service/search/RetrievalFunnel.js +45 -24
  225. package/lib/service/search/SearchEngine.js +255 -103
  226. package/lib/service/skills/EventAggregator.js +32 -15
  227. package/lib/service/skills/SignalCollector.js +140 -64
  228. package/lib/service/skills/SkillAdvisor.js +79 -42
  229. package/lib/service/skills/SkillHooks.js +16 -14
  230. package/lib/service/snippet/PlaceholderConverter.js +5 -0
  231. package/lib/service/snippet/SnippetFactory.js +116 -99
  232. package/lib/service/snippet/SnippetInstaller.js +234 -62
  233. package/lib/service/snippet/codecs/SnippetCodec.js +67 -0
  234. package/lib/service/snippet/codecs/VSCodeCodec.js +102 -0
  235. package/lib/service/snippet/codecs/XcodeCodec.js +5 -0
  236. package/lib/service/wiki/WikiGenerator.js +637 -263
  237. package/lib/shared/DimensionCopyRegistry.js +472 -0
  238. package/lib/shared/LanguageService.js +399 -0
  239. package/lib/shared/PathGuard.js +45 -28
  240. package/lib/shared/RecipeReadinessChecker.js +72 -12
  241. package/lib/shared/constants.js +41 -41
  242. package/lib/shared/errors/BaseError.js +2 -2
  243. package/lib/shared/errors/index.js +4 -4
  244. package/lib/shared/similarity.js +25 -8
  245. package/lib/shared/token-utils.js +6 -2
  246. package/lib/shared/utils/common.js +12 -4
  247. package/package.json +49 -13
  248. package/scripts/bench-real-projects.mjs +256 -0
  249. package/scripts/build-native-ui.js +30 -30
  250. package/scripts/clear-old-vector-index.js +5 -35
  251. package/scripts/clear-vector-cache.js +7 -37
  252. package/scripts/collect-test-project-stats.mjs +160 -0
  253. package/scripts/diagnose-mcp.js +41 -32
  254. package/scripts/ensure-parse-package.js +6 -9
  255. package/scripts/generate-recipe-drafts.js +116 -77
  256. package/scripts/init-db.js +3 -20
  257. package/scripts/init-snippets.js +305 -0
  258. package/scripts/init-vector-db.js +173 -170
  259. package/scripts/install-cursor-skill.js +148 -104
  260. package/scripts/install-full.js +8 -21
  261. package/scripts/install-vscode-copilot.js +146 -145
  262. package/scripts/migrate-md-to-knowledge.mjs +139 -151
  263. package/scripts/postinstall-safe.js +5 -17
  264. package/scripts/recipe-audit.js +106 -82
  265. package/scripts/release.js +283 -323
  266. package/scripts/setup-mcp-config.js +60 -52
  267. package/scripts/verify-context-api.js +20 -20
  268. package/skills/autosnippet-analysis/SKILL.md +10 -6
  269. package/skills/autosnippet-candidates/SKILL.md +27 -26
  270. package/skills/autosnippet-coldstart/SKILL.md +555 -38
  271. package/skills/autosnippet-concepts/SKILL.md +349 -337
  272. package/skills/autosnippet-create/SKILL.md +5 -5
  273. package/skills/autosnippet-reference-dart/SKILL.md +543 -0
  274. package/skills/autosnippet-reference-go/SKILL.md +539 -0
  275. package/skills/autosnippet-reference-java/SKILL.md +534 -0
  276. package/skills/autosnippet-reference-jsts/SKILL.md +41 -9
  277. package/skills/autosnippet-reference-kotlin/SKILL.md +526 -0
  278. package/skills/autosnippet-reference-objc/SKILL.md +29 -6
  279. package/skills/autosnippet-reference-python/SKILL.md +800 -0
  280. package/skills/autosnippet-reference-swift/SKILL.md +70 -14
  281. package/skills/autosnippet-structure/SKILL.md +4 -4
  282. package/templates/cursor-rules/autosnippet-conventions.mdc +2 -2
  283. package/templates/recipes-setup/README.md +2 -2
  284. package/templates/recipes-setup/_template.md +1 -1
  285. package/dashboard/dist/assets/index-Bun3ld_J.css +0 -1
  286. package/dashboard/dist/assets/index-_Sk_Dmg3.js +0 -143
  287. package/resources/asd-entry/main.swift +0 -159
  288. package/scripts/build-asd-entry.js +0 -51
  289. package/scripts/init-xcode-snippets.js +0 -311
  290. package/template.json +0 -39
@@ -0,0 +1,506 @@
1
+ /**
2
+ * @module JvmDiscoverer
3
+ * @description Java / Kotlin 项目结构发现器
4
+ *
5
+ * 检测信号: build.gradle, build.gradle.kts, pom.xml, settings.gradle
6
+ * 支持: Gradle (单模块/多模块), Maven (单模块/多模块)
7
+ *
8
+ * ⚠️ 不尝试精确解析 Gradle DSL,仅用正则启发式提取关键信息
9
+ */
10
+
11
+ import { existsSync, readdirSync, readFileSync } from 'node:fs';
12
+ import { basename, extname, join, relative, resolve } from 'node:path';
13
+ import { LanguageService } from '../../shared/LanguageService.js';
14
+ import { ProjectDiscoverer } from './ProjectDiscoverer.js';
15
+
16
+ const SOURCE_EXTENSIONS = new Set(['.java', '.kt', '.kts']);
17
+ const EXCLUDE_DIRS = new Set([
18
+ '.gradle',
19
+ '.idea',
20
+ 'build',
21
+ 'target',
22
+ '.git',
23
+ 'node_modules',
24
+ 'out',
25
+ '.kotlin',
26
+ ]);
27
+
28
+ export class JvmDiscoverer extends ProjectDiscoverer {
29
+ #projectRoot = null;
30
+ #targets = [];
31
+ #depGraph = { nodes: [], edges: [] };
32
+ #buildSystem = null; // 'gradle' | 'maven'
33
+
34
+ get id() {
35
+ return 'jvm';
36
+ }
37
+ get displayName() {
38
+ return `JVM (${this.#buildSystem === 'maven' ? 'Maven' : 'Gradle'})`;
39
+ }
40
+
41
+ async detect(projectRoot) {
42
+ let confidence = 0;
43
+ const reasons = [];
44
+
45
+ // Gradle
46
+ if (
47
+ existsSync(join(projectRoot, 'build.gradle')) ||
48
+ existsSync(join(projectRoot, 'build.gradle.kts'))
49
+ ) {
50
+ confidence = 0.9;
51
+ reasons.push('build.gradle(.kts) exists');
52
+ }
53
+ if (
54
+ existsSync(join(projectRoot, 'settings.gradle')) ||
55
+ existsSync(join(projectRoot, 'settings.gradle.kts'))
56
+ ) {
57
+ confidence = Math.max(confidence, 0.85);
58
+ confidence = Math.min(confidence + 0.05, 1.0);
59
+ reasons.push('settings.gradle(.kts) exists');
60
+ }
61
+
62
+ // Maven
63
+ if (existsSync(join(projectRoot, 'pom.xml'))) {
64
+ confidence = Math.max(confidence, 0.85);
65
+ reasons.push('pom.xml exists');
66
+ }
67
+
68
+ return {
69
+ match: confidence > 0,
70
+ confidence: Math.min(confidence, 1.0),
71
+ reason: reasons.join(', ') || 'No JVM markers found',
72
+ };
73
+ }
74
+
75
+ async load(projectRoot) {
76
+ this.#projectRoot = projectRoot;
77
+ this.#targets = [];
78
+ this.#depGraph = { nodes: [], edges: [] };
79
+
80
+ // 判断构建系统
81
+ const hasGradle =
82
+ existsSync(join(projectRoot, 'build.gradle')) ||
83
+ existsSync(join(projectRoot, 'build.gradle.kts'));
84
+ const hasMaven = existsSync(join(projectRoot, 'pom.xml'));
85
+
86
+ if (hasGradle) {
87
+ this.#buildSystem = 'gradle';
88
+ this.#loadGradle(projectRoot);
89
+ } else if (hasMaven) {
90
+ this.#buildSystem = 'maven';
91
+ this.#loadMaven(projectRoot);
92
+ }
93
+ }
94
+
95
+ async listTargets() {
96
+ return this.#targets;
97
+ }
98
+
99
+ async getTargetFiles(target) {
100
+ const targetObj =
101
+ typeof target === 'string' ? this.#targets.find((t) => t.name === target) : target;
102
+
103
+ if (!targetObj?.path || !existsSync(targetObj.path)) {
104
+ return [];
105
+ }
106
+
107
+ const files = [];
108
+ // JVM 约定: src/main/java, src/main/kotlin, src/test/java, src/test/kotlin
109
+ const sourceDirs = [
110
+ join(targetObj.path, 'src', 'main', 'java'),
111
+ join(targetObj.path, 'src', 'main', 'kotlin'),
112
+ join(targetObj.path, 'src', 'test', 'java'),
113
+ join(targetObj.path, 'src', 'test', 'kotlin'),
114
+ ];
115
+
116
+ // 也支持非标准布局 — 直接在 target 路径下搜索
117
+ const hasSrcDir = sourceDirs.some((d) => existsSync(d));
118
+ if (hasSrcDir) {
119
+ for (const srcDir of sourceDirs) {
120
+ if (existsSync(srcDir)) {
121
+ this.#collectFiles(srcDir, targetObj.path, files);
122
+ }
123
+ }
124
+ } else {
125
+ this.#collectFiles(targetObj.path, targetObj.path, files);
126
+ }
127
+
128
+ return files;
129
+ }
130
+
131
+ async getDependencyGraph() {
132
+ return this.#depGraph;
133
+ }
134
+
135
+ // ── Gradle ──
136
+
137
+ #loadGradle(projectRoot) {
138
+ // 解析 settings.gradle 找子模块
139
+ const submodules = this.#parseGradleSettings(projectRoot);
140
+
141
+ if (submodules.length > 0) {
142
+ // 多模块 Gradle 项目
143
+ for (const mod of submodules) {
144
+ const modPath = resolve(projectRoot, mod.replace(/:/g, '/'));
145
+ if (!existsSync(modPath)) {
146
+ continue;
147
+ }
148
+
149
+ const framework = this.#detectGradleFramework(modPath);
150
+ const lang = this.#detectPrimaryLang(modPath);
151
+
152
+ this.#targets.push({
153
+ name: mod,
154
+ path: modPath,
155
+ type: this.#inferGradleTargetType(modPath, mod),
156
+ language: lang,
157
+ framework,
158
+ metadata: { buildSystem: 'gradle', module: mod },
159
+ });
160
+ this.#depGraph.nodes.push(mod);
161
+ }
162
+
163
+ // 提取模块间依赖
164
+ this.#parseGradleModuleDeps(projectRoot, submodules);
165
+ } else {
166
+ // 单模块 Gradle 项目
167
+ const framework = this.#detectGradleFramework(projectRoot);
168
+ const lang = this.#detectPrimaryLang(projectRoot);
169
+ const name = basename(projectRoot);
170
+
171
+ this.#targets.push({
172
+ name,
173
+ path: projectRoot,
174
+ type: 'app',
175
+ language: lang,
176
+ framework,
177
+ metadata: { buildSystem: 'gradle' },
178
+ });
179
+ this.#depGraph.nodes.push(name);
180
+ }
181
+
182
+ // 提取外部依赖
183
+ this.#parseGradleExternalDeps(projectRoot);
184
+ }
185
+
186
+ #parseGradleSettings(projectRoot) {
187
+ const modules = [];
188
+ for (const fname of ['settings.gradle', 'settings.gradle.kts']) {
189
+ const settingsPath = join(projectRoot, fname);
190
+ if (!existsSync(settingsPath)) {
191
+ continue;
192
+ }
193
+
194
+ try {
195
+ const content = readFileSync(settingsPath, 'utf8');
196
+ // include ':app', ':lib:core', ...
197
+ const includeMatches = content.matchAll(
198
+ /include\s*\(?[\s]*['"]([^'"]+)['"](?:\s*,\s*['"]([^'"]+)['"])*/g
199
+ );
200
+ for (const m of includeMatches) {
201
+ for (let i = 1; i < m.length; i++) {
202
+ if (m[i]) {
203
+ modules.push(m[i].replace(/^:/, ''));
204
+ }
205
+ }
206
+ }
207
+ // include(":app")
208
+ const includeKtsMatches = content.matchAll(/include\s*\(\s*["']([^"']+)["']\s*\)/g);
209
+ for (const m of includeKtsMatches) {
210
+ if (m[1]) {
211
+ modules.push(m[1].replace(/^:/, ''));
212
+ }
213
+ }
214
+ } catch {
215
+ /* skip */
216
+ }
217
+ }
218
+ return [...new Set(modules)];
219
+ }
220
+
221
+ #detectGradleFramework(dir) {
222
+ for (const fname of ['build.gradle', 'build.gradle.kts']) {
223
+ const buildPath = join(dir, fname);
224
+ if (!existsSync(buildPath)) {
225
+ continue;
226
+ }
227
+ try {
228
+ const content = readFileSync(buildPath, 'utf8');
229
+ if (/com\.android|android\s*\{|apply.*android/.test(content)) {
230
+ return 'android';
231
+ }
232
+ if (/org\.springframework|spring-boot/.test(content)) {
233
+ return 'spring';
234
+ }
235
+ if (/io\.ktor/.test(content)) {
236
+ return 'ktor';
237
+ }
238
+ if (/org\.jetbrains\.compose/.test(content)) {
239
+ return 'compose';
240
+ }
241
+ } catch {
242
+ /* skip */
243
+ }
244
+ }
245
+ return null;
246
+ }
247
+
248
+ #inferGradleTargetType(dir, name) {
249
+ for (const fname of ['build.gradle', 'build.gradle.kts']) {
250
+ const buildPath = join(dir, fname);
251
+ if (!existsSync(buildPath)) {
252
+ continue;
253
+ }
254
+ try {
255
+ const content = readFileSync(buildPath, 'utf8');
256
+ if (/application|com\.android\.application/.test(content)) {
257
+ return 'app';
258
+ }
259
+ if (/java-library|com\.android\.library/.test(content)) {
260
+ return 'library';
261
+ }
262
+ } catch {
263
+ /* skip */
264
+ }
265
+ }
266
+ if (/test/i.test(name)) {
267
+ return 'test';
268
+ }
269
+ return 'library';
270
+ }
271
+
272
+ #parseGradleModuleDeps(projectRoot, submodules) {
273
+ const moduleSet = new Set(submodules);
274
+ for (const mod of submodules) {
275
+ const modPath = resolve(projectRoot, mod.replace(/:/g, '/'));
276
+ for (const fname of ['build.gradle', 'build.gradle.kts']) {
277
+ const buildPath = join(modPath, fname);
278
+ if (!existsSync(buildPath)) {
279
+ continue;
280
+ }
281
+ try {
282
+ const content = readFileSync(buildPath, 'utf8');
283
+ // project(':lib:core'), project(":lib:core")
284
+ const projDeps = content.matchAll(/project\s*\(\s*['"][:.]?([^'"]+)['"]\s*\)/g);
285
+ for (const m of projDeps) {
286
+ const depMod = m[1].replace(/^:/, '');
287
+ if (moduleSet.has(depMod)) {
288
+ this.#depGraph.edges.push({ from: mod, to: depMod, type: 'depends_on' });
289
+ }
290
+ }
291
+ } catch {
292
+ /* skip */
293
+ }
294
+ }
295
+ }
296
+ }
297
+
298
+ #parseGradleExternalDeps(projectRoot) {
299
+ for (const fname of ['build.gradle', 'build.gradle.kts']) {
300
+ const buildPath = join(projectRoot, fname);
301
+ if (!existsSync(buildPath)) {
302
+ continue;
303
+ }
304
+ try {
305
+ const content = readFileSync(buildPath, 'utf8');
306
+ const rootTarget = this.#targets[0]?.name;
307
+ if (!rootTarget) {
308
+ return;
309
+ }
310
+
311
+ // implementation 'group:artifact:version' or implementation("group:artifact:version")
312
+ const depMatches = content.matchAll(
313
+ /(?:implementation|api|compileOnly|runtimeOnly)\s*[("']+([^)'"]+)[)'"]+/g
314
+ );
315
+ for (const m of depMatches) {
316
+ const parts = m[1].split(':');
317
+ if (parts.length >= 2) {
318
+ const depName = `${parts[0]}:${parts[1]}`;
319
+ this.#depGraph.edges.push({ from: rootTarget, to: depName, type: 'depends_on' });
320
+ }
321
+ }
322
+ } catch {
323
+ /* skip */
324
+ }
325
+ }
326
+ }
327
+
328
+ // ── Maven ──
329
+
330
+ #loadMaven(projectRoot) {
331
+ const pomPath = join(projectRoot, 'pom.xml');
332
+ if (!existsSync(pomPath)) {
333
+ return;
334
+ }
335
+
336
+ const pomContent = readFileSync(pomPath, 'utf8');
337
+ const projectName = this.#extractXmlValue(pomContent, 'artifactId') || basename(projectRoot);
338
+
339
+ // 提取子模块
340
+ const modules = this.#parseMavenModules(pomContent);
341
+
342
+ if (modules.length > 0) {
343
+ for (const mod of modules) {
344
+ const modPath = resolve(projectRoot, mod);
345
+ if (!existsSync(modPath)) {
346
+ continue;
347
+ }
348
+
349
+ const lang = this.#detectPrimaryLang(modPath);
350
+ const framework = this.#detectMavenFramework(modPath);
351
+
352
+ this.#targets.push({
353
+ name: mod,
354
+ path: modPath,
355
+ type: /test/i.test(mod) ? 'test' : 'library',
356
+ language: lang,
357
+ framework,
358
+ metadata: { buildSystem: 'maven', module: mod },
359
+ });
360
+ this.#depGraph.nodes.push(mod);
361
+ }
362
+ } else {
363
+ const lang = this.#detectPrimaryLang(projectRoot);
364
+ const framework = this.#detectMavenFramework(projectRoot);
365
+
366
+ this.#targets.push({
367
+ name: projectName,
368
+ path: projectRoot,
369
+ type: 'app',
370
+ language: lang,
371
+ framework,
372
+ metadata: { buildSystem: 'maven' },
373
+ });
374
+ this.#depGraph.nodes.push(projectName);
375
+ }
376
+
377
+ // 提取外部依赖
378
+ this.#parseMavenDeps(pomContent);
379
+ }
380
+
381
+ #parseMavenModules(pomContent) {
382
+ const modules = [];
383
+ const moduleMatches = pomContent.matchAll(/<module>([^<]+)<\/module>/g);
384
+ for (const m of moduleMatches) {
385
+ modules.push(m[1].trim());
386
+ }
387
+ return modules;
388
+ }
389
+
390
+ #detectMavenFramework(dir) {
391
+ const pomPath = join(dir, 'pom.xml');
392
+ if (!existsSync(pomPath)) {
393
+ return null;
394
+ }
395
+ try {
396
+ const content = readFileSync(pomPath, 'utf8');
397
+ if (/spring-boot|springframework/.test(content)) {
398
+ return 'spring';
399
+ }
400
+ if (/android/.test(content)) {
401
+ return 'android';
402
+ }
403
+ } catch {
404
+ /* skip */
405
+ }
406
+ return null;
407
+ }
408
+
409
+ #parseMavenDeps(pomContent) {
410
+ const rootTarget = this.#targets[0]?.name;
411
+ if (!rootTarget) {
412
+ return;
413
+ }
414
+
415
+ // 简化: 提取 <dependency> 中的 groupId:artifactId
416
+ const depBlocks = pomContent.matchAll(/<dependency>([\s\S]*?)<\/dependency>/g);
417
+ for (const block of depBlocks) {
418
+ const groupId = this.#extractXmlValue(block[1], 'groupId');
419
+ const artifactId = this.#extractXmlValue(block[1], 'artifactId');
420
+ if (groupId && artifactId) {
421
+ this.#depGraph.edges.push({
422
+ from: rootTarget,
423
+ to: `${groupId}:${artifactId}`,
424
+ type: 'depends_on',
425
+ });
426
+ }
427
+ }
428
+ }
429
+
430
+ // ── 共用工具 ──
431
+
432
+ #detectPrimaryLang(dir) {
433
+ let javaCount = 0;
434
+ let kotlinCount = 0;
435
+
436
+ const srcMain = join(dir, 'src', 'main');
437
+ if (existsSync(join(srcMain, 'kotlin'))) {
438
+ kotlinCount += 10;
439
+ }
440
+ if (existsSync(join(srcMain, 'java'))) {
441
+ javaCount += 10;
442
+ }
443
+
444
+ // 快速采样
445
+ const srcDirs = [join(srcMain, 'java'), join(srcMain, 'kotlin'), dir];
446
+ for (const sd of srcDirs) {
447
+ if (!existsSync(sd)) {
448
+ continue;
449
+ }
450
+ try {
451
+ const files = readdirSync(sd).slice(0, 20);
452
+ for (const f of files) {
453
+ if (f.endsWith('.kt') || f.endsWith('.kts')) {
454
+ kotlinCount++;
455
+ }
456
+ if (f.endsWith('.java')) {
457
+ javaCount++;
458
+ }
459
+ }
460
+ } catch {
461
+ /* skip */
462
+ }
463
+ }
464
+
465
+ return kotlinCount > javaCount ? 'kotlin' : 'java';
466
+ }
467
+
468
+ #collectFiles(dir, rootDir, files, depth = 0) {
469
+ if (depth > 15) {
470
+ return;
471
+ }
472
+ try {
473
+ const entries = readdirSync(dir, { withFileTypes: true });
474
+ for (const entry of entries) {
475
+ if (entry.name.startsWith('.')) {
476
+ continue;
477
+ }
478
+ if (EXCLUDE_DIRS.has(entry.name)) {
479
+ continue;
480
+ }
481
+
482
+ const fullPath = join(dir, entry.name);
483
+ if (entry.isDirectory()) {
484
+ this.#collectFiles(fullPath, rootDir, files, depth + 1);
485
+ } else if (entry.isFile()) {
486
+ const ext = extname(entry.name);
487
+ if (SOURCE_EXTENSIONS.has(ext)) {
488
+ files.push({
489
+ name: entry.name,
490
+ path: fullPath,
491
+ relativePath: relative(rootDir, fullPath),
492
+ language: LanguageService.inferLang(entry.name),
493
+ });
494
+ }
495
+ }
496
+ }
497
+ } catch {
498
+ /* skip */
499
+ }
500
+ }
501
+
502
+ #extractXmlValue(xml, tag) {
503
+ const match = xml.match(new RegExp(`<${tag}>([^<]*)</${tag}>`));
504
+ return match ? match[1].trim() : null;
505
+ }
506
+ }