@ngockhoale/ukit 1.1.6

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 (344) hide show
  1. package/CHANGELOG.md +179 -0
  2. package/LICENSE +21 -0
  3. package/README.md +189 -0
  4. package/bin/ukit +30 -0
  5. package/manifests/platform.full.yaml +1194 -0
  6. package/package.json +71 -0
  7. package/scripts/bug/triage.mjs +37 -0
  8. package/scripts/index/build-index.mjs +35 -0
  9. package/scripts/index/query-index.mjs +92 -0
  10. package/scripts/index/refresh-index.mjs +85 -0
  11. package/scripts/release/verify-release.mjs +56 -0
  12. package/src/bug/triageBug.js +123 -0
  13. package/src/cli/adapters.js +148 -0
  14. package/src/cli/commands/diff.js +51 -0
  15. package/src/cli/commands/doctor.js +125 -0
  16. package/src/cli/commands/indexArgs.js +73 -0
  17. package/src/cli/commands/indexTools.js +509 -0
  18. package/src/cli/commands/install.js +293 -0
  19. package/src/cli/commands/memory.js +126 -0
  20. package/src/cli/commands/status.js +8 -0
  21. package/src/cli/commands/uninstall.js +51 -0
  22. package/src/cli/index.js +109 -0
  23. package/src/context/detectProjectContext.js +49 -0
  24. package/src/context/detectProviders.js +12 -0
  25. package/src/core/applyPlan.js +89 -0
  26. package/src/core/buildPlan.js +228 -0
  27. package/src/core/compact/index.js +294 -0
  28. package/src/core/compact/threshold.js +936 -0
  29. package/src/core/diffPlan.js +73 -0
  30. package/src/core/ensureGitignore.js +117 -0
  31. package/src/core/fileOps.js +188 -0
  32. package/src/core/memory/hygiene.js +160 -0
  33. package/src/core/memory/index.js +2 -0
  34. package/src/core/memory/retrieval.js +476 -0
  35. package/src/core/memory/store.js +202 -0
  36. package/src/core/metadata.js +132 -0
  37. package/src/core/migrateLegacy.js +139 -0
  38. package/src/core/output/index.js +1309 -0
  39. package/src/core/paths.js +13 -0
  40. package/src/core/report.js +17 -0
  41. package/src/core/router/advisor.js +42 -0
  42. package/src/core/router/index.js +2 -0
  43. package/src/core/router/router.js +164 -0
  44. package/src/core/runInstallPipeline.js +365 -0
  45. package/src/core/runtimeConfig.js +190 -0
  46. package/src/core/runtimePaths.js +24 -0
  47. package/src/core/status.js +186 -0
  48. package/src/core/token/index.js +328 -0
  49. package/src/core/uninstall.js +246 -0
  50. package/src/core/validation/confidence.js +89 -0
  51. package/src/core/validation/index.js +2 -0
  52. package/src/core/validation/validator.js +165 -0
  53. package/src/index/buildIndex.js +1392 -0
  54. package/src/index/gitHooks.js +109 -0
  55. package/src/index/importResolution.js +377 -0
  56. package/src/index/languageTools.js +127 -0
  57. package/src/index/paths.js +27 -0
  58. package/src/index/queryIndex.js +637 -0
  59. package/src/index/relatedTests.js +237 -0
  60. package/src/index/resolveContext.js +345 -0
  61. package/src/index/routeCatalog.js +258 -0
  62. package/src/index/taskRouting.js +677 -0
  63. package/src/index/verificationPlan.js +437 -0
  64. package/src/manifest/loadManifest.js +22 -0
  65. package/src/manifest/selectItems.js +78 -0
  66. package/src/manifest/validateManifest.js +115 -0
  67. package/src/render/buildVariables.js +39 -0
  68. package/src/render/renderTemplate.js +44 -0
  69. package/src/stack/detectStack.js +213 -0
  70. package/templates/.claude/agents/bug-debugger.md +57 -0
  71. package/templates/.claude/agents/feature-implementer.md +55 -0
  72. package/templates/.claude/config/providers.md +25 -0
  73. package/templates/.claude/hooks/auto-allow-bash.sh +155 -0
  74. package/templates/.claude/hooks/auto-prune-bash.sh +75 -0
  75. package/templates/.claude/hooks/block-dangerous.sh +54 -0
  76. package/templates/.claude/hooks/compress-output.sh +17 -0
  77. package/templates/.claude/hooks/protect-files.sh +37 -0
  78. package/templates/.claude/hooks/reinject-context.sh +28 -0
  79. package/templates/.claude/hooks/session-start.md +13 -0
  80. package/templates/.claude/hooks/skill-router.sh +1681 -0
  81. package/templates/.claude/hooks/verification-guard.sh +271 -0
  82. package/templates/.claude/settings.json +144 -0
  83. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  84. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  85. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  86. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  87. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  88. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  89. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  90. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  91. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  92. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  93. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  94. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  95. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  96. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  97. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  98. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  99. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  100. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  101. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  102. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  103. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  104. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  105. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  106. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  107. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  108. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  109. package/templates/.claude/skills/_shared/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  110. package/templates/.claude/skills/_shared/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  111. package/templates/.claude/skills/_shared/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  112. package/templates/.claude/skills/_shared/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  113. package/templates/.claude/skills/_shared/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  114. package/templates/.claude/skills/_shared/ooxml/schemas/mce/mc.xsd +75 -0
  115. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
  116. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
  117. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
  118. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
  119. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
  120. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  121. package/templates/.claude/skills/_shared/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
  122. package/templates/.claude/skills/_shared/ooxml/scripts/pack.py +159 -0
  123. package/templates/.claude/skills/_shared/ooxml/scripts/unpack.py +29 -0
  124. package/templates/.claude/skills/_shared/ooxml/scripts/validate.py +69 -0
  125. package/templates/.claude/skills/_shared/ooxml/scripts/validation/__init__.py +15 -0
  126. package/templates/.claude/skills/_shared/ooxml/scripts/validation/base.py +951 -0
  127. package/templates/.claude/skills/_shared/ooxml/scripts/validation/docx.py +274 -0
  128. package/templates/.claude/skills/_shared/ooxml/scripts/validation/pptx.py +315 -0
  129. package/templates/.claude/skills/_shared/ooxml/scripts/validation/redlining.py +279 -0
  130. package/templates/.claude/skills/backend-api/SKILL.md +26 -0
  131. package/templates/.claude/skills/canvas-design/LICENSE.txt +202 -0
  132. package/templates/.claude/skills/canvas-design/SKILL.md +130 -0
  133. package/templates/.claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
  134. package/templates/.claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
  135. package/templates/.claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
  136. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
  137. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
  138. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
  139. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSans-OFL.txt +93 -0
  140. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
  141. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
  142. package/templates/.claude/skills/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
  143. package/templates/.claude/skills/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
  144. package/templates/.claude/skills/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
  145. package/templates/.claude/skills/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
  146. package/templates/.claude/skills/canvas-design/canvas-fonts/Lora-Bold.ttf +0 -0
  147. package/templates/.claude/skills/canvas-design/canvas-fonts/Lora-BoldItalic.ttf +0 -0
  148. package/templates/.claude/skills/canvas-design/canvas-fonts/Lora-Italic.ttf +0 -0
  149. package/templates/.claude/skills/canvas-design/canvas-fonts/Lora-OFL.txt +93 -0
  150. package/templates/.claude/skills/canvas-design/canvas-fonts/Lora-Regular.ttf +0 -0
  151. package/templates/.claude/skills/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
  152. package/templates/.claude/skills/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
  153. package/templates/.claude/skills/canvas-design/canvas-fonts/Outfit-Bold.ttf +0 -0
  154. package/templates/.claude/skills/canvas-design/canvas-fonts/Outfit-OFL.txt +93 -0
  155. package/templates/.claude/skills/canvas-design/canvas-fonts/Outfit-Regular.ttf +0 -0
  156. package/templates/.claude/skills/canvas-design/canvas-fonts/Tektur-Medium.ttf +0 -0
  157. package/templates/.claude/skills/canvas-design/canvas-fonts/Tektur-OFL.txt +93 -0
  158. package/templates/.claude/skills/canvas-design/canvas-fonts/Tektur-Regular.ttf +0 -0
  159. package/templates/.claude/skills/canvas-design/canvas-fonts/YoungSerif-OFL.txt +93 -0
  160. package/templates/.claude/skills/canvas-design/canvas-fonts/YoungSerif-Regular.ttf +0 -0
  161. package/templates/.claude/skills/code-review/SKILL.md +97 -0
  162. package/templates/.claude/skills/debugging-toolkit/SKILL.md +156 -0
  163. package/templates/.claude/skills/delivery/SKILL.md +92 -0
  164. package/templates/.claude/skills/discover-security/SKILL.md +86 -0
  165. package/templates/.claude/skills/docker-packaging/SKILL.md +60 -0
  166. package/templates/.claude/skills/docs-manager/SKILL.md +465 -0
  167. package/templates/.claude/skills/docs-manager/init-project-docs.sh +70 -0
  168. package/templates/.claude/skills/docs-manager/templates/README.md.template +50 -0
  169. package/templates/.claude/skills/docs-manager/templates/agent-roles.md.template +24 -0
  170. package/templates/.claude/skills/docs-manager/templates/coding-conventions.md.template +28 -0
  171. package/templates/.claude/skills/docs-manager/templates/memory.md.template +30 -0
  172. package/templates/.claude/skills/docs-manager/templates/onboarding.md.template +20 -0
  173. package/templates/.claude/skills/docs-manager/templates/project.md.template +26 -0
  174. package/templates/.claude/skills/docs-quality/SKILL.md +148 -0
  175. package/templates/.claude/skills/docx/LICENSE.txt +30 -0
  176. package/templates/.claude/skills/docx/SKILL.md +197 -0
  177. package/templates/.claude/skills/docx/docx-js.md +350 -0
  178. package/templates/.claude/skills/docx/ooxml.md +610 -0
  179. package/templates/.claude/skills/docx/scripts/__init__.py +1 -0
  180. package/templates/.claude/skills/docx/scripts/document.py +1276 -0
  181. package/templates/.claude/skills/docx/scripts/templates/comments.xml +3 -0
  182. package/templates/.claude/skills/docx/scripts/templates/commentsExtended.xml +3 -0
  183. package/templates/.claude/skills/docx/scripts/templates/commentsExtensible.xml +3 -0
  184. package/templates/.claude/skills/docx/scripts/templates/commentsIds.xml +3 -0
  185. package/templates/.claude/skills/docx/scripts/templates/people.xml +3 -0
  186. package/templates/.claude/skills/docx/scripts/utilities.py +374 -0
  187. package/templates/.claude/skills/duraone/SKILL.md +204 -0
  188. package/templates/.claude/skills/duraone/references/backend.md +636 -0
  189. package/templates/.claude/skills/duraone/references/frontend.md +1506 -0
  190. package/templates/.claude/skills/duraone/references/sql.md +631 -0
  191. package/templates/.claude/skills/duraone/references/workflow.md +520 -0
  192. package/templates/.claude/skills/executing-plans/SKILL.md +76 -0
  193. package/templates/.claude/skills/file-organizer/SKILL.md +433 -0
  194. package/templates/.claude/skills/frontend/SKILL.md +26 -0
  195. package/templates/.claude/skills/frontend-design/LICENSE.txt +177 -0
  196. package/templates/.claude/skills/frontend-design/SKILL.md +42 -0
  197. package/templates/.claude/skills/frontend-vue/SKILL.md +127 -0
  198. package/templates/.claude/skills/frontend-vue/components/Control/Box.vue +137 -0
  199. package/templates/.claude/skills/frontend-vue/components/Control/Button.vue +93 -0
  200. package/templates/.claude/skills/frontend-vue/components/Control/ButtonBar.vue +29 -0
  201. package/templates/.claude/skills/frontend-vue/components/Control/ButtonFloat.vue +62 -0
  202. package/templates/.claude/skills/frontend-vue/components/Control/CheckButton.vue +75 -0
  203. package/templates/.claude/skills/frontend-vue/components/Control/Checkbox.vue +58 -0
  204. package/templates/.claude/skills/frontend-vue/components/Control/Datetime.vue +148 -0
  205. package/templates/.claude/skills/frontend-vue/components/Control/Dropdownlist.vue +156 -0
  206. package/templates/.claude/skills/frontend-vue/components/Control/Input.vue +106 -0
  207. package/templates/.claude/skills/frontend-vue/components/Control/Label.vue +38 -0
  208. package/templates/.claude/skills/frontend-vue/components/Control/Master/BoxColumn.vue +24 -0
  209. package/templates/.claude/skills/frontend-vue/components/Control/Popup/Confirm.vue +33 -0
  210. package/templates/.claude/skills/frontend-vue/components/Control/Popup/Info.vue +32 -0
  211. package/templates/.claude/skills/frontend-vue/components/Control/Popup/ModalInfo.vue +39 -0
  212. package/templates/.claude/skills/frontend-vue/components/Control/Popup/Reject.vue +64 -0
  213. package/templates/.claude/skills/frontend-vue/components/Control/Tag.vue +82 -0
  214. package/templates/.claude/skills/frontend-vue/components/Control/Upload.vue +61 -0
  215. package/templates/.claude/skills/frontend-vue/components/ControlMobile/Dropdownlist.vue +103 -0
  216. package/templates/.claude/skills/frontend-vue/components/ControlMobile/PagingBar.vue +108 -0
  217. package/templates/.claude/skills/frontend-vue/components/ControlMobile/UploadImage.vue +137 -0
  218. package/templates/.claude/skills/frontend-vue/components/Grid/AG.vue +806 -0
  219. package/templates/.claude/skills/frontend-vue/components/Grid/AntTable.vue +253 -0
  220. package/templates/.claude/skills/frontend-vue/components/Grid/CustomDropdownEditor.vue +43 -0
  221. package/templates/.claude/skills/frontend-vue/components/Grid/CustomDropdownEditorEnable.vue +55 -0
  222. package/templates/.claude/skills/frontend-vue/components/Grid/HtmlTable.vue +40 -0
  223. package/templates/.claude/skills/frontend-vue/components/PDFViewer.vue +25 -0
  224. package/templates/.claude/skills/frontend-vue/components/Panel/FormView.vue +309 -0
  225. package/templates/.claude/skills/frontend-vue/components/Partial/Footer.vue +23 -0
  226. package/templates/.claude/skills/frontend-vue/components/Partial/Header.vue +265 -0
  227. package/templates/.claude/skills/frontend-vue/components/Partial/Sidebar.vue +122 -0
  228. package/templates/.claude/skills/frontend-vue/components/Template.vue +16 -0
  229. package/templates/.claude/skills/frontend-vue/components/View/Form.vue +89 -0
  230. package/templates/.claude/skills/frontend-vue/composables/indexDBStore.js +140 -0
  231. package/templates/.claude/skills/frontend-vue/composables/masterApi.js +362 -0
  232. package/templates/.claude/skills/frontend-vue/composables/state.js +578 -0
  233. package/templates/.claude/skills/frontend-vue/composables/useRequest.js +221 -0
  234. package/templates/.claude/skills/frontend-vue/composables/useSession.js +179 -0
  235. package/templates/.claude/skills/frontend-vue/composables/useTranslation.js +54 -0
  236. package/templates/.claude/skills/frontend-vue/composables/useWebSocket.js +257 -0
  237. package/templates/.claude/skills/frontend-vue/composables/userObj.js +111 -0
  238. package/templates/.claude/skills/frontend-vue/composables/utils.js +322 -0
  239. package/templates/.claude/skills/frontend-vue/reference/composables-example.vue +320 -0
  240. package/templates/.claude/skills/frontend-vue/reference/form-example.vue +183 -0
  241. package/templates/.claude/skills/frontend-vue/reference/grid-example.vue +147 -0
  242. package/templates/.claude/skills/frontend-vue/reference/masterdata-example/[id].vue +106 -0
  243. package/templates/.claude/skills/frontend-vue/reference/masterdata-example/index.vue +58 -0
  244. package/templates/.claude/skills/frontend-vue/reference/popup-example.vue +159 -0
  245. package/templates/.claude/skills/pdf/LICENSE.txt +30 -0
  246. package/templates/.claude/skills/pdf/SKILL.md +294 -0
  247. package/templates/.claude/skills/pdf/forms.md +205 -0
  248. package/templates/.claude/skills/pdf/reference.md +612 -0
  249. package/templates/.claude/skills/pdf/scripts/check_bounding_boxes.py +70 -0
  250. package/templates/.claude/skills/pdf/scripts/check_bounding_boxes_test.py +226 -0
  251. package/templates/.claude/skills/pdf/scripts/check_fillable_fields.py +12 -0
  252. package/templates/.claude/skills/pdf/scripts/convert_pdf_to_images.py +35 -0
  253. package/templates/.claude/skills/pdf/scripts/create_validation_image.py +41 -0
  254. package/templates/.claude/skills/pdf/scripts/extract_form_field_info.py +152 -0
  255. package/templates/.claude/skills/pdf/scripts/fill_fillable_fields.py +114 -0
  256. package/templates/.claude/skills/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
  257. package/templates/.claude/skills/pdf-processing/SKILL.md +107 -0
  258. package/templates/.claude/skills/pdf-processing-pro/FORMS.md +610 -0
  259. package/templates/.claude/skills/pdf-processing-pro/OCR.md +137 -0
  260. package/templates/.claude/skills/pdf-processing-pro/SKILL.md +296 -0
  261. package/templates/.claude/skills/pdf-processing-pro/TABLES.md +626 -0
  262. package/templates/.claude/skills/pdf-processing-pro/scripts/analyze_form.py +307 -0
  263. package/templates/.claude/skills/postgres/SKILL.md +69 -0
  264. package/templates/.claude/skills/postgres/reference/fn_get_examples.sql +208 -0
  265. package/templates/.claude/skills/postgres/reference/fn_rpt_examples.sql +239 -0
  266. package/templates/.claude/skills/postgres/reference/utility_functions.sql +94 -0
  267. package/templates/.claude/skills/pptx/LICENSE.txt +30 -0
  268. package/templates/.claude/skills/pptx/SKILL.md +484 -0
  269. package/templates/.claude/skills/pptx/html2pptx.md +625 -0
  270. package/templates/.claude/skills/pptx/ooxml.md +427 -0
  271. package/templates/.claude/skills/pptx/scripts/html2pptx.js +979 -0
  272. package/templates/.claude/skills/pptx/scripts/inventory.py +1020 -0
  273. package/templates/.claude/skills/pptx/scripts/rearrange.py +231 -0
  274. package/templates/.claude/skills/pptx/scripts/replace.py +385 -0
  275. package/templates/.claude/skills/pptx/scripts/thumbnail.py +450 -0
  276. package/templates/.claude/skills/repo-maintenance/SKILL.md +97 -0
  277. package/templates/.claude/skills/research/EXAMPLES.md +434 -0
  278. package/templates/.claude/skills/research/REFERENCE.md +399 -0
  279. package/templates/.claude/skills/research/SKILL.md +136 -0
  280. package/templates/.claude/skills/root-cause-tracing/SKILL.md +174 -0
  281. package/templates/.claude/skills/root-cause-tracing/find-polluter.sh +63 -0
  282. package/templates/.claude/skills/sharing-skills/SKILL.md +194 -0
  283. package/templates/.claude/skills/sql-optimization-patterns/SKILL.md +493 -0
  284. package/templates/.claude/skills/subagent-driven-development/SKILL.md +189 -0
  285. package/templates/.claude/skills/systematic-debugging/CREATION-LOG.md +119 -0
  286. package/templates/.claude/skills/systematic-debugging/SKILL.md +295 -0
  287. package/templates/.claude/skills/systematic-debugging/test-academic.md +14 -0
  288. package/templates/.claude/skills/systematic-debugging/test-pressure-1.md +58 -0
  289. package/templates/.claude/skills/systematic-debugging/test-pressure-2.md +68 -0
  290. package/templates/.claude/skills/systematic-debugging/test-pressure-3.md +69 -0
  291. package/templates/.claude/skills/test-driven-development/SKILL.md +364 -0
  292. package/templates/.claude/skills/testing-anti-patterns/SKILL.md +302 -0
  293. package/templates/.claude/skills/testing-quality/SKILL.md +97 -0
  294. package/templates/.claude/skills/verification-before-completion/SKILL.md +139 -0
  295. package/templates/.claude/skills/webapp-testing/LICENSE.txt +202 -0
  296. package/templates/.claude/skills/webapp-testing/SKILL.md +96 -0
  297. package/templates/.claude/skills/webapp-testing/examples/console_logging.py +35 -0
  298. package/templates/.claude/skills/webapp-testing/examples/element_discovery.py +40 -0
  299. package/templates/.claude/skills/webapp-testing/examples/static_html_automation.py +33 -0
  300. package/templates/.claude/skills/webapp-testing/scripts/with_server.py +106 -0
  301. package/templates/.claude/ukit/index/build-index.mjs +28 -0
  302. package/templates/.claude/ukit/index/cache-utils.mjs +140 -0
  303. package/templates/.claude/ukit/index/lib/index-core.mjs +2800 -0
  304. package/templates/.claude/ukit/index/query-index.mjs +150 -0
  305. package/templates/.claude/ukit/index/refresh-index.mjs +57 -0
  306. package/templates/.claude/ukit/index/reset-auto-permissions.mjs +76 -0
  307. package/templates/.claude/ukit/index/resolve-context.mjs +279 -0
  308. package/templates/.claude/ukit/index/route-catalog.mjs +258 -0
  309. package/templates/.claude/ukit/index/route-task.mjs +1994 -0
  310. package/templates/.claude/ukit/index/triage.mjs +133 -0
  311. package/templates/.claude/ukit/index/verify-context.mjs +689 -0
  312. package/templates/.claude/ukit/runtime/compact-threshold.mjs +1013 -0
  313. package/templates/.claude/ukit/runtime/output-compression.mjs +1340 -0
  314. package/templates/.claude/ukit/runtime/reinject-context.mjs +874 -0
  315. package/templates/.claude/ukit/runtime/token-utils.mjs +500 -0
  316. package/templates/.codex/README.md +83 -0
  317. package/templates/.codex/settings.json +187 -0
  318. package/templates/.gitignore +75 -0
  319. package/templates/AGENTS.md +116 -0
  320. package/templates/CLAUDE.md +93 -0
  321. package/templates/adapter-presets/antigravity/README.md +22 -0
  322. package/templates/adapter-presets/antigravity/rules.md +49 -0
  323. package/templates/adapter-presets/claude/settings.local.json +42 -0
  324. package/templates/adapter-presets/codex/settings.local.json +6 -0
  325. package/templates/adapter-presets/opencode/opencode.template.json +1 -0
  326. package/templates/docs/BUGFIX.md +20 -0
  327. package/templates/docs/BUG_INDEX.md +12 -0
  328. package/templates/docs/BUG_METRICS.md +7 -0
  329. package/templates/docs/BUG_TEMPLATE.md +13 -0
  330. package/templates/docs/CODE_MAP.md +35 -0
  331. package/templates/docs/INSTALL.md +113 -0
  332. package/templates/docs/MEMORY.md +49 -0
  333. package/templates/docs/PROJECT.md +50 -0
  334. package/templates/docs/UKIT_USAGE_GUIDE.md +147 -0
  335. package/templates/docs/WORKLOG.md +10 -0
  336. package/templates/ukit/README.md +14 -0
  337. package/templates/ukit/storage/cache/compact-history.json +3 -0
  338. package/templates/ukit/storage/cache/compact-pressure.json +1 -0
  339. package/templates/ukit/storage/cache/output-history.json +3 -0
  340. package/templates/ukit/storage/cache/prompt-cache.json +3 -0
  341. package/templates/ukit/storage/config.json +37 -0
  342. package/templates/ukit/storage/memory/projects/.gitkeep +2 -0
  343. package/templates/ukit/storage/memory/sessions/.gitkeep +0 -0
  344. package/templates/ukit/storage/memory/user.json +5 -0
@@ -0,0 +1,237 @@
1
+ import fs from 'node:fs/promises';
2
+ import path from 'node:path';
3
+
4
+ import { getArtifactPath, getIndexDir, INDEX_ARTIFACTS } from './paths.js';
5
+
6
+ const RELATED_TEST_RELATION_TYPES = [
7
+ { key: 'components', label: 'component' },
8
+ { key: 'composables', label: 'composable' },
9
+ { key: 'services', label: 'service' },
10
+ { key: 'stores', label: 'store' },
11
+ { key: 'utils', label: 'util' },
12
+ ];
13
+ const RELATED_TEST_ARTIFACT_CACHE = new Map();
14
+ const RELATED_TEST_LOOKUP_CACHE = new Map();
15
+
16
+ export async function inferRelatedTests({
17
+ rootDir = process.cwd(),
18
+ candidateFiles = [],
19
+ limit = 5,
20
+ analogsArtifact = null,
21
+ relationsArtifact = null,
22
+ } = {}) {
23
+ const {
24
+ analogsArtifact: resolvedAnalogsArtifact,
25
+ relationsArtifact: resolvedRelationsArtifact,
26
+ analogsMap,
27
+ relationsMap,
28
+ } = await loadRelatedTestArtifacts({
29
+ rootDir,
30
+ analogsArtifact,
31
+ relationsArtifact,
32
+ });
33
+
34
+ return inferRelatedTestsFromArtifacts({
35
+ candidateFiles,
36
+ limit,
37
+ analogsMap,
38
+ relationsMap,
39
+ });
40
+ }
41
+
42
+ export async function loadRelatedTestArtifacts({
43
+ rootDir = process.cwd(),
44
+ analogsArtifact = null,
45
+ relationsArtifact = null,
46
+ } = {}) {
47
+ const absoluteRoot = path.resolve(rootDir);
48
+
49
+ if (!analogsArtifact && !relationsArtifact) {
50
+ if (!RELATED_TEST_LOOKUP_CACHE.has(absoluteRoot)) {
51
+ RELATED_TEST_LOOKUP_CACHE.set(
52
+ absoluteRoot,
53
+ Promise.all([
54
+ readArtifact(absoluteRoot, INDEX_ARTIFACTS.analogs),
55
+ readArtifact(absoluteRoot, INDEX_ARTIFACTS.relations),
56
+ ])
57
+ .then(([resolvedAnalogsArtifact, resolvedRelationsArtifact]) => ({
58
+ analogsArtifact: resolvedAnalogsArtifact,
59
+ relationsArtifact: resolvedRelationsArtifact,
60
+ ...buildRelatedTestLookups({
61
+ analogsArtifact: resolvedAnalogsArtifact,
62
+ relationsArtifact: resolvedRelationsArtifact,
63
+ }),
64
+ }))
65
+ .catch((error) => {
66
+ RELATED_TEST_LOOKUP_CACHE.delete(absoluteRoot);
67
+ throw error;
68
+ }),
69
+ );
70
+ }
71
+
72
+ return RELATED_TEST_LOOKUP_CACHE.get(absoluteRoot);
73
+ }
74
+
75
+ const [resolvedAnalogsArtifact, resolvedRelationsArtifact] = await Promise.all([
76
+ analogsArtifact ?? readArtifact(absoluteRoot, INDEX_ARTIFACTS.analogs),
77
+ relationsArtifact ?? readArtifact(absoluteRoot, INDEX_ARTIFACTS.relations),
78
+ ]);
79
+
80
+ return {
81
+ analogsArtifact: resolvedAnalogsArtifact,
82
+ relationsArtifact: resolvedRelationsArtifact,
83
+ ...buildRelatedTestLookups({
84
+ analogsArtifact: resolvedAnalogsArtifact,
85
+ relationsArtifact: resolvedRelationsArtifact,
86
+ }),
87
+ };
88
+ }
89
+
90
+ export function clearRelatedTestArtifactCache(rootDir = null) {
91
+ if (!rootDir) {
92
+ RELATED_TEST_ARTIFACT_CACHE.clear();
93
+ RELATED_TEST_LOOKUP_CACHE.clear();
94
+ return;
95
+ }
96
+
97
+ const absoluteRoot = path.resolve(rootDir);
98
+ const indexDir = getIndexDir(absoluteRoot);
99
+
100
+ for (const artifactPath of RELATED_TEST_ARTIFACT_CACHE.keys()) {
101
+ if (artifactPath.startsWith(indexDir)) {
102
+ RELATED_TEST_ARTIFACT_CACHE.delete(artifactPath);
103
+ }
104
+ }
105
+
106
+ RELATED_TEST_LOOKUP_CACHE.delete(absoluteRoot);
107
+ }
108
+
109
+ export function buildRelatedTestLookups({
110
+ analogsArtifact = { items: [] },
111
+ relationsArtifact = { items: [] },
112
+ } = {}) {
113
+ return {
114
+ analogsMap: new Map((analogsArtifact.items ?? []).map((item) => [item.filePath, item.analogs ?? []])),
115
+ relationsMap: new Map((relationsArtifact.items ?? []).map((item) => [item.filePath, item.relations ?? {}])),
116
+ };
117
+ }
118
+
119
+ export function inferRelatedTestsFromArtifacts({
120
+ candidateFiles = [],
121
+ analogsMap = new Map(),
122
+ relationsMap = new Map(),
123
+ limit = 5,
124
+ } = {}) {
125
+ const normalizedCandidates = [...new Set(
126
+ candidateFiles
127
+ .map((filePath) => String(filePath ?? '').trim())
128
+ .filter(Boolean),
129
+ )];
130
+
131
+ if (normalizedCandidates.length === 0) {
132
+ return [];
133
+ }
134
+
135
+ const suggestions = new Map();
136
+
137
+ for (const [candidateRank, candidateFile] of normalizedCandidates.entries()) {
138
+ const rankPenalty = candidateRank * 5;
139
+ const relations = relationsMap.get(candidateFile) ?? {};
140
+
141
+ addTestsFromSource({
142
+ sourceFilePath: candidateFile,
143
+ relationsMap,
144
+ suggestions,
145
+ score: 120 - rankPenalty,
146
+ reason: `direct test for ${candidateFile}`,
147
+ });
148
+
149
+ for (const siblingFile of (relations.siblings ?? []).slice(0, 3)) {
150
+ addTestsFromSource({
151
+ sourceFilePath: siblingFile,
152
+ relationsMap,
153
+ suggestions,
154
+ score: 90 - rankPenalty,
155
+ reason: `sibling test via ${siblingFile}`,
156
+ });
157
+ }
158
+
159
+ for (const analog of (analogsMap.get(candidateFile) ?? []).slice(0, 3)) {
160
+ addTestsFromSource({
161
+ sourceFilePath: analog.filePath,
162
+ relationsMap,
163
+ suggestions,
164
+ score: 80 + Math.round((Number(analog.score) || 0) * 20) - rankPenalty,
165
+ reason: `analog test via ${analog.filePath}${analog.reason ? `; ${analog.reason}` : ''}`,
166
+ });
167
+ }
168
+
169
+ for (const { key, label } of RELATED_TEST_RELATION_TYPES) {
170
+ for (const relatedFile of (relations[key] ?? []).slice(0, 2)) {
171
+ addTestsFromSource({
172
+ sourceFilePath: relatedFile,
173
+ relationsMap,
174
+ suggestions,
175
+ score: 60 - rankPenalty,
176
+ reason: `${label} test via ${relatedFile}`,
177
+ });
178
+ }
179
+ }
180
+ }
181
+
182
+ return [...suggestions.values()]
183
+ .sort((a, b) => b.score - a.score || a.filePath.localeCompare(b.filePath))
184
+ .slice(0, Math.max(limit, 0))
185
+ .map((suggestion) => ({
186
+ filePath: suggestion.filePath,
187
+ score: suggestion.score,
188
+ reason: suggestion.reasons.join('; '),
189
+ }));
190
+ }
191
+
192
+ async function readArtifact(rootDir, artifactName) {
193
+ const artifactPath = getArtifactPath(path.resolve(rootDir), artifactName);
194
+
195
+ if (!RELATED_TEST_ARTIFACT_CACHE.has(artifactPath)) {
196
+ RELATED_TEST_ARTIFACT_CACHE.set(
197
+ artifactPath,
198
+ fs.readFile(artifactPath, 'utf8')
199
+ .then((content) => JSON.parse(content))
200
+ .catch(() => {
201
+ RELATED_TEST_ARTIFACT_CACHE.delete(artifactPath);
202
+ return { items: [] };
203
+ }),
204
+ );
205
+ }
206
+
207
+ return RELATED_TEST_ARTIFACT_CACHE.get(artifactPath);
208
+ }
209
+
210
+ function addTestsFromSource({
211
+ sourceFilePath,
212
+ relationsMap,
213
+ suggestions,
214
+ score,
215
+ reason,
216
+ }) {
217
+ if (!sourceFilePath) {
218
+ return;
219
+ }
220
+
221
+ const tests = relationsMap.get(sourceFilePath)?.tests ?? [];
222
+ for (const testFilePath of tests) {
223
+ if (!suggestions.has(testFilePath)) {
224
+ suggestions.set(testFilePath, {
225
+ filePath: testFilePath,
226
+ score,
227
+ reasons: [],
228
+ });
229
+ }
230
+
231
+ const suggestion = suggestions.get(testFilePath);
232
+ suggestion.score = Math.max(suggestion.score, score);
233
+ if (!suggestion.reasons.includes(reason)) {
234
+ suggestion.reasons.push(reason);
235
+ }
236
+ }
237
+ }
@@ -0,0 +1,345 @@
1
+ import { queryCodeIndex } from './queryIndex.js';
2
+ import { inferRelatedTestsFromArtifacts, loadRelatedTestArtifacts } from './relatedTests.js';
3
+
4
+ const TASK_TYPE_BUDGETS = {
5
+ trivial: { minFiles: 1, maxFiles: 2 },
6
+ simple: { minFiles: 2, maxFiles: 5 },
7
+ 'non-trivial': { minFiles: 4, maxFiles: 8 },
8
+ };
9
+
10
+ const TRIVIAL_SIGNALS = ['typo', 'label', 'text', 'rename', 'color', 'spacing', 'toggle', 'config', 'comment'];
11
+ const RISKY_SIGNALS = [
12
+ 'auth', 'security', 'migration', 'uninstall', 'password', 'token', 'permission',
13
+ 'delete all', 'drop table', 'race', 'flaky', 'intermittent', 'timeout', 'deadlock',
14
+ 'core', 'shared', 'runtime',
15
+ ];
16
+ const EXPANSION_SIGNALS = [
17
+ 'similar',
18
+ 'same as',
19
+ 'like',
20
+ 'pattern',
21
+ 'clone',
22
+ 'analog',
23
+ 'follow',
24
+ 'example',
25
+ 'search',
26
+ 'find',
27
+ 'where',
28
+ 'which file',
29
+ 'across',
30
+ 'everywhere',
31
+ 'multiple',
32
+ 'project',
33
+ 'repo',
34
+ 'workspace',
35
+ 'all ',
36
+ 'every ',
37
+ ];
38
+
39
+ export async function resolveContext({
40
+ rootDir = process.cwd(),
41
+ intent = '',
42
+ targetFile = null,
43
+ taskType = null,
44
+ } = {}) {
45
+ const classifiedType = taskType ?? classifyTask(intent);
46
+ const budget = TASK_TYPE_BUDGETS[classifiedType] ?? TASK_TYPE_BUDGETS.simple;
47
+ const queryWasUsed = shouldQueryIndex({ intent, targetFile, taskType: classifiedType });
48
+
49
+ const result = {
50
+ taskType: classifiedType,
51
+ contextBudget: { ...budget, taskType: classifiedType },
52
+ primaryTargets: [],
53
+ analogFiles: [],
54
+ sharedAbstractions: [],
55
+ relatedTests: [],
56
+ styleFiles: [],
57
+ explanations: {
58
+ primaryTargets: [],
59
+ analogFiles: [],
60
+ sharedAbstractions: [],
61
+ relatedTests: [],
62
+ styleFiles: [],
63
+ },
64
+ };
65
+
66
+ // Step 1: Find primary targets
67
+ if (targetFile) {
68
+ addExplainedFile(result, 'primaryTargets', targetFile, 'explicit target');
69
+ }
70
+
71
+ if (queryWasUsed) {
72
+ const queryResults = await queryCodeIndex({ rootDir, query: intent, limit: budget.maxFiles });
73
+ for (const qr of queryResults) {
74
+ addExplainedFile(result, 'primaryTargets', qr.filePath, buildQueryReason(qr));
75
+ }
76
+ }
77
+
78
+ if (result.primaryTargets.length === 0) {
79
+ return result;
80
+ }
81
+
82
+ const { relationsArtifact, relationsMap } = await loadRelatedTestArtifacts({
83
+ rootDir,
84
+ analogsArtifact: { items: [] },
85
+ });
86
+
87
+ // Step 2: Get relations for primary targets to find shared abstractions
88
+ for (const primary of result.primaryTargets) {
89
+ const rels = relationsMap.get(primary) ?? {};
90
+
91
+ // Related tests
92
+ for (const testPath of rels.tests ?? []) {
93
+ addExplainedFile(result, 'relatedTests', testPath, `test linked to ${primary}`);
94
+ }
95
+
96
+ for (const stylePath of rels.styles ?? []) {
97
+ addExplainedFile(result, 'styleFiles', stylePath, `style imported by ${primary}`);
98
+ }
99
+ }
100
+
101
+ const preferLocalizedDirectContext = shouldPreferLocalizedDirectTargetContext({
102
+ result,
103
+ taskType: classifiedType,
104
+ targetFile,
105
+ queryWasUsed,
106
+ });
107
+
108
+ if (!preferLocalizedDirectContext) {
109
+ for (const primary of result.primaryTargets) {
110
+ const rels = relationsMap.get(primary) ?? {};
111
+
112
+ for (const absType of ['composables', 'utils', 'services']) {
113
+ for (const absPath of rels[absType] ?? []) {
114
+ if (result.primaryTargets.includes(absPath)) {
115
+ continue;
116
+ }
117
+ addExplainedFile(result, 'sharedAbstractions', absPath, `related ${absType} used by ${primary}`);
118
+ }
119
+ }
120
+ }
121
+
122
+ const inferredTests = inferRelatedTestsFromArtifacts({
123
+ candidateFiles: result.primaryTargets,
124
+ analogsMap: new Map(),
125
+ relationsMap,
126
+ limit: budget.maxFiles,
127
+ });
128
+
129
+ for (const inferredTest of inferredTests) {
130
+ addExplainedFile(result, 'relatedTests', inferredTest.filePath, inferredTest.reason);
131
+ }
132
+ }
133
+
134
+ if (shouldLoadAnalogArtifacts({
135
+ result,
136
+ taskType: classifiedType,
137
+ targetFile,
138
+ maxFiles: budget.maxFiles,
139
+ queryWasUsed,
140
+ preferLocalizedDirectContext,
141
+ })) {
142
+ const { analogsMap } = await loadRelatedTestArtifacts({
143
+ rootDir,
144
+ relationsArtifact,
145
+ });
146
+
147
+ // Step 3: Find analog files for primary targets
148
+ for (const primary of result.primaryTargets.slice(0, 2)) {
149
+ const analogs = analogsMap.get(primary) ?? [];
150
+ for (const analog of analogs.slice(0, 3)) {
151
+ if (result.primaryTargets.includes(analog.filePath)) {
152
+ continue;
153
+ }
154
+ addExplainedFile(
155
+ result,
156
+ 'analogFiles',
157
+ analog.filePath,
158
+ `analog of ${primary}${analog.reason ? `; ${analog.reason}` : ''}`,
159
+ );
160
+ }
161
+ }
162
+
163
+ const inferredAnalogTests = inferRelatedTestsFromArtifacts({
164
+ candidateFiles: result.primaryTargets,
165
+ analogsMap,
166
+ relationsMap,
167
+ limit: budget.maxFiles,
168
+ });
169
+
170
+ for (const inferredTest of inferredAnalogTests) {
171
+ addExplainedFile(result, 'relatedTests', inferredTest.filePath, inferredTest.reason);
172
+ }
173
+ }
174
+
175
+ enforceContextBudget(result, budget.maxFiles);
176
+ includeDirectTestTargets(result);
177
+
178
+ return result;
179
+ }
180
+
181
+ export function classifyTask(intent) {
182
+ const lower = intent.toLowerCase();
183
+
184
+ // Check risky first (highest priority)
185
+ if (RISKY_SIGNALS.some((signal) => lower.includes(signal))) {
186
+ return 'non-trivial';
187
+ }
188
+
189
+ // Check trivial
190
+ if (TRIVIAL_SIGNALS.some((signal) => lower.includes(signal))) {
191
+ // But if intent also mentions multiple files or broad scope, upgrade
192
+ if (lower.includes('all ') || lower.includes('every ') || lower.includes('multiple')) {
193
+ return 'simple';
194
+ }
195
+ return 'trivial';
196
+ }
197
+
198
+ // Default to simple
199
+ return 'simple';
200
+ }
201
+
202
+ function shouldQueryIndex({
203
+ intent = '',
204
+ targetFile = null,
205
+ taskType = 'simple',
206
+ } = {}) {
207
+ const normalizedIntent = String(intent || '').trim().toLowerCase();
208
+ if (!normalizedIntent) {
209
+ return false;
210
+ }
211
+
212
+ if (!targetFile) {
213
+ return true;
214
+ }
215
+
216
+ if (taskType === 'non-trivial') {
217
+ return true;
218
+ }
219
+
220
+ return EXPANSION_SIGNALS.some((signal) => normalizedIntent.includes(signal));
221
+ }
222
+ function enforceContextBudget(result, maxFiles) {
223
+ const prioritizedGroups = [
224
+ 'primaryTargets',
225
+ 'relatedTests',
226
+ 'styleFiles',
227
+ 'sharedAbstractions',
228
+ 'analogFiles',
229
+ ];
230
+ const kept = new Set();
231
+
232
+ for (const group of prioritizedGroups) {
233
+ const nextItems = [];
234
+ for (const item of result[group] ?? []) {
235
+ if (kept.size >= maxFiles) {
236
+ break;
237
+ }
238
+ if (kept.has(item)) {
239
+ continue;
240
+ }
241
+ kept.add(item);
242
+ nextItems.push(item);
243
+ }
244
+ result[group] = nextItems;
245
+ result.explanations[group] = (result.explanations[group] ?? [])
246
+ .filter((entry) => nextItems.includes(entry.filePath));
247
+ }
248
+ }
249
+
250
+ function addExplainedFile(result, group, filePath, reason) {
251
+ if (!result[group].includes(filePath)) {
252
+ result[group].push(filePath);
253
+ }
254
+
255
+ const explanations = result.explanations[group] ?? [];
256
+ const existing = explanations.find((entry) => entry.filePath === filePath);
257
+ if (!existing) {
258
+ explanations.push({ filePath, reason });
259
+ result.explanations[group] = explanations;
260
+ return;
261
+ }
262
+
263
+ if (!existing.reason.includes(reason)) {
264
+ existing.reason = `${existing.reason}; ${reason}`;
265
+ }
266
+ }
267
+
268
+ function shouldLoadAnalogArtifacts({
269
+ result,
270
+ taskType = 'simple',
271
+ targetFile = null,
272
+ maxFiles = 5,
273
+ queryWasUsed = false,
274
+ preferLocalizedDirectContext = false,
275
+ } = {}) {
276
+ const selectedCount = countSelectedFiles(result);
277
+ if (selectedCount >= maxFiles) {
278
+ return false;
279
+ }
280
+
281
+ if (preferLocalizedDirectContext) {
282
+ return false;
283
+ }
284
+
285
+ if (
286
+ taskType === 'trivial'
287
+ && targetFile
288
+ && !queryWasUsed
289
+ && (result.relatedTests?.length ?? 0) > 0
290
+ ) {
291
+ return false;
292
+ }
293
+
294
+ return true;
295
+ }
296
+
297
+ function shouldPreferLocalizedDirectTargetContext({
298
+ result,
299
+ taskType = 'simple',
300
+ targetFile = null,
301
+ queryWasUsed = false,
302
+ } = {}) {
303
+ if (queryWasUsed || !targetFile) {
304
+ return false;
305
+ }
306
+
307
+ if (taskType !== 'simple' && taskType !== 'trivial') {
308
+ return false;
309
+ }
310
+
311
+ return isTestLikeFile(targetFile)
312
+ || (result.relatedTests?.length ?? 0) > 0;
313
+ }
314
+
315
+ function countSelectedFiles(result) {
316
+ const selected = new Set();
317
+ for (const group of ['primaryTargets', 'relatedTests', 'styleFiles', 'sharedAbstractions', 'analogFiles']) {
318
+ for (const filePath of result[group] ?? []) {
319
+ selected.add(filePath);
320
+ }
321
+ }
322
+ return selected.size;
323
+ }
324
+
325
+ function buildQueryReason(queryResult) {
326
+ const reasons = queryResult.reasons ?? [];
327
+ if (reasons.length === 0) {
328
+ return 'query match';
329
+ }
330
+ return `query match: ${reasons.join(', ')}`;
331
+ }
332
+
333
+ function includeDirectTestTargets(result) {
334
+ for (const primaryTarget of result.primaryTargets ?? []) {
335
+ if (!isTestLikeFile(primaryTarget)) {
336
+ continue;
337
+ }
338
+ addExplainedFile(result, 'relatedTests', primaryTarget, `direct test target ${primaryTarget}`);
339
+ }
340
+ }
341
+
342
+ function isTestLikeFile(filePath) {
343
+ return /\.(test|spec)\.[a-z0-9]+$/i.test(filePath)
344
+ || /(^|\/)(?:__tests__|tests?|specs?)\//i.test(filePath);
345
+ }