xtrm-tools 0.7.2 → 0.7.4

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 (529) hide show
  1. package/.xtrm/config/pi/extensions/xtrm-ui/format.ts +189 -0
  2. package/.xtrm/config/pi/extensions/xtrm-ui/index.ts +86 -18
  3. package/.xtrm/config/pi/extensions/xtrm-ui/package.json +16 -5
  4. package/.xtrm/config/pi/extensions/xtrm-ui/themes/pidex-dark.json +7 -3
  5. package/.xtrm/config/pi/extensions/xtrm-ui/themes/pidex-light.json +4 -0
  6. package/.xtrm/hooks/specialists/specialists-complete.mjs +70 -0
  7. package/.xtrm/hooks/specialists/specialists-session-start.mjs +105 -0
  8. package/.xtrm/registry.json +600 -344
  9. package/.xtrm/skills/default/README.txt +31 -0
  10. package/.xtrm/skills/default/clean-code/SKILL.md +201 -0
  11. package/.xtrm/skills/default/creating-service-skills/SKILL.md +433 -0
  12. package/.xtrm/skills/default/creating-service-skills/references/script_quality_standards.md +425 -0
  13. package/.xtrm/skills/default/creating-service-skills/references/service_skill_system_guide.md +278 -0
  14. package/.xtrm/skills/default/creating-service-skills/scripts/bootstrap.py +326 -0
  15. package/.xtrm/skills/default/creating-service-skills/scripts/deep_dive.py +304 -0
  16. package/.xtrm/skills/default/creating-service-skills/scripts/scaffolder.py +482 -0
  17. package/.xtrm/skills/default/deepwiki/SKILL.md +50 -0
  18. package/.xtrm/skills/default/delegating/SKILL.md +196 -0
  19. package/.xtrm/skills/default/delegating/config.yaml +210 -0
  20. package/.xtrm/skills/default/delegating/references/orchestration-protocols.md +41 -0
  21. package/.xtrm/skills/default/documenting/CHANGELOG.md +23 -0
  22. package/.xtrm/skills/default/documenting/README.md +148 -0
  23. package/.xtrm/skills/default/documenting/SKILL.md +113 -0
  24. package/.xtrm/skills/default/documenting/examples/example_pattern.md +70 -0
  25. package/.xtrm/skills/default/documenting/examples/example_reference.md +70 -0
  26. package/.xtrm/skills/default/documenting/examples/example_ssot_analytics.md +64 -0
  27. package/.xtrm/skills/default/documenting/examples/example_workflow.md +141 -0
  28. package/.xtrm/skills/default/documenting/references/changelog-format.md +97 -0
  29. package/.xtrm/skills/default/documenting/references/metadata-schema.md +136 -0
  30. package/.xtrm/skills/default/documenting/references/taxonomy.md +81 -0
  31. package/.xtrm/skills/default/documenting/references/versioning-rules.md +78 -0
  32. package/.xtrm/skills/default/documenting/scripts/bump_version.sh +60 -0
  33. package/.xtrm/skills/default/documenting/scripts/changelog/__init__.py +0 -0
  34. package/.xtrm/skills/default/documenting/scripts/changelog/add_entry.py +216 -0
  35. package/.xtrm/skills/default/documenting/scripts/changelog/bump_release.py +117 -0
  36. package/.xtrm/skills/default/documenting/scripts/changelog/init_changelog.py +54 -0
  37. package/.xtrm/skills/default/documenting/scripts/changelog/validate_changelog.py +128 -0
  38. package/.xtrm/skills/default/documenting/scripts/drift_detector.py +266 -0
  39. package/.xtrm/skills/default/documenting/scripts/generate_template.py +311 -0
  40. package/.xtrm/skills/default/documenting/scripts/list_by_category.sh +84 -0
  41. package/.xtrm/skills/default/documenting/scripts/orchestrator.py +255 -0
  42. package/.xtrm/skills/default/documenting/scripts/validate_metadata.py +242 -0
  43. package/.xtrm/skills/default/documenting/templates/CHANGELOG.md.template +13 -0
  44. package/.xtrm/skills/default/find-docs/SKILL.md +175 -0
  45. package/.xtrm/skills/default/find-skills/SKILL.md +133 -0
  46. package/.xtrm/skills/default/github-search/SKILL.md +49 -0
  47. package/.xtrm/skills/default/gitnexus-debugging/SKILL.md +89 -0
  48. package/.xtrm/skills/default/gitnexus-impact-analysis/SKILL.md +97 -0
  49. package/.xtrm/skills/default/gitnexus-pr-review/SKILL.md +163 -0
  50. package/.xtrm/skills/default/gitnexus-refactoring/SKILL.md +121 -0
  51. package/.xtrm/skills/default/hook-development/SKILL.md +797 -0
  52. package/.xtrm/skills/default/hook-development/examples/load-context.sh +55 -0
  53. package/.xtrm/skills/default/hook-development/examples/quality-check.js +1168 -0
  54. package/.xtrm/skills/default/hook-development/examples/validate-bash.sh +43 -0
  55. package/.xtrm/skills/default/hook-development/examples/validate-write.sh +38 -0
  56. package/.xtrm/skills/default/hook-development/references/advanced.md +527 -0
  57. package/.xtrm/skills/default/hook-development/references/migration.md +369 -0
  58. package/.xtrm/skills/default/hook-development/references/patterns.md +412 -0
  59. package/.xtrm/skills/default/hook-development/scripts/README.md +164 -0
  60. package/.xtrm/skills/default/hook-development/scripts/hook-linter.sh +153 -0
  61. package/.xtrm/skills/default/hook-development/scripts/test-hook.sh +252 -0
  62. package/.xtrm/skills/default/hook-development/scripts/validate-hook-schema.sh +159 -0
  63. package/.xtrm/skills/default/init-session/SKILL.md +69 -0
  64. package/.xtrm/skills/default/last30days/SKILL.md +881 -0
  65. package/.xtrm/skills/default/last30days/scripts/briefing.py +260 -0
  66. package/.xtrm/skills/default/last30days/scripts/evaluate-synthesis.py +120 -0
  67. package/.xtrm/skills/default/last30days/scripts/evaluate_search_quality.py +641 -0
  68. package/.xtrm/skills/default/last30days/scripts/generate-synthesis-inputs.py +53 -0
  69. package/.xtrm/skills/default/last30days/scripts/last30days.py +2137 -0
  70. package/.xtrm/skills/default/last30days/scripts/lib/__init__.py +1 -0
  71. package/.xtrm/skills/default/last30days/scripts/lib/bird_x.py +458 -0
  72. package/.xtrm/skills/default/last30days/scripts/lib/bluesky.py +225 -0
  73. package/.xtrm/skills/default/last30days/scripts/lib/brave_search.py +329 -0
  74. package/.xtrm/skills/default/last30days/scripts/lib/cache.py +165 -0
  75. package/.xtrm/skills/default/last30days/scripts/lib/chrome_cookies.py +265 -0
  76. package/.xtrm/skills/default/last30days/scripts/lib/cookie_extract.py +295 -0
  77. package/.xtrm/skills/default/last30days/scripts/lib/dates.py +124 -0
  78. package/.xtrm/skills/default/last30days/scripts/lib/dedupe.py +290 -0
  79. package/.xtrm/skills/default/last30days/scripts/lib/entity_extract.py +127 -0
  80. package/.xtrm/skills/default/last30days/scripts/lib/env.py +807 -0
  81. package/.xtrm/skills/default/last30days/scripts/lib/exa_search.py +176 -0
  82. package/.xtrm/skills/default/last30days/scripts/lib/hackernews.py +266 -0
  83. package/.xtrm/skills/default/last30days/scripts/lib/http.py +174 -0
  84. package/.xtrm/skills/default/last30days/scripts/lib/instagram.py +365 -0
  85. package/.xtrm/skills/default/last30days/scripts/lib/models.py +221 -0
  86. package/.xtrm/skills/default/last30days/scripts/lib/normalize.py +489 -0
  87. package/.xtrm/skills/default/last30days/scripts/lib/openai_reddit.py +631 -0
  88. package/.xtrm/skills/default/last30days/scripts/lib/openrouter_search.py +216 -0
  89. package/.xtrm/skills/default/last30days/scripts/lib/parallel_search.py +139 -0
  90. package/.xtrm/skills/default/last30days/scripts/lib/polymarket.py +580 -0
  91. package/.xtrm/skills/default/last30days/scripts/lib/quality_nudge.py +201 -0
  92. package/.xtrm/skills/default/last30days/scripts/lib/query.py +117 -0
  93. package/.xtrm/skills/default/last30days/scripts/lib/query_type.py +111 -0
  94. package/.xtrm/skills/default/last30days/scripts/lib/reddit.py +617 -0
  95. package/.xtrm/skills/default/last30days/scripts/lib/reddit_enrich.py +325 -0
  96. package/.xtrm/skills/default/last30days/scripts/lib/reddit_public.py +259 -0
  97. package/.xtrm/skills/default/last30days/scripts/lib/relevance.py +148 -0
  98. package/.xtrm/skills/default/last30days/scripts/lib/render.py +1018 -0
  99. package/.xtrm/skills/default/last30days/scripts/lib/safari_cookies.py +182 -0
  100. package/.xtrm/skills/default/last30days/scripts/lib/schema.py +843 -0
  101. package/.xtrm/skills/default/last30days/scripts/lib/score.py +775 -0
  102. package/.xtrm/skills/default/last30days/scripts/lib/scrapecreators_x.py +182 -0
  103. package/.xtrm/skills/default/last30days/scripts/lib/setup_wizard.py +186 -0
  104. package/.xtrm/skills/default/last30days/scripts/lib/tiktok.py +349 -0
  105. package/.xtrm/skills/default/last30days/scripts/lib/truthsocial.py +183 -0
  106. package/.xtrm/skills/default/last30days/scripts/lib/ui.py +620 -0
  107. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/LICENSE +21 -0
  108. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/bird-search.mjs +134 -0
  109. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/cookies.js +191 -0
  110. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/features.json +17 -0
  111. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/paginate-cursor.js +37 -0
  112. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/query-ids.json +20 -0
  113. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/runtime-features.js +151 -0
  114. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/runtime-query-ids.js +264 -0
  115. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-base.js +129 -0
  116. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-constants.js +50 -0
  117. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-features.js +347 -0
  118. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-search.js +157 -0
  119. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-types.js +2 -0
  120. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/lib/twitter-client-utils.js +511 -0
  121. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/LICENSE +22 -0
  122. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/README.md +29 -0
  123. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/index.d.ts +3 -0
  124. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/index.d.ts.map +1 -0
  125. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/index.js +2 -0
  126. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/index.js.map +1 -0
  127. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chrome.d.ts +8 -0
  128. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chrome.d.ts.map +1 -0
  129. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chrome.js +27 -0
  130. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chrome.js.map +1 -0
  131. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/crypto.d.ts +11 -0
  132. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/crypto.d.ts.map +1 -0
  133. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/crypto.js +100 -0
  134. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/crypto.js.map +1 -0
  135. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/linuxKeyring.d.ts +25 -0
  136. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/linuxKeyring.d.ts.map +1 -0
  137. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/linuxKeyring.js +104 -0
  138. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/linuxKeyring.js.map +1 -0
  139. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/shared.d.ts +10 -0
  140. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/shared.d.ts.map +1 -0
  141. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/shared.js +293 -0
  142. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/shared.js.map +1 -0
  143. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/windowsDpapi.d.ts +10 -0
  144. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/windowsDpapi.d.ts.map +1 -0
  145. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/windowsDpapi.js +26 -0
  146. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqlite/windowsDpapi.js.map +1 -0
  147. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteLinux.d.ts +7 -0
  148. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteLinux.d.ts.map +1 -0
  149. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteLinux.js +51 -0
  150. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteLinux.js.map +1 -0
  151. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteMac.d.ts +7 -0
  152. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteMac.d.ts.map +1 -0
  153. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteMac.js +60 -0
  154. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteMac.js.map +1 -0
  155. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteWindows.d.ts +7 -0
  156. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteWindows.d.ts.map +1 -0
  157. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteWindows.js +38 -0
  158. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromeSqliteWindows.js.map +1 -0
  159. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/linuxPaths.d.ts +5 -0
  160. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/linuxPaths.d.ts.map +1 -0
  161. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/linuxPaths.js +33 -0
  162. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/linuxPaths.js.map +1 -0
  163. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/macosKeychain.d.ts +24 -0
  164. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/macosKeychain.d.ts.map +1 -0
  165. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/macosKeychain.js +30 -0
  166. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/macosKeychain.js.map +1 -0
  167. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/paths.d.ts +11 -0
  168. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/paths.d.ts.map +1 -0
  169. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/paths.js +43 -0
  170. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/paths.js.map +1 -0
  171. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsMasterKey.d.ts +8 -0
  172. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsMasterKey.d.ts.map +1 -0
  173. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsMasterKey.js +41 -0
  174. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsMasterKey.js.map +1 -0
  175. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsPaths.d.ts +8 -0
  176. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsPaths.d.ts.map +1 -0
  177. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsPaths.js +53 -0
  178. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/chromium/windowsPaths.js.map +1 -0
  179. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edge.d.ts +8 -0
  180. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edge.d.ts.map +1 -0
  181. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edge.js +27 -0
  182. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edge.js.map +1 -0
  183. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteLinux.d.ts +7 -0
  184. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteLinux.d.ts.map +1 -0
  185. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteLinux.js +53 -0
  186. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteLinux.js.map +1 -0
  187. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteMac.d.ts +8 -0
  188. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteMac.d.ts.map +1 -0
  189. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteMac.js +60 -0
  190. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteMac.js.map +1 -0
  191. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteWindows.d.ts +7 -0
  192. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteWindows.d.ts.map +1 -0
  193. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteWindows.js +38 -0
  194. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/edgeSqliteWindows.js.map +1 -0
  195. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/firefoxSqlite.d.ts +6 -0
  196. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/firefoxSqlite.d.ts.map +1 -0
  197. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/firefoxSqlite.js +257 -0
  198. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/firefoxSqlite.js.map +1 -0
  199. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/inline.d.ts +8 -0
  200. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/inline.d.ts.map +1 -0
  201. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/inline.js +71 -0
  202. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/inline.js.map +1 -0
  203. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/safariBinaryCookies.d.ts +6 -0
  204. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/safariBinaryCookies.d.ts.map +1 -0
  205. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/safariBinaryCookies.js +173 -0
  206. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/providers/safariBinaryCookies.js.map +1 -0
  207. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/public.d.ts +26 -0
  208. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/public.d.ts.map +1 -0
  209. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/public.js +195 -0
  210. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/public.js.map +1 -0
  211. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/types.d.ts +121 -0
  212. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/types.d.ts.map +1 -0
  213. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/types.js +2 -0
  214. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/types.js.map +1 -0
  215. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/base64.d.ts +2 -0
  216. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/base64.d.ts.map +1 -0
  217. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/base64.js +18 -0
  218. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/base64.js.map +1 -0
  219. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/exec.d.ts +8 -0
  220. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/exec.d.ts.map +1 -0
  221. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/exec.js +110 -0
  222. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/exec.js.map +1 -0
  223. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/expire.d.ts +2 -0
  224. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/expire.d.ts.map +1 -0
  225. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/expire.js +32 -0
  226. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/expire.js.map +1 -0
  227. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/fs.d.ts +2 -0
  228. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/fs.d.ts.map +1 -0
  229. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/fs.js +13 -0
  230. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/fs.js.map +1 -0
  231. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/hostMatch.d.ts +2 -0
  232. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/hostMatch.d.ts.map +1 -0
  233. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/hostMatch.js +7 -0
  234. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/hostMatch.js.map +1 -0
  235. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/nodeSqlite.d.ts +5 -0
  236. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/nodeSqlite.d.ts.map +1 -0
  237. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/nodeSqlite.js +58 -0
  238. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/nodeSqlite.js.map +1 -0
  239. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/origins.d.ts +2 -0
  240. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/origins.d.ts.map +1 -0
  241. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/origins.js +27 -0
  242. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/origins.js.map +1 -0
  243. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/runtime.d.ts +2 -0
  244. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/runtime.d.ts.map +1 -0
  245. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/runtime.js +8 -0
  246. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/dist/util/runtime.js.map +1 -0
  247. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/node_modules/@steipete/sweet-cookie/package.json +40 -0
  248. package/.xtrm/skills/default/last30days/scripts/lib/vendor/bird-search/package.json +13 -0
  249. package/.xtrm/skills/default/last30days/scripts/lib/websearch.py +401 -0
  250. package/.xtrm/skills/default/last30days/scripts/lib/xai_x.py +217 -0
  251. package/.xtrm/skills/default/last30days/scripts/lib/xiaohongshu_api.py +162 -0
  252. package/.xtrm/skills/default/last30days/scripts/lib/youtube_yt.py +538 -0
  253. package/.xtrm/skills/default/last30days/scripts/store.py +654 -0
  254. package/.xtrm/skills/default/last30days/scripts/sync.sh +50 -0
  255. package/.xtrm/skills/default/last30days/scripts/test-v1-vs-v2.sh +219 -0
  256. package/.xtrm/skills/default/last30days/scripts/watchlist.py +329 -0
  257. package/.xtrm/skills/default/planning/SKILL.md +405 -0
  258. package/.xtrm/skills/default/planning/evals/evals.json +19 -0
  259. package/.xtrm/skills/default/prompt-improving/README.md +162 -0
  260. package/.xtrm/skills/default/prompt-improving/SKILL.md +74 -0
  261. package/.xtrm/skills/default/prompt-improving/references/analysis_commands.md +24 -0
  262. package/.xtrm/skills/default/prompt-improving/references/chain_of_thought.md +24 -0
  263. package/.xtrm/skills/default/prompt-improving/references/mcp_definitions.md +20 -0
  264. package/.xtrm/skills/default/prompt-improving/references/multishot.md +23 -0
  265. package/.xtrm/skills/default/prompt-improving/references/xml_core.md +60 -0
  266. package/.xtrm/skills/default/quality-gates/.claude/hooks/hook-config.json +66 -0
  267. package/.xtrm/skills/default/quality-gates/.claude/hooks/quality-check.cjs +1286 -0
  268. package/.xtrm/skills/default/quality-gates/.claude/hooks/quality-check.py +334 -0
  269. package/.xtrm/skills/default/quality-gates/.claude/settings.json +3 -0
  270. package/.xtrm/skills/default/quality-gates/.claude/skills/using-quality-gates/SKILL.md +254 -0
  271. package/.xtrm/skills/default/quality-gates/README.md +109 -0
  272. package/.xtrm/skills/default/quality-gates/evals/evals.json +181 -0
  273. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/FINAL-EVAL-SUMMARY.md +75 -0
  274. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/edge-case-auto-fix-verification/with_skill/outputs/response.md +59 -0
  275. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/edge-case-mixed-language-project/with_skill/outputs/response.md +60 -0
  276. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/eval-summary.md +105 -0
  277. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/partial-install-python-only/with_skill/outputs/response.md +93 -0
  278. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/python-refactor-request/with_skill/outputs/response.md +104 -0
  279. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/quality-gate-error-fix/with_skill/outputs/response.md +74 -0
  280. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/should-not-trigger-general-chat/with_skill/outputs/response.md +18 -0
  281. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/should-not-trigger-math-question/with_skill/outputs/response.md +18 -0
  282. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/should-not-trigger-unrelated-coding/with_skill/outputs/response.md +56 -0
  283. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/tdd-guard-blocking-confusion/with_skill/outputs/response.md +67 -0
  284. package/.xtrm/skills/default/quality-gates/workspace/iteration-1/typescript-feature-with-tests/with_skill/outputs/response.md +97 -0
  285. package/.xtrm/skills/default/scoping-service-skills/SKILL.md +231 -0
  286. package/.xtrm/skills/default/scoping-service-skills/scripts/scope.py +74 -0
  287. package/.xtrm/skills/default/service-skills-set/README.md +93 -0
  288. package/.xtrm/skills/default/service-skills-set/git-hooks/doc_reminder.py +67 -0
  289. package/.xtrm/skills/default/service-skills-set/git-hooks/skill_staleness.py +194 -0
  290. package/.xtrm/skills/default/service-skills-set/install-service-skills.py +193 -0
  291. package/.xtrm/skills/default/service-skills-set/service-registry.json +4 -0
  292. package/.xtrm/skills/default/service-skills-set/service-skills-readme.md +236 -0
  293. package/.xtrm/skills/default/service-skills-set/settings.json +37 -0
  294. package/.xtrm/skills/default/session-close-report/SKILL.md +131 -0
  295. package/.xtrm/skills/default/skill-creator/LICENSE.txt +202 -0
  296. package/.xtrm/skills/default/skill-creator/SKILL.md +479 -0
  297. package/.xtrm/skills/default/skill-creator/agents/analyzer.md +274 -0
  298. package/.xtrm/skills/default/skill-creator/agents/comparator.md +202 -0
  299. package/.xtrm/skills/default/skill-creator/agents/grader.md +223 -0
  300. package/.xtrm/skills/default/skill-creator/assets/eval_review.html +146 -0
  301. package/.xtrm/skills/default/skill-creator/eval-viewer/generate_review.py +471 -0
  302. package/.xtrm/skills/default/skill-creator/eval-viewer/viewer.html +1325 -0
  303. package/.xtrm/skills/default/skill-creator/references/schemas.md +430 -0
  304. package/.xtrm/skills/default/skill-creator/scripts/__init__.py +0 -0
  305. package/.xtrm/skills/default/skill-creator/scripts/aggregate_benchmark.py +401 -0
  306. package/.xtrm/skills/default/skill-creator/scripts/generate_report.py +326 -0
  307. package/.xtrm/skills/default/skill-creator/scripts/improve_description.py +248 -0
  308. package/.xtrm/skills/default/skill-creator/scripts/package_skill.py +136 -0
  309. package/.xtrm/skills/default/skill-creator/scripts/quick_validate.py +103 -0
  310. package/.xtrm/skills/default/skill-creator/scripts/run_eval.py +310 -0
  311. package/.xtrm/skills/default/skill-creator/scripts/run_loop.py +332 -0
  312. package/.xtrm/skills/default/skill-creator/scripts/utils.py +47 -0
  313. package/.xtrm/skills/default/specialists-creator/SKILL.md +705 -0
  314. package/.xtrm/skills/default/specialists-creator/scripts/validate-specialist.ts +41 -0
  315. package/.xtrm/skills/default/sync-docs/SKILL.md +262 -0
  316. package/.xtrm/skills/default/sync-docs/evals/evals.json +89 -0
  317. package/.xtrm/skills/default/sync-docs/references/doc-structure.md +99 -0
  318. package/.xtrm/skills/default/sync-docs/references/schema.md +103 -0
  319. package/.xtrm/skills/default/sync-docs/scripts/changelog/add_entry.py +216 -0
  320. package/.xtrm/skills/default/sync-docs/scripts/context_gatherer.py +405 -0
  321. package/.xtrm/skills/default/sync-docs/scripts/doc_structure_analyzer.py +495 -0
  322. package/.xtrm/skills/default/sync-docs/scripts/drift_detector.py +563 -0
  323. package/.xtrm/skills/default/sync-docs/scripts/validate_doc.py +365 -0
  324. package/.xtrm/skills/default/sync-docs/scripts/validate_metadata.py +185 -0
  325. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/benchmark.json +293 -0
  326. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/benchmark.md +13 -0
  327. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/eval_metadata.json +27 -0
  328. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/outputs/result.md +210 -0
  329. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/run-1/grading.json +28 -0
  330. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/with_skill/run-1/timing.json +1 -0
  331. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/outputs/result.md +101 -0
  332. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/run-1/grading.json +28 -0
  333. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/run-1/timing.json +5 -0
  334. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-doc-audit/without_skill/timing.json +5 -0
  335. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/eval_metadata.json +27 -0
  336. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/outputs/result.md +198 -0
  337. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/run-1/grading.json +28 -0
  338. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/with_skill/run-1/timing.json +1 -0
  339. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/outputs/result.md +94 -0
  340. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/run-1/grading.json +28 -0
  341. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-fix-mode/without_skill/run-1/timing.json +1 -0
  342. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/eval_metadata.json +27 -0
  343. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/outputs/result.md +237 -0
  344. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/run-1/grading.json +28 -0
  345. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/with_skill/run-1/timing.json +1 -0
  346. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/outputs/result.md +134 -0
  347. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/run-1/grading.json +28 -0
  348. package/.xtrm/skills/default/sync-docs-workspace/iteration-1/eval-sprint-closeout/without_skill/run-1/timing.json +1 -0
  349. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/benchmark.json +297 -0
  350. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/benchmark.md +13 -0
  351. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/eval_metadata.json +27 -0
  352. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/outputs/result.md +137 -0
  353. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/run-1/grading.json +92 -0
  354. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/with_skill/run-1/timing.json +1 -0
  355. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/outputs/result.md +134 -0
  356. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/run-1/grading.json +86 -0
  357. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-doc-audit/without_skill/run-1/timing.json +1 -0
  358. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/eval_metadata.json +27 -0
  359. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/outputs/result.md +193 -0
  360. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/run-1/grading.json +72 -0
  361. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/with_skill/run-1/timing.json +1 -0
  362. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/outputs/result.md +211 -0
  363. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/run-1/grading.json +91 -0
  364. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-fix-mode/without_skill/run-1/timing.json +5 -0
  365. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/eval_metadata.json +27 -0
  366. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/outputs/result.md +182 -0
  367. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/run-1/grading.json +95 -0
  368. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/with_skill/run-1/timing.json +1 -0
  369. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/outputs/result.md +222 -0
  370. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/run-1/grading.json +88 -0
  371. package/.xtrm/skills/default/sync-docs-workspace/iteration-2/eval-sprint-closeout/without_skill/run-1/timing.json +5 -0
  372. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/benchmark.json +298 -0
  373. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/benchmark.md +13 -0
  374. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/eval_metadata.json +27 -0
  375. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/outputs/result.md +125 -0
  376. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/run-1/grading.json +97 -0
  377. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/with_skill/run-1/timing.json +5 -0
  378. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/outputs/result.md +144 -0
  379. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/run-1/grading.json +78 -0
  380. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-doc-audit/without_skill/run-1/timing.json +5 -0
  381. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/eval_metadata.json +27 -0
  382. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/outputs/result.md +104 -0
  383. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/run-1/grading.json +91 -0
  384. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/with_skill/run-1/timing.json +5 -0
  385. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/outputs/result.md +79 -0
  386. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/run-1/grading.json +82 -0
  387. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-fix-mode/without_skill/run-1/timing.json +5 -0
  388. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/eval_metadata.json +27 -0
  389. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase1_context.json +302 -0
  390. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase2_drift.txt +33 -0
  391. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase3_analysis.json +114 -0
  392. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase4_fix.txt +118 -0
  393. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/phase5_validate.txt +38 -0
  394. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/outputs/result.md +158 -0
  395. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/run-1/grading.json +95 -0
  396. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/with_skill/run-1/timing.json +5 -0
  397. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/outputs/result.md +71 -0
  398. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/run-1/grading.json +90 -0
  399. package/.xtrm/skills/default/sync-docs-workspace/iteration-3/eval-sprint-closeout/without_skill/run-1/timing.json +5 -0
  400. package/.xtrm/skills/default/test-planning/SKILL.md +465 -0
  401. package/.xtrm/skills/default/test-planning/evals/evals.json +23 -0
  402. package/.xtrm/skills/default/updating-service-skills/SKILL.md +136 -0
  403. package/.xtrm/skills/default/updating-service-skills/scripts/drift_detector.py +222 -0
  404. package/.xtrm/skills/default/using-nodes/SKILL.md +333 -0
  405. package/.xtrm/skills/default/using-quality-gates/SKILL.md +254 -0
  406. package/.xtrm/skills/default/using-service-skills/SKILL.md +108 -0
  407. package/.xtrm/skills/default/using-service-skills/scripts/cataloger.py +74 -0
  408. package/.xtrm/skills/default/using-service-skills/scripts/skill_activator.py +152 -0
  409. package/.xtrm/skills/default/using-specialists/SKILL.md +848 -0
  410. package/.xtrm/skills/default/using-specialists/evals/evals.json +68 -0
  411. package/.xtrm/skills/default/using-tdd/SKILL.md +410 -0
  412. package/.xtrm/skills/default/using-xtrm/SKILL.md +127 -0
  413. package/.xtrm/skills/default/xt-debugging/SKILL.md +149 -0
  414. package/.xtrm/skills/default/xt-end/SKILL.md +297 -0
  415. package/.xtrm/skills/default/xt-merge/SKILL.md +326 -0
  416. package/.xtrm/skills/optional/README.txt +2 -0
  417. package/.xtrm/skills/optional/architecture-design/PACK.json +11 -0
  418. package/.xtrm/skills/optional/architecture-design/architecture-patterns/SKILL.md +494 -0
  419. package/.xtrm/skills/optional/architecture-design/architecture-patterns/references/advanced-patterns.md +391 -0
  420. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/SKILL.md +473 -0
  421. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/assets/few-shot-examples.json +106 -0
  422. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/assets/prompt-template-library.md +264 -0
  423. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/references/chain-of-thought.md +412 -0
  424. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/references/few-shot-learning.md +386 -0
  425. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/references/prompt-optimization.md +428 -0
  426. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/references/prompt-templates.md +484 -0
  427. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/references/system-prompts.md +195 -0
  428. package/.xtrm/skills/optional/architecture-design/prompt-engineering-patterns/scripts/optimize-prompt.py +279 -0
  429. package/.xtrm/skills/optional/architecture-design/subagent-driven-development/SKILL.md +277 -0
  430. package/.xtrm/skills/optional/architecture-design/subagent-driven-development/code-quality-reviewer-prompt.md +26 -0
  431. package/.xtrm/skills/optional/architecture-design/subagent-driven-development/implementer-prompt.md +113 -0
  432. package/.xtrm/skills/optional/architecture-design/subagent-driven-development/spec-reviewer-prompt.md +61 -0
  433. package/.xtrm/skills/optional/code-quality/PACK.json +12 -0
  434. package/.xtrm/skills/optional/code-quality/code-review-excellence/SKILL.md +529 -0
  435. package/.xtrm/skills/optional/code-quality/multi-reviewer-patterns/SKILL.md +127 -0
  436. package/.xtrm/skills/optional/code-quality/systematic-debugging/SKILL.md +296 -0
  437. package/.xtrm/skills/optional/code-quality/verification-before-completion/SKILL.md +139 -0
  438. package/.xtrm/skills/optional/data-engineering/PACK.json +9 -0
  439. package/.xtrm/skills/optional/data-engineering/data-analyst/SKILL.md +57 -0
  440. package/.xtrm/skills/optional/research-methods/PACK.json +12 -0
  441. package/.xtrm/skills/optional/research-methods/academic-researcher/SKILL.md +269 -0
  442. package/.xtrm/skills/optional/research-methods/brainstorming/SKILL.md +164 -0
  443. package/.xtrm/skills/optional/research-methods/brainstorming/scripts/frame-template.html +214 -0
  444. package/.xtrm/skills/optional/research-methods/brainstorming/scripts/helper.js +88 -0
  445. package/.xtrm/skills/optional/research-methods/brainstorming/scripts/server.cjs +354 -0
  446. package/.xtrm/skills/optional/research-methods/brainstorming/scripts/start-server.sh +148 -0
  447. package/.xtrm/skills/optional/research-methods/brainstorming/scripts/stop-server.sh +56 -0
  448. package/.xtrm/skills/optional/research-methods/brainstorming/spec-document-reviewer-prompt.md +49 -0
  449. package/.xtrm/skills/optional/research-methods/brainstorming/visual-companion.md +287 -0
  450. package/.xtrm/skills/optional/research-methods/deep-research/SKILL.md +192 -0
  451. package/.xtrm/skills/optional/research-methods/fact-checker/SKILL.md +182 -0
  452. package/.xtrm/skills/optional/security-ops/PACK.json +9 -0
  453. package/.xtrm/skills/optional/security-ops/security-auditor/SKILL.md +165 -0
  454. package/.xtrm/skills/optional/xt-optional/PACK.json +16 -0
  455. package/.xtrm/skills/optional/xt-optional/docker-expert/SKILL.md +409 -0
  456. package/.xtrm/skills/optional/xt-optional/obsidian-cli/SKILL.md +106 -0
  457. package/.xtrm/skills/optional/xt-optional/python-testing/SKILL.md +815 -0
  458. package/.xtrm/skills/optional/xt-optional/senior-backend/SKILL.md +209 -0
  459. package/.xtrm/skills/optional/xt-optional/senior-backend/references/api_design_patterns.md +103 -0
  460. package/.xtrm/skills/optional/xt-optional/senior-backend/references/backend_security_practices.md +103 -0
  461. package/.xtrm/skills/optional/xt-optional/senior-backend/references/database_optimization_guide.md +103 -0
  462. package/.xtrm/skills/optional/xt-optional/senior-backend/scripts/api_load_tester.py +114 -0
  463. package/.xtrm/skills/optional/xt-optional/senior-backend/scripts/api_scaffolder.py +114 -0
  464. package/.xtrm/skills/optional/xt-optional/senior-backend/scripts/database_migration_tool.py +114 -0
  465. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/SKILL.md +226 -0
  466. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/references/experiment_design_frameworks.md +80 -0
  467. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/references/feature_engineering_patterns.md +80 -0
  468. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/references/statistical_methods_advanced.md +80 -0
  469. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/scripts/experiment_designer.py +100 -0
  470. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/scripts/feature_engineering_pipeline.py +100 -0
  471. package/.xtrm/skills/optional/xt-optional/senior-data-scientist/scripts/model_evaluation_suite.py +100 -0
  472. package/.xtrm/skills/optional/xt-optional/senior-devops/SKILL.md +209 -0
  473. package/.xtrm/skills/optional/xt-optional/senior-devops/references/cicd_pipeline_guide.md +103 -0
  474. package/.xtrm/skills/optional/xt-optional/senior-devops/references/deployment_strategies.md +103 -0
  475. package/.xtrm/skills/optional/xt-optional/senior-devops/references/infrastructure_as_code.md +103 -0
  476. package/.xtrm/skills/optional/xt-optional/senior-devops/scripts/deployment_manager.py +114 -0
  477. package/.xtrm/skills/optional/xt-optional/senior-devops/scripts/pipeline_generator.py +114 -0
  478. package/.xtrm/skills/optional/xt-optional/senior-devops/scripts/terraform_scaffolder.py +114 -0
  479. package/.xtrm/skills/optional/xt-optional/senior-security/SKILL.md +209 -0
  480. package/.xtrm/skills/optional/xt-optional/senior-security/references/cryptography_implementation.md +103 -0
  481. package/.xtrm/skills/optional/xt-optional/senior-security/references/penetration_testing_guide.md +103 -0
  482. package/.xtrm/skills/optional/xt-optional/senior-security/references/security_architecture_patterns.md +103 -0
  483. package/.xtrm/skills/optional/xt-optional/senior-security/scripts/pentest_automator.py +114 -0
  484. package/.xtrm/skills/optional/xt-optional/senior-security/scripts/security_auditor.py +114 -0
  485. package/.xtrm/skills/optional/xt-optional/senior-security/scripts/threat_modeler.py +114 -0
  486. package/CHANGELOG.md +16 -0
  487. package/README.md +5 -0
  488. package/cli/dist/index.cjs +833 -517
  489. package/cli/dist/index.cjs.map +1 -1
  490. package/cli/package.json +1 -1
  491. package/package.json +3 -1
  492. package/.xtrm/extensions/auto-session-name/index.ts +0 -29
  493. package/.xtrm/extensions/auto-session-name/package.json +0 -16
  494. package/.xtrm/extensions/auto-update/index.ts +0 -71
  495. package/.xtrm/extensions/auto-update/package.json +0 -16
  496. package/.xtrm/extensions/beads/index.ts +0 -232
  497. package/.xtrm/extensions/beads/package.json +0 -19
  498. package/.xtrm/extensions/compact-header/index.ts +0 -69
  499. package/.xtrm/extensions/compact-header/package.json +0 -16
  500. package/.xtrm/extensions/core/adapter.ts +0 -52
  501. package/.xtrm/extensions/core/guard-rules.ts +0 -100
  502. package/.xtrm/extensions/core/lib.ts +0 -3
  503. package/.xtrm/extensions/core/logger.ts +0 -45
  504. package/.xtrm/extensions/core/package.json +0 -18
  505. package/.xtrm/extensions/core/runner.ts +0 -71
  506. package/.xtrm/extensions/core/session-state.ts +0 -59
  507. package/.xtrm/extensions/custom-footer/index.ts +0 -398
  508. package/.xtrm/extensions/custom-footer/package.json +0 -19
  509. package/.xtrm/extensions/custom-provider-qwen-cli/index.ts +0 -363
  510. package/.xtrm/extensions/custom-provider-qwen-cli/package.json +0 -1
  511. package/.xtrm/extensions/git-checkpoint/index.ts +0 -53
  512. package/.xtrm/extensions/git-checkpoint/package.json +0 -16
  513. package/.xtrm/extensions/lsp-bootstrap/index.ts +0 -134
  514. package/.xtrm/extensions/lsp-bootstrap/package.json +0 -17
  515. package/.xtrm/extensions/pi-serena-compact/index.ts +0 -121
  516. package/.xtrm/extensions/pi-serena-compact/package.json +0 -16
  517. package/.xtrm/extensions/quality-gates/index.ts +0 -66
  518. package/.xtrm/extensions/quality-gates/package.json +0 -19
  519. package/.xtrm/extensions/service-skills/index.ts +0 -108
  520. package/.xtrm/extensions/service-skills/package.json +0 -19
  521. package/.xtrm/extensions/session-flow/index.ts +0 -96
  522. package/.xtrm/extensions/session-flow/package.json +0 -19
  523. package/.xtrm/extensions/xtrm-loader/index.ts +0 -152
  524. package/.xtrm/extensions/xtrm-loader/package.json +0 -19
  525. package/.xtrm/extensions/xtrm-ui/format.ts +0 -93
  526. package/.xtrm/extensions/xtrm-ui/index.ts +0 -1044
  527. package/.xtrm/extensions/xtrm-ui/package.json +0 -10
  528. package/.xtrm/extensions/xtrm-ui/themes/pidex-dark.json +0 -85
  529. package/.xtrm/extensions/xtrm-ui/themes/pidex-light.json +0 -85
@@ -0,0 +1,162 @@
1
+ """Xiaohongshu HTTP API search client for last30days.
2
+
3
+ Uses xpzouying/xiaohongshu-mcp REST endpoints:
4
+ - GET/POST /api/v1/feeds/search
5
+ - GET /api/v1/login/status
6
+ """
7
+
8
+ from datetime import datetime, timezone
9
+ from typing import Any, Dict, List, Optional
10
+
11
+ from . import http
12
+
13
+
14
+ def _to_int(value: Any) -> int:
15
+ """Convert Xiaohongshu count strings to int.
16
+
17
+ Supports plain ints and Chinese suffixes like 1.2万 / 3亿.
18
+ """
19
+ if value is None:
20
+ return 0
21
+ if isinstance(value, (int, float)):
22
+ return int(value)
23
+
24
+ text = str(value).strip().lower().replace(",", "")
25
+ if not text:
26
+ return 0
27
+
28
+ try:
29
+ if text.endswith("万"):
30
+ return int(float(text[:-1]) * 10000)
31
+ if text.endswith("亿"):
32
+ return int(float(text[:-1]) * 100000000)
33
+ return int(float(text))
34
+ except (TypeError, ValueError):
35
+ return 0
36
+
37
+
38
+ def _timestamp_to_date_ms(ts: Any) -> Optional[str]:
39
+ """Convert millisecond timestamp to YYYY-MM-DD."""
40
+ try:
41
+ iv = int(ts)
42
+ if iv <= 0:
43
+ return None
44
+ # API examples use milliseconds.
45
+ dt = datetime.fromtimestamp(iv / 1000.0, tz=timezone.utc)
46
+ return dt.strftime("%Y-%m-%d")
47
+ except (TypeError, ValueError, OSError):
48
+ return None
49
+
50
+
51
+ def _relevance_from_interactions(likes: int, comments: int, favorites: int) -> float:
52
+ """Heuristic relevance score from engagement metrics."""
53
+ # Weighted engagement with soft caps to [0, 1].
54
+ weighted = (likes * 1.0) + (comments * 2.5) + (favorites * 1.5)
55
+ # 5000 weighted engagement ~= strong relevance.
56
+ score = min(1.0, max(0.05, weighted / 5000.0))
57
+ return round(score, 3)
58
+
59
+
60
+ def _build_note_url(feed_id: str, xsec_token: str) -> str:
61
+ """Build a stable Xiaohongshu note URL."""
62
+ if xsec_token:
63
+ return f"https://www.xiaohongshu.com/explore/{feed_id}?xsec_token={xsec_token}"
64
+ return f"https://www.xiaohongshu.com/explore/{feed_id}"
65
+
66
+
67
+ def search_feeds(
68
+ topic: str,
69
+ from_date: str,
70
+ to_date: str,
71
+ base_url: str,
72
+ depth: str = "default",
73
+ ) -> List[Dict[str, Any]]:
74
+ """Search Xiaohongshu feeds and normalize to web-item shape."""
75
+ base = (base_url or "").rstrip("/")
76
+ if not base:
77
+ raise ValueError("Missing Xiaohongshu API base URL")
78
+
79
+ # Quick login sanity check.
80
+ login = http.get(f"{base}/api/v1/login/status", timeout=8, retries=1)
81
+ is_logged_in = (
82
+ login.get("data", {}).get("is_logged_in")
83
+ if isinstance(login, dict) else False
84
+ )
85
+ if not is_logged_in:
86
+ raise http.HTTPError("Xiaohongshu API reachable but not logged in")
87
+
88
+ # API supports filters; use recency-oriented defaults.
89
+ publish_time = "一天内" if depth == "quick" else "一周内" if depth == "default" else "半年内"
90
+ payload = {
91
+ "keyword": topic,
92
+ "filters": {
93
+ "sort_by": "综合",
94
+ "note_type": "不限",
95
+ "publish_time": publish_time,
96
+ "search_scope": "不限",
97
+ "location": "不限",
98
+ },
99
+ }
100
+
101
+ resp = http.post(f"{base}/api/v1/feeds/search", payload, timeout=20, retries=1)
102
+ feeds = resp.get("data", {}).get("feeds", []) if isinstance(resp, dict) else []
103
+ if not isinstance(feeds, list):
104
+ feeds = []
105
+
106
+ # Cap source volume similarly to other web sources.
107
+ limit = {"quick": 8, "default": 15, "deep": 25}.get(depth, 15)
108
+ items: List[Dict[str, Any]] = []
109
+
110
+ for i, feed in enumerate(feeds[:limit]):
111
+ if not isinstance(feed, dict):
112
+ continue
113
+ note = feed.get("noteCard") or {}
114
+ if not isinstance(note, dict):
115
+ note = {}
116
+ interact = note.get("interactInfo") or {}
117
+ if not isinstance(interact, dict):
118
+ interact = {}
119
+
120
+ feed_id = str(feed.get("id") or note.get("noteId") or "").strip()
121
+ if not feed_id:
122
+ continue
123
+
124
+ xsec_token = str(feed.get("xsecToken") or note.get("xsecToken") or "").strip()
125
+ title = str(
126
+ note.get("displayTitle")
127
+ or note.get("title")
128
+ or ""
129
+ ).strip()
130
+ snippet = str(
131
+ note.get("desc")
132
+ or note.get("displayDesc")
133
+ or title
134
+ or ""
135
+ ).strip()
136
+
137
+ likes = _to_int(interact.get("likedCount"))
138
+ comments = _to_int(interact.get("commentCount"))
139
+ favorites = _to_int(interact.get("collectedCount"))
140
+
141
+ date_value = _timestamp_to_date_ms(note.get("time"))
142
+ why = f"Xiaohongshu engagement: likes={likes}, comments={comments}, favorites={favorites}"
143
+
144
+ items.append({
145
+ "id": f"XHS{i+1}",
146
+ "title": title[:200] if title else f"Xiaohongshu note {feed_id}",
147
+ "url": _build_note_url(feed_id, xsec_token),
148
+ "source_domain": "xiaohongshu.com",
149
+ "snippet": snippet[:500],
150
+ "date": date_value,
151
+ "date_confidence": "high" if date_value else "low",
152
+ "relevance": _relevance_from_interactions(likes, comments, favorites),
153
+ "why_relevant": why,
154
+ # Keep raw engagement for debugging/possible future rendering.
155
+ "engagement": {
156
+ "likes": likes,
157
+ "comments": comments,
158
+ "favorites": favorites,
159
+ },
160
+ })
161
+
162
+ return items
@@ -0,0 +1,538 @@
1
+ """YouTube search and transcript extraction via yt-dlp for /last30days v2.1.
2
+
3
+ Uses yt-dlp (https://github.com/yt-dlp/yt-dlp) for both YouTube search and
4
+ transcript extraction. No API keys needed — just have yt-dlp installed.
5
+
6
+ Inspired by Peter Steinberger's toolchain approach (yt-dlp + summarize CLI).
7
+ """
8
+
9
+ import json
10
+ import math
11
+ import os
12
+ import re
13
+ import signal
14
+ import shutil
15
+ import subprocess
16
+ import sys
17
+ import tempfile
18
+ import urllib.error
19
+ import urllib.request
20
+ from concurrent.futures import ThreadPoolExecutor, as_completed
21
+ from pathlib import Path
22
+ from typing import Any, Dict, List, Optional, Set, Tuple
23
+
24
+ # Depth configurations: how many videos to search / transcribe
25
+ DEPTH_CONFIG = {
26
+ "quick": 10,
27
+ "default": 20,
28
+ "deep": 40,
29
+ }
30
+
31
+ TRANSCRIPT_LIMITS = {
32
+ "quick": 3,
33
+ "default": 5,
34
+ "deep": 8,
35
+ }
36
+
37
+ # Max words to keep from each transcript
38
+ TRANSCRIPT_MAX_WORDS = 5000
39
+
40
+ from .relevance import token_overlap_relevance as _compute_relevance
41
+
42
+
43
+ def extract_transcript_highlights(transcript: str, topic: str, limit: int = 5) -> List[str]:
44
+ """Extract quotable highlights from a YouTube transcript.
45
+
46
+ Similar to reddit_enrich.extract_comment_insights() but for
47
+ continuous speech-to-text rather than threaded comments.
48
+ """
49
+ if not transcript:
50
+ return []
51
+
52
+ sentences = re.split(r'(?<=[.!?])\s+', transcript)
53
+
54
+ filler = [
55
+ r"^(hey |hi |what's up|welcome back|in today's video|don't forget to)",
56
+ r"(subscribe|like and comment|hit the bell|check out the link|down below)",
57
+ r"^(so |and |but |okay |alright |um |uh )",
58
+ r"(thanks for watching|see you (next|in the)|bye)",
59
+ ]
60
+
61
+ topic_words = [w.lower() for w in topic.lower().split() if len(w) > 2]
62
+
63
+ candidates = []
64
+ for sent in sentences:
65
+ sent = sent.strip()
66
+ words = sent.split()
67
+ if len(words) < 8 or len(words) > 50:
68
+ continue
69
+ if any(re.search(p, sent, re.IGNORECASE) for p in filler):
70
+ continue
71
+
72
+ score = 0
73
+ if re.search(r'\d', sent):
74
+ score += 2
75
+ if re.search(r'[A-Z][a-z]+', sent):
76
+ score += 1
77
+ if '?' in sent:
78
+ score += 1
79
+ sent_lower = sent.lower()
80
+ if any(w in sent_lower for w in topic_words):
81
+ score += 2
82
+
83
+ candidates.append((score, sent))
84
+
85
+ candidates.sort(key=lambda x: -x[0])
86
+ return [sent for _, sent in candidates[:limit]]
87
+
88
+
89
+ def _log(msg: str):
90
+ """Log to stderr."""
91
+ sys.stderr.write(f"[YouTube] {msg}\n")
92
+ sys.stderr.flush()
93
+
94
+
95
+ def is_ytdlp_installed() -> bool:
96
+ """Check if yt-dlp is available in PATH."""
97
+ return shutil.which("yt-dlp") is not None
98
+
99
+
100
+ def _extract_core_subject(topic: str) -> str:
101
+ """Extract core subject from verbose query for YouTube search.
102
+
103
+ NOTE: 'tips', 'tricks', 'tutorial', 'guide', 'review', 'reviews'
104
+ are intentionally KEPT — they're YouTube content types that improve search.
105
+ """
106
+ from .query import extract_core_subject
107
+ # YouTube-specific noise set: smaller than default, keeps content-type words
108
+ _YT_NOISE = frozenset({
109
+ 'best', 'top', 'good', 'great', 'awesome', 'killer',
110
+ 'latest', 'new', 'news', 'update', 'updates',
111
+ 'trending', 'hottest', 'popular', 'viral',
112
+ 'practices', 'features',
113
+ 'recommendations', 'advice',
114
+ 'prompt', 'prompts', 'prompting',
115
+ 'methods', 'strategies', 'approaches',
116
+ })
117
+ return extract_core_subject(topic, noise=_YT_NOISE)
118
+
119
+
120
+ def search_youtube(
121
+ topic: str,
122
+ from_date: str,
123
+ to_date: str,
124
+ depth: str = "default",
125
+ ) -> Dict[str, Any]:
126
+ """Search YouTube via yt-dlp. No API key needed.
127
+
128
+ Args:
129
+ topic: Search topic
130
+ from_date: Start date (YYYY-MM-DD)
131
+ to_date: End date (YYYY-MM-DD)
132
+ depth: 'quick', 'default', or 'deep'
133
+
134
+ Returns:
135
+ Dict with 'items' list of video metadata dicts.
136
+ """
137
+ if not is_ytdlp_installed():
138
+ return {"items": [], "error": "yt-dlp not installed"}
139
+
140
+ count = DEPTH_CONFIG.get(depth, DEPTH_CONFIG["default"])
141
+ core_topic = _extract_core_subject(topic)
142
+
143
+ _log(f"Searching YouTube for '{core_topic}' (since {from_date}, count={count})")
144
+
145
+ # yt-dlp search with full metadata (no --flat-playlist so dates are real).
146
+ # NOTE: --dateafter intentionally omitted — YouTube search returns
147
+ # relevance-sorted results and strict date filtering returns 0 for
148
+ # evergreen topics. Python soft filter (below) handles date filtering.
149
+ cmd = [
150
+ "yt-dlp",
151
+ "--ignore-config",
152
+ "--no-cookies-from-browser",
153
+ f"ytsearch{count}:{core_topic}",
154
+ "--dump-json",
155
+ "--no-warnings",
156
+ "--no-download",
157
+ ]
158
+
159
+ preexec = os.setsid if hasattr(os, 'setsid') else None
160
+
161
+ try:
162
+ proc = subprocess.Popen(
163
+ cmd,
164
+ stdout=subprocess.PIPE,
165
+ stderr=subprocess.PIPE,
166
+ text=True,
167
+ preexec_fn=preexec,
168
+ )
169
+ try:
170
+ stdout, stderr = proc.communicate(timeout=120)
171
+ except subprocess.TimeoutExpired:
172
+ try:
173
+ os.killpg(os.getpgid(proc.pid), signal.SIGTERM)
174
+ except (ProcessLookupError, PermissionError, OSError):
175
+ proc.kill()
176
+ proc.wait(timeout=5)
177
+ _log("YouTube search timed out (120s)")
178
+ return {"items": [], "error": "Search timed out"}
179
+ except FileNotFoundError:
180
+ return {"items": [], "error": "yt-dlp not found"}
181
+
182
+ if not (stdout or "").strip():
183
+ _log("YouTube search returned 0 results")
184
+ return {"items": []}
185
+
186
+ # Parse JSON-per-line output
187
+ items = []
188
+ for line in stdout.strip().split("\n"):
189
+ line = line.strip()
190
+ if not line:
191
+ continue
192
+ try:
193
+ video = json.loads(line)
194
+ except json.JSONDecodeError:
195
+ continue
196
+
197
+ video_id = video.get("id", "")
198
+ view_count = video.get("view_count") or 0
199
+ like_count = video.get("like_count") or 0
200
+ comment_count = video.get("comment_count") or 0
201
+ upload_date = video.get("upload_date", "") # YYYYMMDD
202
+
203
+ # Convert YYYYMMDD to YYYY-MM-DD
204
+ date_str = None
205
+ if upload_date and len(upload_date) == 8:
206
+ date_str = f"{upload_date[:4]}-{upload_date[4:6]}-{upload_date[6:8]}"
207
+
208
+ items.append({
209
+ "video_id": video_id,
210
+ "title": video.get("title", ""),
211
+ "url": f"https://www.youtube.com/watch?v={video_id}",
212
+ "channel_name": video.get("channel", video.get("uploader", "")),
213
+ "date": date_str,
214
+ "engagement": {
215
+ "views": view_count,
216
+ "likes": like_count,
217
+ "comments": comment_count,
218
+ },
219
+ "duration": video.get("duration"),
220
+ "relevance": _compute_relevance(core_topic, video.get("title", "")),
221
+ "why_relevant": f"YouTube: {video.get('title', core_topic)[:60]}",
222
+ })
223
+
224
+ # Soft date filter: prefer recent items but fall back to all if too few
225
+ recent = [i for i in items if i["date"] and i["date"] >= from_date]
226
+ if len(recent) >= 3:
227
+ items = recent
228
+ _log(f"Found {len(items)} videos within date range")
229
+ else:
230
+ _log(f"Found {len(items)} videos ({len(recent)} within date range, keeping all)")
231
+
232
+ # Sort by views descending
233
+ items.sort(key=lambda x: x["engagement"]["views"], reverse=True)
234
+
235
+ return {"items": items}
236
+
237
+
238
+ def _clean_vtt(vtt_text: str) -> str:
239
+ """Convert VTT subtitle format to clean plaintext."""
240
+ # Strip VTT header
241
+ text = re.sub(r'^WEBVTT.*?\n\n', '', vtt_text, flags=re.DOTALL)
242
+ # Strip timestamps
243
+ text = re.sub(r'\d{2}:\d{2}:\d{2}\.\d{3}\s*-->\s*\d{2}:\d{2}:\d{2}\.\d{3}.*\n', '', text)
244
+ # Strip position/alignment tags
245
+ text = re.sub(r'<[^>]+>', '', text)
246
+ # Strip cue numbers
247
+ text = re.sub(r'^\d+\s*$', '', text, flags=re.MULTILINE)
248
+ # Deduplicate overlapping lines
249
+ lines = text.strip().split('\n')
250
+ seen = set()
251
+ unique = []
252
+ for line in lines:
253
+ stripped = line.strip()
254
+ if stripped and stripped not in seen:
255
+ seen.add(stripped)
256
+ unique.append(stripped)
257
+ return re.sub(r'\s+', ' ', ' '.join(unique)).strip()
258
+
259
+
260
+ _YT_USER_AGENT = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36"
261
+
262
+
263
+ def _fetch_transcript_direct(video_id: str, timeout: int = 30) -> Optional[str]:
264
+ """Fetch YouTube transcript via direct HTTP without yt-dlp.
265
+
266
+ Scrapes the watch page HTML for the captions track URL in
267
+ ytInitialPlayerResponse, then fetches the VTT subtitle file.
268
+
269
+ Args:
270
+ video_id: YouTube video ID
271
+ timeout: HTTP request timeout in seconds
272
+
273
+ Returns:
274
+ Raw VTT text, or None if captions are unavailable.
275
+ """
276
+ watch_url = f"https://www.youtube.com/watch?v={video_id}"
277
+ headers = {
278
+ "User-Agent": _YT_USER_AGENT,
279
+ "Accept-Language": "en-US,en;q=0.9",
280
+ }
281
+
282
+ # Step 1: Fetch the watch page HTML
283
+ req = urllib.request.Request(watch_url, headers=headers)
284
+ try:
285
+ with urllib.request.urlopen(req, timeout=timeout) as resp:
286
+ html = resp.read().decode("utf-8", errors="replace")
287
+ except (urllib.error.URLError, urllib.error.HTTPError, OSError, TimeoutError) as exc:
288
+ _log(f"Direct transcript: failed to fetch watch page for {video_id}: {exc}")
289
+ return None
290
+
291
+ # Step 2: Extract captions URL from ytInitialPlayerResponse
292
+ # YouTube embeds this as a JS variable in the page HTML
293
+ match = re.search(
294
+ r'ytInitialPlayerResponse\s*=\s*(\{.+?\})\s*;(?:\s*var\s|\s*<\/script>)',
295
+ html,
296
+ )
297
+ if not match:
298
+ # Fallback: try the JSON embedded in the script tag
299
+ match = re.search(
300
+ r'var\s+ytInitialPlayerResponse\s*=\s*(\{.+?\})\s*;',
301
+ html,
302
+ )
303
+ if not match:
304
+ _log(f"Direct transcript: no ytInitialPlayerResponse found for {video_id}")
305
+ return None
306
+
307
+ try:
308
+ player_response = json.loads(match.group(1))
309
+ except json.JSONDecodeError:
310
+ _log(f"Direct transcript: failed to parse ytInitialPlayerResponse for {video_id}")
311
+ return None
312
+
313
+ # Navigate to caption tracks
314
+ captions = player_response.get("captions", {})
315
+ renderer = captions.get("playerCaptionsTracklistRenderer", {})
316
+ caption_tracks = renderer.get("captionTracks", [])
317
+
318
+ if not caption_tracks:
319
+ _log(f"Direct transcript: no caption tracks for {video_id}")
320
+ return None
321
+
322
+ # Find English track (prefer exact 'en', then any en variant, then first track)
323
+ base_url = None
324
+ for track in caption_tracks:
325
+ lang = track.get("languageCode", "")
326
+ if lang == "en":
327
+ base_url = track.get("baseUrl")
328
+ break
329
+ if not base_url:
330
+ for track in caption_tracks:
331
+ lang = track.get("languageCode", "")
332
+ if lang.startswith("en"):
333
+ base_url = track.get("baseUrl")
334
+ break
335
+ if not base_url:
336
+ # Fall back to first available track
337
+ base_url = caption_tracks[0].get("baseUrl")
338
+ if not base_url:
339
+ _log(f"Direct transcript: no baseUrl in caption tracks for {video_id}")
340
+ return None
341
+
342
+ # Step 3: Fetch the VTT subtitle file
343
+ sep = "&" if "?" in base_url else "?"
344
+ vtt_url = f"{base_url}{sep}fmt=vtt"
345
+ vtt_req = urllib.request.Request(vtt_url, headers=headers)
346
+ try:
347
+ with urllib.request.urlopen(vtt_req, timeout=timeout) as resp:
348
+ vtt_text = resp.read().decode("utf-8", errors="replace")
349
+ except (urllib.error.URLError, urllib.error.HTTPError, OSError, TimeoutError) as exc:
350
+ _log(f"Direct transcript: failed to fetch VTT for {video_id}: {exc}")
351
+ return None
352
+
353
+ if not vtt_text or not vtt_text.strip():
354
+ return None
355
+
356
+ return vtt_text
357
+
358
+
359
+ def _fetch_transcript_ytdlp(video_id: str, temp_dir: str) -> Optional[str]:
360
+ """Fetch transcript using yt-dlp (original implementation).
361
+
362
+ Args:
363
+ video_id: YouTube video ID
364
+ temp_dir: Temporary directory for subtitle files
365
+
366
+ Returns:
367
+ Raw VTT text, or None if no captions available.
368
+ """
369
+ cmd = [
370
+ "yt-dlp",
371
+ "--ignore-config",
372
+ "--no-cookies-from-browser",
373
+ "--write-auto-subs",
374
+ "--sub-lang", "en",
375
+ "--sub-format", "vtt",
376
+ "--skip-download",
377
+ "--no-warnings",
378
+ "-o", f"{temp_dir}/%(id)s",
379
+ f"https://www.youtube.com/watch?v={video_id}",
380
+ ]
381
+
382
+ preexec = os.setsid if hasattr(os, 'setsid') else None
383
+
384
+ try:
385
+ proc = subprocess.Popen(
386
+ cmd,
387
+ stdout=subprocess.PIPE,
388
+ stderr=subprocess.PIPE,
389
+ text=True,
390
+ preexec_fn=preexec,
391
+ )
392
+ try:
393
+ proc.communicate(timeout=30)
394
+ except subprocess.TimeoutExpired:
395
+ try:
396
+ os.killpg(os.getpgid(proc.pid), signal.SIGTERM)
397
+ except (ProcessLookupError, PermissionError, OSError):
398
+ proc.kill()
399
+ proc.wait(timeout=5)
400
+ return None
401
+ except FileNotFoundError:
402
+ return None
403
+
404
+ # yt-dlp may save as .en.vtt or .en-orig.vtt
405
+ vtt_path = Path(temp_dir) / f"{video_id}.en.vtt"
406
+ if not vtt_path.exists():
407
+ # Try alternate naming
408
+ for p in Path(temp_dir).glob(f"{video_id}*.vtt"):
409
+ vtt_path = p
410
+ break
411
+ else:
412
+ return None
413
+
414
+ try:
415
+ return vtt_path.read_text(encoding="utf-8", errors="replace")
416
+ except OSError:
417
+ return None
418
+
419
+
420
+ def fetch_transcript(video_id: str, temp_dir: str) -> Optional[str]:
421
+ """Fetch auto-generated transcript for a YouTube video.
422
+
423
+ Uses yt-dlp when available (preferred, more robust). Falls back to
424
+ direct HTTP transcript fetching when yt-dlp is not installed.
425
+
426
+ Args:
427
+ video_id: YouTube video ID
428
+ temp_dir: Temporary directory for subtitle files
429
+
430
+ Returns:
431
+ Plaintext transcript string, or None if no captions available.
432
+ """
433
+ raw_vtt = None
434
+ if is_ytdlp_installed():
435
+ raw_vtt = _fetch_transcript_ytdlp(video_id, temp_dir)
436
+ else:
437
+ _log("yt-dlp not installed, using direct HTTP transcript fetch")
438
+ raw_vtt = _fetch_transcript_direct(video_id)
439
+
440
+ if not raw_vtt:
441
+ return None
442
+
443
+ transcript = _clean_vtt(raw_vtt)
444
+
445
+ # Truncate to max words
446
+ words = transcript.split()
447
+ if len(words) > TRANSCRIPT_MAX_WORDS:
448
+ transcript = ' '.join(words[:TRANSCRIPT_MAX_WORDS]) + '...'
449
+
450
+ return transcript if transcript else None
451
+
452
+
453
+ def fetch_transcripts_parallel(
454
+ video_ids: List[str],
455
+ max_workers: int = 5,
456
+ ) -> Dict[str, Optional[str]]:
457
+ """Fetch transcripts for multiple videos in parallel.
458
+
459
+ Args:
460
+ video_ids: List of YouTube video IDs
461
+ max_workers: Max parallel fetches
462
+
463
+ Returns:
464
+ Dict mapping video_id to transcript text (or None).
465
+ """
466
+ if not video_ids:
467
+ return {}
468
+
469
+ _log(f"Fetching transcripts for {len(video_ids)} videos")
470
+
471
+ results = {}
472
+ with tempfile.TemporaryDirectory() as temp_dir:
473
+ with ThreadPoolExecutor(max_workers=max_workers) as executor:
474
+ futures = {
475
+ executor.submit(fetch_transcript, vid, temp_dir): vid
476
+ for vid in video_ids
477
+ }
478
+ for future in as_completed(futures):
479
+ vid = futures[future]
480
+ try:
481
+ results[vid] = future.result()
482
+ except Exception:
483
+ results[vid] = None
484
+
485
+ got = sum(1 for v in results.values() if v)
486
+ _log(f"Got transcripts for {got}/{len(video_ids)} videos")
487
+ return results
488
+
489
+
490
+ def search_and_transcribe(
491
+ topic: str,
492
+ from_date: str,
493
+ to_date: str,
494
+ depth: str = "default",
495
+ ) -> Dict[str, Any]:
496
+ """Full YouTube search: find videos, then fetch transcripts for top results.
497
+
498
+ Args:
499
+ topic: Search topic
500
+ from_date: Start date (YYYY-MM-DD)
501
+ to_date: End date (YYYY-MM-DD)
502
+ depth: 'quick', 'default', or 'deep'
503
+
504
+ Returns:
505
+ Dict with 'items' list. Each item has a 'transcript_snippet' field.
506
+ """
507
+ # Step 1: Search
508
+ search_result = search_youtube(topic, from_date, to_date, depth)
509
+ items = search_result.get("items", [])
510
+
511
+ if not items:
512
+ return search_result
513
+
514
+ # Step 2: Fetch transcripts for top N by views
515
+ transcript_limit = TRANSCRIPT_LIMITS.get(depth, TRANSCRIPT_LIMITS["default"])
516
+ top_ids = [item["video_id"] for item in items[:transcript_limit]]
517
+ transcripts = fetch_transcripts_parallel(top_ids)
518
+
519
+ # Step 3: Attach transcripts and extract highlights
520
+ core_topic = _extract_core_subject(topic)
521
+ for item in items:
522
+ vid = item["video_id"]
523
+ transcript = transcripts.get(vid)
524
+ item["transcript_snippet"] = transcript or ""
525
+ item["transcript_highlights"] = extract_transcript_highlights(
526
+ transcript or "", core_topic,
527
+ )
528
+
529
+ return {"items": items}
530
+
531
+
532
+ def parse_youtube_response(response: Dict[str, Any]) -> List[Dict[str, Any]]:
533
+ """Parse YouTube search response to normalized format.
534
+
535
+ Returns:
536
+ List of item dicts ready for normalization.
537
+ """
538
+ return response.get("items", [])