@researai/deepscientist 1.5.17 → 1.6.0

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 (894) hide show
  1. package/AGENTS.md +309 -130
  2. package/AISB/catalog/aisb.b1.agentic_coding.yaml +244 -0
  3. package/AISB/catalog/aisb.b10.climate_earth.yaml +235 -0
  4. package/AISB/catalog/aisb.b11.model_efficiency.yaml +231 -0
  5. package/AISB/catalog/aisb.b12.embodied_ai.yaml +238 -0
  6. package/AISB/catalog/aisb.b2.agent_systems.yaml +229 -0
  7. package/AISB/catalog/aisb.b3.self_evolving_rl.yaml +237 -0
  8. package/AISB/catalog/aisb.b4.lm_reasoning.yaml +240 -0
  9. package/AISB/catalog/aisb.b5.math_proof.yaml +235 -0
  10. package/AISB/catalog/aisb.b6.research_process.yaml +243 -0
  11. package/AISB/catalog/aisb.b7.multimodal_fusion.yaml +232 -0
  12. package/AISB/catalog/aisb.b8.lifesci_drug.yaml +275 -0
  13. package/AISB/catalog/aisb.b9.material_science.yaml +237 -0
  14. package/AISB/catalog/aisb.t3.001_savvy.yaml +159 -0
  15. package/AISB/catalog/aisb.t3.001_savvy.zh.yaml +121 -0
  16. package/AISB/catalog/aisb.t3.002_pinet.yaml +189 -0
  17. package/AISB/catalog/aisb.t3.002_pinet.zh.yaml +130 -0
  18. package/AISB/catalog/aisb.t3.004_decentralattn.yaml +184 -0
  19. package/AISB/catalog/aisb.t3.004_decentralattn.zh.yaml +153 -0
  20. package/AISB/catalog/aisb.t3.005_tsae.yaml +193 -0
  21. package/AISB/catalog/aisb.t3.005_tsae.zh.yaml +139 -0
  22. package/AISB/catalog/aisb.t3.006_physense.yaml +194 -0
  23. package/AISB/catalog/aisb.t3.006_physense.zh.yaml +118 -0
  24. package/AISB/catalog/aisb.t3.007_reasoningiqa.yaml +169 -0
  25. package/AISB/catalog/aisb.t3.007_reasoningiqa.zh.yaml +133 -0
  26. package/AISB/catalog/aisb.t3.008_meanflows.yaml +188 -0
  27. package/AISB/catalog/aisb.t3.008_meanflows.zh.yaml +140 -0
  28. package/AISB/catalog/aisb.t3.009_scoremissing.yaml +179 -0
  29. package/AISB/catalog/aisb.t3.009_scoremissing.zh.yaml +119 -0
  30. package/AISB/catalog/aisb.t3.010_suitabilityfilter.yaml +221 -0
  31. package/AISB/catalog/aisb.t3.010_suitabilityfilter.zh.yaml +141 -0
  32. package/AISB/catalog/aisb.t3.011_osd.yaml +206 -0
  33. package/AISB/catalog/aisb.t3.011_osd.zh.yaml +163 -0
  34. package/AISB/catalog/aisb.t3.012_efficientqat.yaml +206 -0
  35. package/AISB/catalog/aisb.t3.012_efficientqat.zh.yaml +159 -0
  36. package/AISB/catalog/aisb.t3.013_appl.yaml +152 -0
  37. package/AISB/catalog/aisb.t3.013_appl.zh.yaml +126 -0
  38. package/AISB/catalog/aisb.t3.014_piguard.yaml +207 -0
  39. package/AISB/catalog/aisb.t3.014_piguard.zh.yaml +164 -0
  40. package/AISB/catalog/aisb.t3.015_frspec.yaml +209 -0
  41. package/AISB/catalog/aisb.t3.015_frspec.zh.yaml +163 -0
  42. package/AISB/catalog/aisb.t3.016_mathfusion.yaml +166 -0
  43. package/AISB/catalog/aisb.t3.016_mathfusion.zh.yaml +145 -0
  44. package/AISB/catalog/aisb.t3.017_multimodalglp.yaml +171 -0
  45. package/AISB/catalog/aisb.t3.017_multimodalglp.zh.yaml +122 -0
  46. package/AISB/catalog/aisb.t3.018_cotsynth.yaml +206 -0
  47. package/AISB/catalog/aisb.t3.018_cotsynth.zh.yaml +162 -0
  48. package/AISB/catalog/aisb.t3.019_dyscaleut.yaml +211 -0
  49. package/AISB/catalog/aisb.t3.019_dyscaleut.zh.yaml +148 -0
  50. package/AISB/catalog/aisb.t3.020_aristotle.yaml +173 -0
  51. package/AISB/catalog/aisb.t3.020_aristotle.zh.yaml +119 -0
  52. package/AISB/catalog/aisb.t3.021_tokenrecycling.yaml +160 -0
  53. package/AISB/catalog/aisb.t3.021_tokenrecycling.zh.yaml +129 -0
  54. package/AISB/catalog/aisb.t3.022_chainofreasoning.yaml +204 -0
  55. package/AISB/catalog/aisb.t3.022_chainofreasoning.zh.yaml +161 -0
  56. package/AISB/catalog/aisb.t3.023_guidedembed.yaml +211 -0
  57. package/AISB/catalog/aisb.t3.023_guidedembed.zh.yaml +189 -0
  58. package/AISB/catalog/aisb.t3.024_outputcentric.yaml +148 -0
  59. package/AISB/catalog/aisb.t3.024_outputcentric.zh.yaml +131 -0
  60. package/AISB/catalog/aisb.t3.025_deeper.yaml +143 -0
  61. package/AISB/catalog/aisb.t3.025_deeper.zh.yaml +116 -0
  62. package/AISB/catalog/aisb.t3.026_gartkg.yaml +195 -0
  63. package/AISB/catalog/aisb.t3.026_gartkg.zh.yaml +127 -0
  64. package/AISB/catalog/aisb.t3.027_citeeval.yaml +182 -0
  65. package/AISB/catalog/aisb.t3.027_citeeval.zh.yaml +135 -0
  66. package/AISB/catalog/aisb.t3.028_sbam.yaml +206 -0
  67. package/AISB/catalog/aisb.t3.028_sbam.zh.yaml +166 -0
  68. package/AISB/catalog/aisb.t3.029_cdqgeoembed.yaml +224 -0
  69. package/AISB/catalog/aisb.t3.029_cdqgeoembed.zh.yaml +142 -0
  70. package/AISB/catalog/aisb.t3.030_processrm.yaml +211 -0
  71. package/AISB/catalog/aisb.t3.030_processrm.zh.yaml +166 -0
  72. package/AISB/catalog/aisb.t3.031_circuitstability.yaml +172 -0
  73. package/AISB/catalog/aisb.t3.031_circuitstability.zh.yaml +134 -0
  74. package/AISB/catalog/aisb.t3.032_ptsolver.yaml +169 -0
  75. package/AISB/catalog/aisb.t3.032_ptsolver.zh.yaml +135 -0
  76. package/AISB/catalog/aisb.t3.033_gcse.yaml +144 -0
  77. package/AISB/catalog/aisb.t3.033_gcse.zh.yaml +126 -0
  78. package/AISB/catalog/aisb.t3.034_ensemblewm.yaml +183 -0
  79. package/AISB/catalog/aisb.t3.034_ensemblewm.zh.yaml +146 -0
  80. package/AISB/catalog/aisb.t3.035_moralvalueswa.yaml +207 -0
  81. package/AISB/catalog/aisb.t3.035_moralvalueswa.zh.yaml +165 -0
  82. package/AISB/catalog/aisb.t3.036_weakstrongpref.yaml +210 -0
  83. package/AISB/catalog/aisb.t3.036_weakstrongpref.zh.yaml +194 -0
  84. package/AISB/catalog/aisb.t3.037_dementiamask.yaml +172 -0
  85. package/AISB/catalog/aisb.t3.037_dementiamask.zh.yaml +132 -0
  86. package/AISB/catalog/aisb.t3.038_tinysam.yaml +284 -0
  87. package/AISB/catalog/aisb.t3.038_tinysam.zh.yaml +240 -0
  88. package/AISB/catalog/aisb.t3.039_calf.yaml +224 -0
  89. package/AISB/catalog/aisb.t3.039_calf.zh.yaml +194 -0
  90. package/AISB/catalog/aisb.t3.040_graniteguardian.yaml +199 -0
  91. package/AISB/catalog/aisb.t3.040_graniteguardian.zh.yaml +174 -0
  92. package/AISB/catalog/aisb.t3.041_amdm.yaml +149 -0
  93. package/AISB/catalog/aisb.t3.041_amdm.zh.yaml +137 -0
  94. package/AISB/catalog/aisb.t3.042_xpatch.yaml +216 -0
  95. package/AISB/catalog/aisb.t3.042_xpatch.zh.yaml +182 -0
  96. package/AISB/catalog/aisb.t3.043_vhm.yaml +268 -0
  97. package/AISB/catalog/aisb.t3.043_vhm.zh.yaml +193 -0
  98. package/AISB/catalog/aisb.t3.044_rgvi.yaml +224 -0
  99. package/AISB/catalog/aisb.t3.044_rgvi.zh.yaml +176 -0
  100. package/AISB/catalog/aisb.t3.045_pslstm.yaml +203 -0
  101. package/AISB/catalog/aisb.t3.045_pslstm.zh.yaml +179 -0
  102. package/AISB/catalog/aisb.t3.046_nonstatts.yaml +208 -0
  103. package/AISB/catalog/aisb.t3.046_nonstatts.zh.yaml +194 -0
  104. package/AISB/catalog/aisb.t3.047_timepfn.yaml +156 -0
  105. package/AISB/catalog/aisb.t3.047_timepfn.zh.yaml +124 -0
  106. package/AISB/catalog/aisb.t3.048_proxyspex.yaml +148 -0
  107. package/AISB/catalog/aisb.t3.048_proxyspex.zh.yaml +125 -0
  108. package/AISB/catalog/aisb.t3.049_hogwildinference.yaml +183 -0
  109. package/AISB/catalog/aisb.t3.049_hogwildinference.zh.yaml +138 -0
  110. package/AISB/catalog/aisb.t3.050_causalpfn.yaml +214 -0
  111. package/AISB/catalog/aisb.t3.050_causalpfn.zh.yaml +190 -0
  112. package/AISB/catalog/aisb.t3.051_flashtp.yaml +169 -0
  113. package/AISB/catalog/aisb.t3.051_flashtp.zh.yaml +124 -0
  114. package/AISB/catalog/aisb.t3.052_nsdiff.yaml +155 -0
  115. package/AISB/catalog/aisb.t3.052_nsdiff.zh.yaml +138 -0
  116. package/AISB/catalog/aisb.t3.053_k2vae.yaml +158 -0
  117. package/AISB/catalog/aisb.t3.053_k2vae.zh.yaml +132 -0
  118. package/AISB/catalog/aisb.t3.054_timebase.yaml +178 -0
  119. package/AISB/catalog/aisb.t3.054_timebase.zh.yaml +158 -0
  120. package/AISB/catalog/aisb.t3.055_csbrain.yaml +238 -0
  121. package/AISB/catalog/aisb.t3.055_csbrain.zh.yaml +184 -0
  122. package/AISB/catalog/aisb.t3.056_infosam.yaml +224 -0
  123. package/AISB/catalog/aisb.t3.056_infosam.zh.yaml +189 -0
  124. package/AISB/catalog/aisb.t3.057_mdreid.yaml +129 -0
  125. package/AISB/catalog/aisb.t3.057_mdreid.zh.yaml +117 -0
  126. package/AISB/catalog/aisb.t3.058_mindglitch.yaml +171 -0
  127. package/AISB/catalog/aisb.t3.058_mindglitch.zh.yaml +145 -0
  128. package/AISB/catalog/aisb.t3.059_selfsupervised.yaml +154 -0
  129. package/AISB/catalog/aisb.t3.059_selfsupervised.zh.yaml +125 -0
  130. package/AISB/catalog/aisb.t3.060_iaggad.yaml +121 -0
  131. package/AISB/catalog/aisb.t3.060_iaggad.zh.yaml +100 -0
  132. package/AISB/catalog/aisb.t3.061_hsgkn.yaml +136 -0
  133. package/AISB/catalog/aisb.t3.061_hsgkn.zh.yaml +113 -0
  134. package/AISB/catalog/aisb.t3.062_visionts.yaml +237 -0
  135. package/AISB/catalog/aisb.t3.062_visionts.zh.yaml +216 -0
  136. package/AISB/catalog/aisb.t3.063_tsrag.yaml +162 -0
  137. package/AISB/catalog/aisb.t3.063_tsrag.zh.yaml +138 -0
  138. package/AISB/catalog/aisb.t3.064_pir.yaml +221 -0
  139. package/AISB/catalog/aisb.t3.064_pir.zh.yaml +197 -0
  140. package/AISB/catalog/aisb.t3.065_proteinbinding.yaml +234 -0
  141. package/AISB/catalog/aisb.t3.065_proteinbinding.zh.yaml +167 -0
  142. package/AISB/catalog/aisb.t3.066_tropicalattention.yaml +267 -0
  143. package/AISB/catalog/aisb.t3.066_tropicalattention.zh.yaml +229 -0
  144. package/AISB/catalog/aisb.t3.067_kanad.yaml +193 -0
  145. package/AISB/catalog/aisb.t3.067_kanad.zh.yaml +167 -0
  146. package/AISB/catalog/aisb.t3.068_sempo.yaml +187 -0
  147. package/AISB/catalog/aisb.t3.068_sempo.zh.yaml +148 -0
  148. package/AISB/catalog/aisb.t3.069_treehfd.yaml +129 -0
  149. package/AISB/catalog/aisb.t3.069_treehfd.zh.yaml +111 -0
  150. package/AISB/catalog/aisb.t3.070_certifiedunlearning.yaml +224 -0
  151. package/AISB/catalog/aisb.t3.070_certifiedunlearning.zh.yaml +171 -0
  152. package/AISB/catalog/aisb.t3.071_neuralmjd.yaml +142 -0
  153. package/AISB/catalog/aisb.t3.071_neuralmjd.zh.yaml +120 -0
  154. package/AISB/catalog/aisb.t3.072_fedgmt.yaml +181 -0
  155. package/AISB/catalog/aisb.t3.072_fedgmt.zh.yaml +158 -0
  156. package/AISB/catalog/aisb.t3.073_rld.yaml +161 -0
  157. package/AISB/catalog/aisb.t3.073_rld.zh.yaml +129 -0
  158. package/AISB/catalog/aisb.t3.074_lsvi.yaml +163 -0
  159. package/AISB/catalog/aisb.t3.074_lsvi.zh.yaml +129 -0
  160. package/AISB/catalog/aisb.t3.075_treeslicedentropy.yaml +201 -0
  161. package/AISB/catalog/aisb.t3.075_treeslicedentropy.zh.yaml +148 -0
  162. package/AISB/catalog/aisb.t3.076_aanet.yaml +169 -0
  163. package/AISB/catalog/aisb.t3.076_aanet.zh.yaml +129 -0
  164. package/AISB/catalog/aisb.t3.077_cmnn.yaml +199 -0
  165. package/AISB/catalog/aisb.t3.077_cmnn.zh.yaml +165 -0
  166. package/AISB/catalog/aisb.t3.078_conformalanomaly.yaml +146 -0
  167. package/AISB/catalog/aisb.t3.078_conformalanomaly.zh.yaml +117 -0
  168. package/AISB/catalog/aisb.t3.079_dpfkmeans.yaml +131 -0
  169. package/AISB/catalog/aisb.t3.079_dpfkmeans.zh.yaml +104 -0
  170. package/AISB/catalog/aisb.t3.080_latentscorereweight.yaml +169 -0
  171. package/AISB/catalog/aisb.t3.080_latentscorereweight.zh.yaml +123 -0
  172. package/AISB/catalog/aisb.t3.081_qmamba.yaml +150 -0
  173. package/AISB/catalog/aisb.t3.081_qmamba.zh.yaml +117 -0
  174. package/AISB/catalog/aisb.t3.082_onlinellmrouting.yaml +160 -0
  175. package/AISB/catalog/aisb.t3.082_onlinellmrouting.zh.yaml +133 -0
  176. package/AISB/catalog/aisb.t3.083_starformer.yaml +178 -0
  177. package/AISB/catalog/aisb.t3.083_starformer.zh.yaml +140 -0
  178. package/AISB/catalog/aisb.t3.084_ift.yaml +139 -0
  179. package/AISB/catalog/aisb.t3.084_ift.zh.yaml +111 -0
  180. package/AISB/catalog/aisb.t3.085_neuralsurv.yaml +183 -0
  181. package/AISB/catalog/aisb.t3.085_neuralsurv.zh.yaml +143 -0
  182. package/AISB/catalog/aisb.t3.086_stella.yaml +197 -0
  183. package/AISB/catalog/aisb.t3.086_stella.zh.yaml +142 -0
  184. package/AISB/catalog/aisb.t3.087_moses.yaml +167 -0
  185. package/AISB/catalog/aisb.t3.087_moses.zh.yaml +132 -0
  186. package/AISB/catalog/aisb.t3.088_channelnorm.yaml +140 -0
  187. package/AISB/catalog/aisb.t3.088_channelnorm.zh.yaml +109 -0
  188. package/AISB/catalog/aisb.t3.089_causalvelocity.yaml +730 -0
  189. package/AISB/catalog/aisb.t3.089_causalvelocity.zh.yaml +668 -0
  190. package/AISB/catalog/aisb.t3.090_rstib.yaml +144 -0
  191. package/AISB/catalog/aisb.t3.090_rstib.zh.yaml +109 -0
  192. package/AISB/catalog/aisb.t3.091_timeawarecausal.yaml +132 -0
  193. package/AISB/catalog/aisb.t3.091_timeawarecausal.zh.yaml +107 -0
  194. package/AISB/catalog/aisb.t3.092_kmeanslocalopt.yaml +138 -0
  195. package/AISB/catalog/aisb.t3.092_kmeanslocalopt.zh.yaml +110 -0
  196. package/AISB/catalog/aisb.t3.093_fedwmsam.yaml +134 -0
  197. package/AISB/catalog/aisb.t3.093_fedwmsam.zh.yaml +106 -0
  198. package/AISB/catalog/aisb.t3.094_boundre.yaml +147 -0
  199. package/AISB/catalog/aisb.t3.094_boundre.zh.yaml +114 -0
  200. package/AISB/catalog/aisb.t3.095_fastfeaturecp.yaml +153 -0
  201. package/AISB/catalog/aisb.t3.095_fastfeaturecp.zh.yaml +118 -0
  202. package/AISB/catalog/aisb.t3.096_m3svm.yaml +189 -0
  203. package/AISB/catalog/aisb.t3.096_m3svm.zh.yaml +149 -0
  204. package/AISB/catalog/aisb.t3.097_wassersteintl.yaml +212 -0
  205. package/AISB/catalog/aisb.t3.097_wassersteintl.zh.yaml +169 -0
  206. package/AISB/catalog/aisb.t3.098_xmahalanobis.yaml +171 -0
  207. package/AISB/catalog/aisb.t3.098_xmahalanobis.zh.yaml +127 -0
  208. package/AISB/catalog/aisb.t3.099_ollalanding.yaml +248 -0
  209. package/AISB/catalog/aisb.t3.099_ollalanding.zh.yaml +182 -0
  210. package/AISB/catalog/aisb.t3.100_invmissingdata.yaml +179 -0
  211. package/AISB/catalog/aisb.t3.100_invmissingdata.zh.yaml +150 -0
  212. package/AISB/catalog/aisb.t3.101_acia.yaml +164 -0
  213. package/AISB/catalog/aisb.t3.101_acia.zh.yaml +109 -0
  214. package/AISB/catalog/aisb.t3.102_stochasticff.yaml +178 -0
  215. package/AISB/catalog/aisb.t3.102_stochasticff.zh.yaml +130 -0
  216. package/AISB/catalog/aisb.t3.103_qdcp.yaml +150 -0
  217. package/AISB/catalog/aisb.t3.103_qdcp.zh.yaml +116 -0
  218. package/AISB/catalog/aisb.t3.104_balancedactiveinf.yaml +137 -0
  219. package/AISB/catalog/aisb.t3.104_balancedactiveinf.zh.yaml +104 -0
  220. package/AISB/catalog/aisb.t3.105_binaryclasseval.yaml +161 -0
  221. package/AISB/catalog/aisb.t3.105_binaryclasseval.zh.yaml +130 -0
  222. package/AISB/image/001_aisb.t3.001_savvy.jpg +0 -0
  223. package/AISB/image/002_aisb.t3.002_pinet.jpg +0 -0
  224. package/AISB/image/003_aisb.t3.003_dmsqd.jpg +0 -0
  225. package/AISB/image/004_aisb.t3.004_decentralattn.jpg +0 -0
  226. package/AISB/image/005_aisb.t3.005_tsae.jpg +0 -0
  227. package/AISB/image/006_aisb.t3.006_physense.jpg +0 -0
  228. package/AISB/image/007_aisb.t3.007_reasoningiqa.jpg +0 -0
  229. package/AISB/image/008_aisb.t3.008_meanflows.jpg +0 -0
  230. package/AISB/image/009_aisb.t3.009_scoremissing.jpg +0 -0
  231. package/AISB/image/010_aisb.t3.010_suitabilityfilter.jpg +0 -0
  232. package/AISB/image/011_aisb.t3.011_osd.jpg +0 -0
  233. package/AISB/image/012_aisb.t3.012_efficientqat.jpg +0 -0
  234. package/AISB/image/013_aisb.t3.013_appl.jpg +0 -0
  235. package/AISB/image/014_aisb.t3.014_piguard.jpg +0 -0
  236. package/AISB/image/015_aisb.t3.015_frspec.jpg +0 -0
  237. package/AISB/image/016_aisb.t3.016_mathfusion.jpg +0 -0
  238. package/AISB/image/017_aisb.t3.017_multimodalglp.jpg +0 -0
  239. package/AISB/image/018_aisb.t3.018_cotsynth.jpg +0 -0
  240. package/AISB/image/019_aisb.t3.019_dyscaleut.jpg +0 -0
  241. package/AISB/image/020_aisb.t3.020_aristotle.jpg +0 -0
  242. package/AISB/image/021_aisb.t3.021_tokenrecycling.jpg +0 -0
  243. package/AISB/image/022_aisb.t3.022_chainofreasoning.jpg +0 -0
  244. package/AISB/image/023_aisb.t3.023_guidedembed.jpg +0 -0
  245. package/AISB/image/024_aisb.t3.024_outputcentric.jpg +0 -0
  246. package/AISB/image/025_aisb.t3.025_deeper.jpg +0 -0
  247. package/AISB/image/026_aisb.t3.026_gartkg.jpg +0 -0
  248. package/AISB/image/027_aisb.t3.027_citeeval.jpg +0 -0
  249. package/AISB/image/028_aisb.t3.028_sbam.jpg +0 -0
  250. package/AISB/image/029_aisb.t3.029_cdqgeoembed.jpg +0 -0
  251. package/AISB/image/030_aisb.t3.030_processrm.jpg +0 -0
  252. package/AISB/image/031_aisb.t3.031_circuitstability.jpg +0 -0
  253. package/AISB/image/032_aisb.t3.032_ptsolver.jpg +0 -0
  254. package/AISB/image/033_aisb.t3.033_gcse.jpg +0 -0
  255. package/AISB/image/034_aisb.t3.034_ensemblewm.jpg +0 -0
  256. package/AISB/image/035_aisb.t3.035_moralvalueswa.jpg +0 -0
  257. package/AISB/image/036_aisb.t3.036_weakstrongpref.jpg +0 -0
  258. package/AISB/image/037_aisb.t3.037_dementiamask.jpg +0 -0
  259. package/AISB/image/038_aisb.t3.038_tinysam.jpg +0 -0
  260. package/AISB/image/039_aisb.t3.039_calf.jpg +0 -0
  261. package/AISB/image/040_aisb.t3.040_graniteguardian.jpg +0 -0
  262. package/AISB/image/041_aisb.t3.041_amdm.jpg +0 -0
  263. package/AISB/image/042_aisb.t3.042_xpatch.jpg +0 -0
  264. package/AISB/image/043_aisb.t3.043_vhm.jpg +0 -0
  265. package/AISB/image/044_aisb.t3.044_rgvi.jpg +0 -0
  266. package/AISB/image/045_aisb.t3.045_pslstm.jpg +0 -0
  267. package/AISB/image/046_aisb.t3.046_nonstatts.jpg +0 -0
  268. package/AISB/image/047_aisb.t3.047_timepfn.jpg +0 -0
  269. package/AISB/image/048_aisb.t3.048_proxyspex.jpg +0 -0
  270. package/AISB/image/049_aisb.t3.049_hogwildinference.jpg +0 -0
  271. package/AISB/image/050_aisb.t3.050_causalpfn.jpg +0 -0
  272. package/AISB/image/051_aisb.t3.051_flashtp.jpg +0 -0
  273. package/AISB/image/052_aisb.t3.052_nsdiff.jpg +0 -0
  274. package/AISB/image/053_aisb.t3.053_k2vae.jpg +0 -0
  275. package/AISB/image/054_aisb.t3.054_timebase.jpg +0 -0
  276. package/AISB/image/055_aisb.t3.055_csbrain.jpg +0 -0
  277. package/AISB/image/056_aisb.t3.056_infosam.jpg +0 -0
  278. package/AISB/image/057_aisb.t3.057_mdreid.jpg +0 -0
  279. package/AISB/image/058_aisb.t3.058_mindglitch.jpg +0 -0
  280. package/AISB/image/059_aisb.t3.059_selfsupervised.jpg +0 -0
  281. package/AISB/image/060_aisb.t3.060_iaggad.jpg +0 -0
  282. package/AISB/image/061_aisb.t3.061_hsgkn.jpg +0 -0
  283. package/AISB/image/062_aisb.t3.062_visionts.jpg +0 -0
  284. package/AISB/image/063_aisb.t3.063_tsrag.jpg +0 -0
  285. package/AISB/image/064_aisb.t3.064_pir.jpg +0 -0
  286. package/AISB/image/065_aisb.t3.065_proteinbinding.jpg +0 -0
  287. package/AISB/image/066_aisb.t3.066_tropicalattention.jpg +0 -0
  288. package/AISB/image/067_aisb.t3.067_kanad.jpg +0 -0
  289. package/AISB/image/068_aisb.t3.068_sempo.jpg +0 -0
  290. package/AISB/image/069_aisb.t3.069_treehfd.jpg +0 -0
  291. package/AISB/image/070_aisb.t3.070_certifiedunlearning.jpg +0 -0
  292. package/AISB/image/071_aisb.t3.071_neuralmjd.jpg +0 -0
  293. package/AISB/image/072_aisb.t3.072_fedgmt.jpg +0 -0
  294. package/AISB/image/073_aisb.t3.073_rld.jpg +0 -0
  295. package/AISB/image/074_aisb.t3.074_lsvi.jpg +0 -0
  296. package/AISB/image/075_aisb.t3.075_treeslicedentropy.jpg +0 -0
  297. package/AISB/image/076_aisb.t3.076_aanet.jpg +0 -0
  298. package/AISB/image/077_aisb.t3.077_cmnn.jpg +0 -0
  299. package/AISB/image/078_aisb.t3.078_conformalanomaly.jpg +0 -0
  300. package/AISB/image/079_aisb.t3.079_dpfkmeans.jpg +0 -0
  301. package/AISB/image/080_aisb.t3.080_latentscorereweight.jpg +0 -0
  302. package/AISB/image/081_aisb.t3.081_qmamba.jpg +0 -0
  303. package/AISB/image/082_aisb.t3.082_onlinellmrouting.jpg +0 -0
  304. package/AISB/image/083_aisb.t3.083_starformer.jpg +0 -0
  305. package/AISB/image/084_aisb.t3.084_ift.jpg +0 -0
  306. package/AISB/image/085_aisb.t3.085_neuralsurv.jpg +0 -0
  307. package/AISB/image/086_aisb.t3.086_stella.jpg +0 -0
  308. package/AISB/image/087_aisb.t3.087_moses.jpg +0 -0
  309. package/AISB/image/088_aisb.t3.088_channelnorm.jpg +0 -0
  310. package/AISB/image/089_aisb.t3.089_causalvelocity.jpg +0 -0
  311. package/AISB/image/090_aisb.t3.090_rstib.jpg +0 -0
  312. package/AISB/image/091_aisb.t3.091_timeawarecausal.jpg +0 -0
  313. package/AISB/image/092_aisb.t3.092_kmeanslocalopt.jpg +0 -0
  314. package/AISB/image/093_aisb.t3.093_fedwmsam.jpg +0 -0
  315. package/AISB/image/094_aisb.t3.094_boundre.jpg +0 -0
  316. package/AISB/image/095_aisb.t3.095_fastfeaturecp.jpg +0 -0
  317. package/AISB/image/096_aisb.t3.096_m3svm.jpg +0 -0
  318. package/AISB/image/097_aisb.t3.097_wassersteintl.jpg +0 -0
  319. package/AISB/image/098_aisb.t3.098_xmahalanobis.jpg +0 -0
  320. package/AISB/image/099_aisb.t3.099_ollalanding.jpg +0 -0
  321. package/AISB/image/100_aisb.t3.100_invmissingdata.jpg +0 -0
  322. package/AISB/image/101_aisb.t3.101_acia.jpg +0 -0
  323. package/AISB/image/102_aisb.t3.102_stochasticff.jpg +0 -0
  324. package/AISB/image/103_aisb.t3.103_qdcp.jpg +0 -0
  325. package/AISB/image/104_aisb.t3.104_balancedactiveinf.jpg +0 -0
  326. package/AISB/image/105_aisb.t3.105_binaryclasseval.jpg +0 -0
  327. package/AISB/image/106_aisb.t1.reasoning_lite.jpg +0 -0
  328. package/AISB/image/107_aisb.t2.paper_audit.jpg +0 -0
  329. package/AISB/image/108_aisb.t3.multi_gpu_search.jpg +0 -0
  330. package/AISB/image/109_aisb.t3.tdc_admet.jpg +0 -0
  331. package/AISB/image/aisb.b1.agentic_coding.svg +16 -0
  332. package/AISB/image/aisb.b10.climate_earth.svg +16 -0
  333. package/AISB/image/aisb.b11.model_efficiency.svg +16 -0
  334. package/AISB/image/aisb.b12.embodied_ai.svg +16 -0
  335. package/AISB/image/aisb.b2.agent_systems.svg +16 -0
  336. package/AISB/image/aisb.b3.self_evolving_rl.svg +16 -0
  337. package/AISB/image/aisb.b4.lm_reasoning.svg +16 -0
  338. package/AISB/image/aisb.b5.math_proof.svg +16 -0
  339. package/AISB/image/aisb.b6.research_process.svg +16 -0
  340. package/AISB/image/aisb.b7.multimodal_fusion.svg +16 -0
  341. package/AISB/image/aisb.b8.lifesci_drug.svg +16 -0
  342. package/AISB/image/aisb.b9.material_science.svg +16 -0
  343. package/README.md +132 -11
  344. package/bin/ds.js +376 -49
  345. package/docs/en/00_QUICK_START.md +135 -18
  346. package/docs/en/01_SETTINGS_REFERENCE.md +468 -96
  347. package/docs/en/02_START_RESEARCH_GUIDE.md +26 -5
  348. package/docs/en/03_QQ_CONNECTOR_GUIDE.md +14 -3
  349. package/docs/en/04_LINGZHU_CONNECTOR_GUIDE.md +2 -0
  350. package/docs/en/05_TUI_GUIDE.md +171 -2
  351. package/docs/en/07_MEMORY_AND_MCP.md +38 -2
  352. package/docs/en/09_DOCTOR.md +64 -4
  353. package/docs/en/10_WEIXIN_CONNECTOR_GUIDE.md +38 -1
  354. package/docs/en/11_LICENSE_AND_RISK.md +4 -0
  355. package/docs/en/12_GUIDED_WORKFLOW_TOUR.md +15 -0
  356. package/docs/en/14_PROMPT_SKILLS_AND_MCP_GUIDE.md +9 -0
  357. package/docs/en/15_CODEX_PROVIDER_SETUP.md +622 -187
  358. package/docs/en/16_TELEGRAM_CONNECTOR_GUIDE.md +14 -0
  359. package/docs/en/17_WHATSAPP_CONNECTOR_GUIDE.md +14 -0
  360. package/docs/en/18_FEISHU_CONNECTOR_GUIDE.md +14 -0
  361. package/docs/en/21_LOCAL_MODEL_BACKENDS_GUIDE.md +105 -2
  362. package/docs/en/22_BENCHSTORE_YAML_REFERENCE.md +469 -0
  363. package/docs/en/23_BENCHSTORE_GITHUB_RELEASES_SPEC.md +316 -0
  364. package/docs/en/24_CLAUDE_CODE_PROVIDER_SETUP.md +469 -0
  365. package/docs/en/25_OPENCODE_PROVIDER_SETUP.md +653 -0
  366. package/docs/en/26_CITATION_AND_ATTRIBUTION.md +119 -0
  367. package/docs/en/27_KIMI_CODE_PROVIDER_SETUP.md +180 -0
  368. package/docs/en/28_DISCORD_CONNECTOR_GUIDE.md +61 -0
  369. package/docs/en/29_SLACK_CONNECTOR_GUIDE.md +60 -0
  370. package/docs/en/30_SETTINGS_CONTROL_CENTER_GUIDE.md +371 -0
  371. package/docs/en/{19_LOCAL_BROWSER_AUTH.md → 31_LOCAL_BROWSER_AUTH.md} +1 -1
  372. package/docs/en/32_WINDOWS_WSL2_DEPLOYMENT_GUIDE.md +273 -0
  373. package/docs/en/33_WORKSPACE_EXPLORER_QA.md +121 -0
  374. package/docs/en/91_DEVELOPMENT.md +29 -0
  375. package/docs/en/99_ACKNOWLEDGEMENTS.md +24 -19
  376. package/docs/en/README.md +44 -7
  377. package/docs/images/admin/admin-connectors-health-en.png +0 -0
  378. package/docs/images/admin/admin-controllers-en.png +0 -0
  379. package/docs/images/admin/admin-diagnostics-en.png +0 -0
  380. package/docs/images/admin/admin-errors-en.png +0 -0
  381. package/docs/images/admin/admin-issues-en.png +0 -0
  382. package/docs/images/admin/admin-logs-en.png +0 -0
  383. package/docs/images/admin/admin-quest-detail-en.png +0 -0
  384. package/docs/images/admin/admin-quests-en.png +0 -0
  385. package/docs/images/admin/admin-repairs-en.png +0 -0
  386. package/docs/images/admin/admin-runtime-en.png +0 -0
  387. package/docs/images/admin/admin-search-en.png +0 -0
  388. package/docs/images/admin/admin-stats-en.png +0 -0
  389. package/docs/images/admin/admin-summary-en.png +0 -0
  390. package/docs/images/connectors/connector-discord-en.png +0 -0
  391. package/docs/images/connectors/connector-feishu-en.png +0 -0
  392. package/docs/images/connectors/connector-lingzhu-en.png +0 -0
  393. package/docs/images/connectors/connector-qq-en.png +0 -0
  394. package/docs/images/connectors/connector-slack-en.png +0 -0
  395. package/docs/images/connectors/connector-telegram-en.png +0 -0
  396. package/docs/images/connectors/connector-weixin-en.png +0 -0
  397. package/docs/images/connectors/connector-whatsapp-en.png +0 -0
  398. package/docs/images/settings/settings-baselines-en.png +0 -0
  399. package/docs/images/settings/settings-config-en.png +0 -0
  400. package/docs/images/settings/settings-connectors-overview-en.png +0 -0
  401. package/docs/images/settings/settings-deepxiv-en.png +0 -0
  402. package/docs/images/settings/settings-mcp-servers-en.png +0 -0
  403. package/docs/images/settings/settings-plugins-en.png +0 -0
  404. package/docs/images/settings/settings-runners-en.png +0 -0
  405. package/docs/zh/00_QUICK_START.md +92 -17
  406. package/docs/zh/01_SETTINGS_REFERENCE.md +219 -98
  407. package/docs/zh/02_START_RESEARCH_GUIDE.md +26 -5
  408. package/docs/zh/05_TUI_GUIDE.md +171 -2
  409. package/docs/zh/07_MEMORY_AND_MCP.md +29 -2
  410. package/docs/zh/09_DOCTOR.md +39 -4
  411. package/docs/zh/10_WEIXIN_CONNECTOR_GUIDE.md +24 -1
  412. package/docs/zh/11_LICENSE_AND_RISK.md +4 -0
  413. package/docs/zh/12_GUIDED_WORKFLOW_TOUR.md +15 -0
  414. package/docs/zh/14_PROMPT_SKILLS_AND_MCP_GUIDE.md +9 -0
  415. package/docs/zh/15_CODEX_PROVIDER_SETUP.md +550 -188
  416. package/docs/zh/21_LOCAL_MODEL_BACKENDS_GUIDE.md +105 -2
  417. package/docs/zh/22_BENCHSTORE_YAML_REFERENCE.md +459 -0
  418. package/docs/zh/23_BENCHSTORE_GITHUB_RELEASES_SPEC.md +287 -0
  419. package/docs/zh/23_CLAUDE_RUNNER_GUIDE.md +103 -0
  420. package/docs/zh/24_CLAUDE_CODE_PROVIDER_SETUP.md +460 -0
  421. package/docs/zh/25_OPENCODE_PROVIDER_SETUP.md +660 -0
  422. package/docs/zh/26_CITATION_AND_ATTRIBUTION.md +102 -0
  423. package/docs/zh/27_KIMI_CODE_PROVIDER_SETUP.md +51 -0
  424. package/docs/zh/{19_LOCAL_BROWSER_AUTH.md → 31_LOCAL_BROWSER_AUTH.md} +1 -1
  425. package/docs/zh/32_WINDOWS_WSL2_DEPLOYMENT_GUIDE.md +264 -0
  426. package/docs/zh/33_WORKSPACE_EXPLORER_QA.md +127 -0
  427. package/docs/zh/99_ACKNOWLEDGEMENTS.md +23 -19
  428. package/docs/zh/README.md +29 -7
  429. package/install.sh +122 -16
  430. package/package.json +4 -1
  431. package/pyproject.toml +2 -1
  432. package/src/deepscientist/__init__.py +1 -1
  433. package/src/deepscientist/acp/envelope.py +13 -0
  434. package/src/deepscientist/admin/__init__.py +3 -0
  435. package/src/deepscientist/admin/charts.py +681 -0
  436. package/src/deepscientist/admin/logs.py +119 -0
  437. package/src/deepscientist/admin/repairs.py +217 -0
  438. package/src/deepscientist/admin/service.py +1310 -0
  439. package/src/deepscientist/admin/system_info.py +700 -0
  440. package/src/deepscientist/admin/tasks.py +465 -0
  441. package/src/deepscientist/admin/tool_metrics.py +600 -0
  442. package/src/deepscientist/artifact/guidance.py +8 -4
  443. package/src/deepscientist/artifact/schemas.py +115 -0
  444. package/src/deepscientist/artifact/service.py +4268 -260
  445. package/src/deepscientist/bash_exec/monitor.py +30 -3
  446. package/src/deepscientist/bash_exec/service.py +134 -1
  447. package/src/deepscientist/benchstore/__init__.py +4 -0
  448. package/src/deepscientist/benchstore/prompt_builder.py +224 -0
  449. package/src/deepscientist/benchstore/service.py +1716 -0
  450. package/src/deepscientist/channels/weixin_ilink.py +8 -1
  451. package/src/deepscientist/cli.py +92 -17
  452. package/src/deepscientist/codex_cli_compat.py +2 -2
  453. package/src/deepscientist/config/models.py +82 -11
  454. package/src/deepscientist/config/service.py +927 -91
  455. package/src/deepscientist/connector/weixin_support.py +48 -17
  456. package/src/deepscientist/daemon/api/handlers.py +697 -210
  457. package/src/deepscientist/daemon/api/router.py +76 -1
  458. package/src/deepscientist/daemon/app.py +1054 -51
  459. package/src/deepscientist/diagnostics/runner_failures.py +147 -0
  460. package/src/deepscientist/doctor.py +212 -65
  461. package/src/deepscientist/evidence_packets.py +590 -0
  462. package/src/deepscientist/home.py +52 -4
  463. package/src/deepscientist/kimi_cli_compat.py +50 -0
  464. package/src/deepscientist/latex_runtime.py +2 -2
  465. package/src/deepscientist/mcp/context.py +2 -0
  466. package/src/deepscientist/mcp/schemas.py +114 -0
  467. package/src/deepscientist/mcp/server.py +1566 -126
  468. package/src/deepscientist/memory/service.py +203 -16
  469. package/src/deepscientist/process_control.py +8 -1
  470. package/src/deepscientist/prompts/builder.py +836 -92
  471. package/src/deepscientist/quest/__init__.py +2 -2
  472. package/src/deepscientist/quest/layout.py +12 -1
  473. package/src/deepscientist/quest/node_traces.py +10 -0
  474. package/src/deepscientist/quest/service.py +1430 -139
  475. package/src/deepscientist/quest/stage_views.py +1 -1
  476. package/src/deepscientist/runners/__init__.py +18 -0
  477. package/src/deepscientist/runners/base.py +89 -1
  478. package/src/deepscientist/runners/builtins.py +13 -1
  479. package/src/deepscientist/runners/claude.py +391 -0
  480. package/src/deepscientist/runners/codex.py +421 -21
  481. package/src/deepscientist/runners/codex_telemetry.py +127 -0
  482. package/src/deepscientist/runners/kimi.py +334 -0
  483. package/src/deepscientist/runners/metadata.py +68 -0
  484. package/src/deepscientist/runners/opencode.py +414 -0
  485. package/src/deepscientist/runners/runtime_overrides.py +100 -0
  486. package/src/deepscientist/runners/simple_cli.py +538 -0
  487. package/src/deepscientist/runtime_storage.py +303 -0
  488. package/src/deepscientist/shared.py +61 -16
  489. package/src/deepscientist/skills/installer.py +37 -0
  490. package/src/deepscientist/skills/registry.py +2 -0
  491. package/src/deepscientist/tinytex.py +2 -2
  492. package/src/deepscientist/tui.py +10 -3
  493. package/src/prompts/benchstore/system.md +77 -0
  494. package/src/prompts/connectors/qq.md +33 -2
  495. package/src/prompts/connectors/weixin.md +208 -23
  496. package/src/prompts/contracts/admin_ops.md +74 -0
  497. package/src/prompts/contracts/admin_ops_knowledge.md +138 -0
  498. package/src/prompts/contracts/shared_interaction.md +5 -11
  499. package/src/prompts/start_setup/system.md +422 -0
  500. package/src/prompts/system.md +409 -315
  501. package/src/prompts/system_copilot.md +88 -12
  502. package/src/skills/analysis-campaign/SKILL.md +239 -578
  503. package/src/skills/analysis-campaign/references/artifact-flow-examples.md +102 -0
  504. package/src/skills/analysis-campaign/references/boundary-cases.md +98 -0
  505. package/src/skills/analysis-campaign/references/campaign-checklist-template.md +39 -24
  506. package/src/skills/analysis-campaign/references/campaign-design.md +26 -10
  507. package/src/skills/analysis-campaign/references/campaign-plan-template.md +53 -54
  508. package/src/skills/analysis-campaign/references/operational-guidance.md +97 -0
  509. package/src/skills/analysis-campaign/references/writing-facing-slice-examples.md +10 -20
  510. package/src/skills/baseline/SKILL.md +183 -461
  511. package/src/skills/baseline/references/artifact-flow-examples.md +106 -0
  512. package/src/skills/baseline/references/artifact-payload-examples.md +1 -1
  513. package/src/skills/baseline/references/baseline-checklist-template.md +27 -35
  514. package/src/skills/baseline/references/baseline-plan-template.md +37 -76
  515. package/src/skills/baseline/references/boundary-cases.md +86 -0
  516. package/src/skills/baseline/references/codebase-audit-checklist.md +2 -6
  517. package/src/skills/baseline/references/comparability-contract.md +7 -12
  518. package/src/skills/baseline/references/operational-guidance.md +56 -0
  519. package/src/skills/baseline/references/route-selection.md +5 -25
  520. package/src/skills/decision/SKILL.md +113 -306
  521. package/src/skills/decision/references/checkpoint-memory-template.md +47 -0
  522. package/src/skills/decision/references/operational-guidance.md +94 -0
  523. package/src/skills/decision/references/research-route-criteria.md +7 -8
  524. package/src/skills/decision/references/strategic-decision-template.md +13 -26
  525. package/src/skills/experiment/SKILL.md +132 -670
  526. package/src/skills/experiment/references/execution-playbook.md +374 -0
  527. package/src/skills/experiment/references/main-experiment-checklist-template.md +26 -2
  528. package/src/skills/experiment/references/main-experiment-plan-template.md +28 -17
  529. package/src/skills/experiment/references/operational-guidance.md +108 -0
  530. package/src/skills/finalize/SKILL.md +62 -0
  531. package/src/skills/finalize/references/checkpoint-memory-template.md +49 -0
  532. package/src/skills/finalize/references/resume-packet-template.md +7 -0
  533. package/src/skills/idea/SKILL.md +228 -15
  534. package/src/skills/idea/references/controlled-brainstorming-playbook.md +78 -0
  535. package/src/skills/idea/references/current-board-packet-template.md +61 -0
  536. package/src/skills/idea/references/high-value-idea-sourcing.md +119 -0
  537. package/src/skills/idea/references/idea-generation-playbook.md +21 -0
  538. package/src/skills/idea/references/idea-thinking-flow.md +6 -0
  539. package/src/skills/idea/references/literature-survey-template.md +3 -0
  540. package/src/skills/idea/references/objective-contract-template.md +54 -0
  541. package/src/skills/idea/references/outline-seeding-example.md +56 -0
  542. package/src/skills/idea/references/pre-idea-draft-template.md +105 -0
  543. package/src/skills/idea/references/related-work-playbook.md +75 -2
  544. package/src/skills/idea/references/research-history-playbook.md +114 -0
  545. package/src/skills/idea/references/selection-gate.md +58 -6
  546. package/src/skills/intake-audit/SKILL.md +43 -2
  547. package/src/skills/intake-audit/references/state-audit-template.md +10 -0
  548. package/src/skills/nature-data/SKILL.md +128 -0
  549. package/src/skills/nature-data/UPSTREAM_LICENSE.txt +21 -0
  550. package/src/skills/nature-data/agents/openai.yaml +4 -0
  551. package/src/skills/nature-data/references/chinese-author-alignment.md +84 -0
  552. package/src/skills/nature-data/references/fair-metadata-checklist.md +105 -0
  553. package/src/skills/nature-data/references/policy-principles.md +103 -0
  554. package/src/skills/nature-data/references/repository-and-identifiers.md +96 -0
  555. package/src/skills/nature-data/references/source-basis.md +54 -0
  556. package/src/skills/nature-data/references/statement-patterns.md +153 -0
  557. package/src/skills/nature-figure/SKILL.md +197 -0
  558. package/src/skills/nature-figure/UPSTREAM_LICENSE.txt +21 -0
  559. package/src/skills/nature-figure/agents/openai.yaml +4 -0
  560. package/src/skills/nature-figure/evals/evals.json +37 -0
  561. package/src/skills/nature-figure/references/api.md +428 -0
  562. package/src/skills/nature-figure/references/backend-selection.md +100 -0
  563. package/src/skills/nature-figure/references/chart-types.md +281 -0
  564. package/src/skills/nature-figure/references/common-patterns.md +349 -0
  565. package/src/skills/nature-figure/references/design-theory.md +436 -0
  566. package/src/skills/nature-figure/references/figure-contract.md +93 -0
  567. package/src/skills/nature-figure/references/nature-2026-observations.md +112 -0
  568. package/src/skills/nature-figure/references/qa-contract.md +119 -0
  569. package/src/skills/nature-figure/references/r-template-index.md +66 -0
  570. package/src/skills/nature-figure/references/r-workflow.md +161 -0
  571. package/src/skills/nature-figure/references/tutorials.md +250 -0
  572. package/src/skills/nature-paper2ppt/SKILL.md +507 -0
  573. package/src/skills/nature-paper2ppt/UPSTREAM_LICENSE.txt +21 -0
  574. package/src/skills/nature-paper2ppt/agents/openai.yaml +4 -0
  575. package/src/skills/nature-polishing/SKILL.md +385 -0
  576. package/src/skills/nature-polishing/UPSTREAM_LICENSE.txt +21 -0
  577. package/src/skills/nature-polishing/agents/openai.yaml +4 -0
  578. package/src/skills/nature-polishing/references/phrasebank-playbook.md +162 -0
  579. package/src/skills/nature-polishing/references/section-moves.md +240 -0
  580. package/src/skills/nature-polishing/references/style-guardrails.md +94 -0
  581. package/src/skills/nature-polishing/references/writing-strategy.md +148 -0
  582. package/src/skills/optimize/SKILL.md +177 -1568
  583. package/src/skills/optimize/references/brief-shaping-playbook.md +95 -0
  584. package/src/skills/optimize/references/candidate-board-template.md +13 -0
  585. package/src/skills/optimize/references/candidate-ranking-template.md +51 -0
  586. package/src/skills/optimize/references/codegen-route-playbook.md +50 -0
  587. package/src/skills/optimize/references/debug-response-template.md +29 -0
  588. package/src/skills/optimize/references/frontier-review-template.md +32 -0
  589. package/src/skills/optimize/references/fusion-playbook.md +36 -0
  590. package/src/skills/optimize/references/method-brief-template.md +73 -0
  591. package/src/skills/optimize/references/operational-guidance.md +621 -0
  592. package/src/skills/optimize/references/optimization-memory-template.md +30 -0
  593. package/src/skills/optimize/references/optimize-checklist-template.md +18 -0
  594. package/src/skills/optimize/references/plateau-response-playbook.md +28 -0
  595. package/src/skills/optimize/references/prompt-patterns.md +49 -0
  596. package/src/skills/paper-outline/SKILL.md +227 -0
  597. package/src/skills/paper-outline/references/outline-patterns.md +87 -0
  598. package/src/skills/paper-plot/SKILL.md +79 -0
  599. package/src/skills/paper-plot/agents/openai.yaml +4 -0
  600. package/src/skills/paper-plot/references/bar_grouped_hatch.md +96 -0
  601. package/src/skills/paper-plot/references/bar_paired_delta.md +72 -0
  602. package/src/skills/paper-plot/references/line_confidence_band.md +75 -0
  603. package/src/skills/paper-plot/references/line_loss_with_inset.md +65 -0
  604. package/src/skills/paper-plot/references/line_training_curve.md +44 -0
  605. package/src/skills/paper-plot/references/radar_dual_series.md +59 -0
  606. package/src/skills/paper-plot/references/scatter_broken_axis.md +59 -0
  607. package/src/skills/paper-plot/references/scatter_tsne_cluster.md +72 -0
  608. package/src/skills/paper-plot/scripts/bar_memevolve.py +109 -0
  609. package/src/skills/paper-plot/scripts/bar_spice.py +166 -0
  610. package/src/skills/paper-plot/scripts/line_aime.py +94 -0
  611. package/src/skills/paper-plot/scripts/line_loss_inset.py +157 -0
  612. package/src/skills/paper-plot/scripts/line_selfdistill.py +168 -0
  613. package/src/skills/paper-plot/scripts/radar_dora.py +151 -0
  614. package/src/skills/paper-plot/scripts/scatter_break.py +169 -0
  615. package/src/skills/paper-plot/scripts/scatter_tsne.py +133 -0
  616. package/src/skills/rebuttal/SKILL.md +9 -0
  617. package/src/skills/references/tool-usage-by-stage.md +438 -0
  618. package/src/skills/review/SKILL.md +105 -7
  619. package/src/skills/science/PROVENANCE.md +44 -0
  620. package/src/skills/science/SKILL.md +137 -0
  621. package/src/skills/science/references/artifact-science-tool.md +110 -0
  622. package/src/skills/science/references/claim-type-discipline.md +56 -0
  623. package/src/skills/science/references/domain-index.md +422 -0
  624. package/src/skills/science/references/hpc-via-bash-exec.md +42 -0
  625. package/src/skills/science/references/package-check-playbook.md +64 -0
  626. package/src/skills/science/references/package-index.min.json +3616 -0
  627. package/src/skills/science/references/packages/abinit.md +80 -0
  628. package/src/skills/science/references/packages/acts.md +73 -0
  629. package/src/skills/science/references/packages/aiida-core.md +80 -0
  630. package/src/skills/science/references/packages/alamode.md +80 -0
  631. package/src/skills/science/references/packages/amuse.md +88 -0
  632. package/src/skills/science/references/packages/anndata.md +88 -0
  633. package/src/skills/science/references/packages/arbor.md +80 -0
  634. package/src/skills/science/references/packages/arc.md +73 -0
  635. package/src/skills/science/references/packages/astropy.md +88 -0
  636. package/src/skills/science/references/packages/astroquery.md +88 -0
  637. package/src/skills/science/references/packages/atomate2.md +80 -0
  638. package/src/skills/science/references/packages/atomsmltr.md +73 -0
  639. package/src/skills/science/references/packages/awkward.md +73 -0
  640. package/src/skills/science/references/packages/batman.md +88 -0
  641. package/src/skills/science/references/packages/biopython.md +88 -0
  642. package/src/skills/science/references/packages/bloqade.md +73 -0
  643. package/src/skills/science/references/packages/brian2.md +73 -0
  644. package/src/skills/science/references/packages/bullet3.md +73 -0
  645. package/src/skills/science/references/packages/calculix.md +80 -0
  646. package/src/skills/science/references/packages/cantera.md +73 -0
  647. package/src/skills/science/references/packages/cavity-md-ipi.md +80 -0
  648. package/src/skills/science/references/packages/ccdproc.md +88 -0
  649. package/src/skills/science/references/packages/celerite2.md +88 -0
  650. package/src/skills/science/references/packages/cellrank.md +73 -0
  651. package/src/skills/science/references/packages/cesm.md +80 -0
  652. package/src/skills/science/references/packages/chemicals.md +73 -0
  653. package/src/skills/science/references/packages/chempy.md +73 -0
  654. package/src/skills/science/references/packages/cirq.md +73 -0
  655. package/src/skills/science/references/packages/coffea.md +73 -0
  656. package/src/skills/science/references/packages/cp2k.md +88 -0
  657. package/src/skills/science/references/packages/custodian.md +80 -0
  658. package/src/skills/science/references/packages/dart.md +73 -0
  659. package/src/skills/science/references/packages/datamol.md +88 -0
  660. package/src/skills/science/references/packages/dd4hep.md +73 -0
  661. package/src/skills/science/references/packages/dealii.md +80 -0
  662. package/src/skills/science/references/packages/deepchem.md +88 -0
  663. package/src/skills/science/references/packages/delphes.md +73 -0
  664. package/src/skills/science/references/packages/devito.md +80 -0
  665. package/src/skills/science/references/packages/dftb.md +88 -0
  666. package/src/skills/science/references/packages/dftd4.md +88 -0
  667. package/src/skills/science/references/packages/dftk-jl.md +80 -0
  668. package/src/skills/science/references/packages/dolfinx.md +80 -0
  669. package/src/skills/science/references/packages/drake.md +73 -0
  670. package/src/skills/science/references/packages/dumux.md +73 -0
  671. package/src/skills/science/references/packages/elk.md +80 -0
  672. package/src/skills/science/references/packages/elmerfem.md +80 -0
  673. package/src/skills/science/references/packages/enzo-e.md +88 -0
  674. package/src/skills/science/references/packages/espresso.md +80 -0
  675. package/src/skills/science/references/packages/exoplanet.md +88 -0
  676. package/src/skills/science/references/packages/fairroot.md +73 -0
  677. package/src/skills/science/references/packages/fbpic.md +80 -0
  678. package/src/skills/science/references/packages/fdtdbath-meep.md +80 -0
  679. package/src/skills/science/references/packages/geant4.md +73 -0
  680. package/src/skills/science/references/packages/geosx.md +80 -0
  681. package/src/skills/science/references/packages/gprmax.md +80 -0
  682. package/src/skills/science/references/packages/gromacs.md +80 -0
  683. package/src/skills/science/references/packages/gwaslab.md +73 -0
  684. package/src/skills/science/references/packages/gz-sim.md +73 -0
  685. package/src/skills/science/references/packages/hail.md +88 -0
  686. package/src/skills/science/references/packages/hiphive.md +80 -0
  687. package/src/skills/science/references/packages/hoomd-blue.md +80 -0
  688. package/src/skills/science/references/packages/itensor.md +73 -0
  689. package/src/skills/science/references/packages/itensors-jl.md +73 -0
  690. package/src/skills/science/references/packages/jdftx.md +73 -0
  691. package/src/skills/science/references/packages/jobflow.md +80 -0
  692. package/src/skills/science/references/packages/kadanoffbaym-jl.md +73 -0
  693. package/src/skills/science/references/packages/kite.md +80 -0
  694. package/src/skills/science/references/packages/kratos.md +80 -0
  695. package/src/skills/science/references/packages/kwant.md +73 -0
  696. package/src/skills/science/references/packages/lammps.md +80 -0
  697. package/src/skills/science/references/packages/lightkurve.md +88 -0
  698. package/src/skills/science/references/packages/limix.md +73 -0
  699. package/src/skills/science/references/packages/maxwelllink.md +80 -0
  700. package/src/skills/science/references/packages/mcdc.md +73 -0
  701. package/src/skills/science/references/packages/meep.md +80 -0
  702. package/src/skills/science/references/packages/mfem.md +80 -0
  703. package/src/skills/science/references/packages/mitgcm.md +73 -0
  704. package/src/skills/science/references/packages/modflow6.md +73 -0
  705. package/src/skills/science/references/packages/molecool.md +73 -0
  706. package/src/skills/science/references/packages/mom6.md +73 -0
  707. package/src/skills/science/references/packages/moose.md +80 -0
  708. package/src/skills/science/references/packages/mpas-model.md +73 -0
  709. package/src/skills/science/references/packages/mujoco.md +73 -0
  710. package/src/skills/science/references/packages/mumax3.md +73 -0
  711. package/src/skills/science/references/packages/nekrs.md +80 -0
  712. package/src/skills/science/references/packages/nessi.md +73 -0
  713. package/src/skills/science/references/packages/nest-simulator.md +73 -0
  714. package/src/skills/science/references/packages/netket.md +73 -0
  715. package/src/skills/science/references/packages/neuron.md +73 -0
  716. package/src/skills/science/references/packages/nextflow.md +88 -0
  717. package/src/skills/science/references/packages/nwchem.md +88 -0
  718. package/src/skills/science/references/packages/openbabel.md +88 -0
  719. package/src/skills/science/references/packages/openems.md +80 -0
  720. package/src/skills/science/references/packages/openff-toolkit.md +88 -0
  721. package/src/skills/science/references/packages/openfoam-dev.md +80 -0
  722. package/src/skills/science/references/packages/openmc.md +73 -0
  723. package/src/skills/science/references/packages/openmm.md +80 -0
  724. package/src/skills/science/references/packages/openmoc.md +73 -0
  725. package/src/skills/science/references/packages/openmx.md +80 -0
  726. package/src/skills/science/references/packages/opensees.md +80 -0
  727. package/src/skills/science/references/packages/opensn.md +80 -0
  728. package/src/skills/science/references/packages/opm-simulators.md +73 -0
  729. package/src/skills/science/references/packages/oqupy.md +73 -0
  730. package/src/skills/science/references/packages/packmol.md +80 -0
  731. package/src/skills/science/references/packages/palabos.md +80 -0
  732. package/src/skills/science/references/packages/parflow.md +80 -0
  733. package/src/skills/science/references/packages/pennylane.md +88 -0
  734. package/src/skills/science/references/packages/perceval.md +73 -0
  735. package/src/skills/science/references/packages/phono3py.md +73 -0
  736. package/src/skills/science/references/packages/phonopy.md +73 -0
  737. package/src/skills/science/references/packages/photutils.md +88 -0
  738. package/src/skills/science/references/packages/picongpu.md +80 -0
  739. package/src/skills/science/references/packages/plink-ng.md +88 -0
  740. package/src/skills/science/references/packages/precice.md +73 -0
  741. package/src/skills/science/references/packages/psc.md +80 -0
  742. package/src/skills/science/references/packages/psi4.md +88 -0
  743. package/src/skills/science/references/packages/pybinding.md +73 -0
  744. package/src/skills/science/references/packages/pyfr.md +80 -0
  745. package/src/skills/science/references/packages/pyhf.md +73 -0
  746. package/src/skills/science/references/packages/pyiron_base.md +80 -0
  747. package/src/skills/science/references/packages/pylcp.md +73 -0
  748. package/src/skills/science/references/packages/pylith.md +80 -0
  749. package/src/skills/science/references/packages/pynbody.md +88 -0
  750. package/src/skills/science/references/packages/pysam.md +88 -0
  751. package/src/skills/science/references/packages/pyscf.md +88 -0
  752. package/src/skills/science/references/packages/q-e.md +73 -0
  753. package/src/skills/science/references/packages/qibo.md +73 -0
  754. package/src/skills/science/references/packages/qiskit.md +73 -0
  755. package/src/skills/science/references/packages/quantica-jl.md +73 -0
  756. package/src/skills/science/references/packages/quantumoptics-jl.md +73 -0
  757. package/src/skills/science/references/packages/quimb.md +73 -0
  758. package/src/skills/science/references/packages/qulacs.md +73 -0
  759. package/src/skills/science/references/packages/qutip.md +73 -0
  760. package/src/skills/science/references/packages/rdkit.md +88 -0
  761. package/src/skills/science/references/packages/rmg-py.md +73 -0
  762. package/src/skills/science/references/packages/root.md +73 -0
  763. package/src/skills/science/references/packages/scanpy.md +88 -0
  764. package/src/skills/science/references/packages/scikit-allel.md +88 -0
  765. package/src/skills/science/references/packages/scikit-bio.md +88 -0
  766. package/src/skills/science/references/packages/scqubits.md +73 -0
  767. package/src/skills/science/references/packages/scuff-em.md +80 -0
  768. package/src/skills/science/references/packages/scvi-tools.md +73 -0
  769. package/src/skills/science/references/packages/seissol.md +73 -0
  770. package/src/skills/science/references/packages/sfepy.md +80 -0
  771. package/src/skills/science/references/packages/sisl.md +73 -0
  772. package/src/skills/science/references/packages/smilei.md +80 -0
  773. package/src/skills/science/references/packages/snakemake.md +88 -0
  774. package/src/skills/science/references/packages/specfem3d-globe.md +80 -0
  775. package/src/skills/science/references/packages/specutils.md +88 -0
  776. package/src/skills/science/references/packages/spglib.md +80 -0
  777. package/src/skills/science/references/packages/squidpy.md +88 -0
  778. package/src/skills/science/references/packages/starry.md +88 -0
  779. package/src/skills/science/references/packages/strawberryfields.md +73 -0
  780. package/src/skills/science/references/packages/su2.md +80 -0
  781. package/src/skills/science/references/packages/sunny-jl.md +73 -0
  782. package/src/skills/science/references/packages/sw4.md +73 -0
  783. package/src/skills/science/references/packages/swift.md +88 -0
  784. package/src/skills/science/references/packages/tdnegf.md +73 -0
  785. package/src/skills/science/references/packages/tenpy.md +73 -0
  786. package/src/skills/science/references/packages/thermo.md +73 -0
  787. package/src/skills/science/references/packages/tkwant.md +73 -0
  788. package/src/skills/science/references/packages/tvb-root.md +73 -0
  789. package/src/skills/science/references/packages/uproot5.md +73 -0
  790. package/src/skills/science/references/packages/vampire.md +80 -0
  791. package/src/skills/science/references/packages/wannier_tools.md +73 -0
  792. package/src/skills/science/references/packages/warpx.md +80 -0
  793. package/src/skills/science/references/packages/wrf.md +73 -0
  794. package/src/skills/science/references/packages/xtb.md +88 -0
  795. package/src/skills/science/references/packages/yt.md +73 -0
  796. package/src/skills/science/references/science-task-brief-template.md +71 -0
  797. package/src/skills/scout/SKILL.md +83 -425
  798. package/src/skills/scout/references/literature-scout-template.md +5 -24
  799. package/src/skills/scout/references/operational-guidance.md +191 -0
  800. package/src/skills/scout/references/paper-triage-playbook.md +11 -35
  801. package/src/skills/write/SKILL.md +744 -1246
  802. package/src/skills/write/references/experiments_analysis_patterns.md +129 -0
  803. package/src/skills/write/references/oral_package_patterns.md +252 -0
  804. package/src/skills/write/references/oral_writing_principles.md +291 -0
  805. package/src/skills/write/references/section_rewrite_checklist.md +234 -0
  806. package/src/tui/dist/app/AppContainer.js +1314 -27
  807. package/src/tui/dist/components/Composer.js +26 -1
  808. package/src/tui/dist/components/ConfigScreen.js +2 -1
  809. package/src/tui/dist/components/InputPrompt.js +25 -9
  810. package/src/tui/dist/components/MainContent.js +18 -3
  811. package/src/tui/dist/components/QuestScreen.js +3 -2
  812. package/src/tui/dist/components/UtilityScreen.js +37 -0
  813. package/src/tui/dist/hooks/useSafeInput.js +10 -0
  814. package/src/tui/dist/index.js +13 -1
  815. package/src/tui/dist/layouts/DefaultAppLayout.js +11 -8
  816. package/src/tui/dist/lib/api.js +89 -1
  817. package/src/tui/package.json +1 -1
  818. package/src/ui/dist/assets/{AnalysisPlugin-BCKAfjba.js → AnalysisPlugin-CA94NGmI.js} +1 -1
  819. package/src/ui/dist/assets/CliPlugin-DHBzphZU.js +79 -0
  820. package/src/ui/dist/assets/CodeEditorPlugin-BOFwD2rn.js +2 -0
  821. package/src/ui/dist/assets/{CodeViewerPlugin-CbaFRrUU.js → CodeViewerPlugin-CqDpgjik.js} +4 -4
  822. package/src/ui/dist/assets/{DocViewerPlugin-DAjLVeQD.js → DocViewerPlugin-UDBgt8-4.js} +3 -3
  823. package/src/ui/dist/assets/GitCommitViewerPlugin-BmHtZ0bZ.js +6 -0
  824. package/src/ui/dist/assets/{GitDiffViewerPlugin-CQACjoAA.js → GitDiffViewerPlugin-CAxjNorQ.js} +2 -2
  825. package/src/ui/dist/assets/{GitSnapshotViewer-0r4nLPke.js → GitSnapshotViewer-CweA6VON.js} +2 -2
  826. package/src/ui/dist/assets/{ImageViewerPlugin-nBOmI2v_.js → ImageViewerPlugin-C8wHGvGN.js} +5 -5
  827. package/src/ui/dist/assets/LabPlugin-COyyLUol.js +32 -0
  828. package/src/ui/dist/assets/{LatexPlugin-ZwtV8pIp.js → LatexPlugin-BQjAaA5J.js} +4 -4
  829. package/src/ui/dist/assets/{MarkdownViewerPlugin-DKqVfKyW.js → MarkdownViewerPlugin-Dy1NE2dI.js} +3 -3
  830. package/src/ui/dist/assets/{MarketplacePlugin-BwxStZ9D.js → MarketplacePlugin-DMIZtEJ2.js} +2 -2
  831. package/src/ui/dist/assets/NotebookEditor-CFHMq_Qt.js +91 -0
  832. package/src/ui/dist/assets/{NotebookEditor-DB9N_T9q.js → NotebookEditor-WFyd8Ybt.js} +3 -3
  833. package/src/ui/dist/assets/{PdfLoader-eWBONbQP.js → PdfLoader-CLE5u5TS.js} +3 -3
  834. package/src/ui/dist/assets/{PdfMarkdownPlugin-D22YOZL3.js → PdfMarkdownPlugin-_iNK_H83.js} +1 -1
  835. package/src/ui/dist/assets/PdfViewerPlugin-DgWsbInT.js +22 -0
  836. package/src/ui/dist/assets/SearchPlugin-DrZmn5iw.js +11 -0
  837. package/src/ui/dist/assets/{TextViewerPlugin-C5xqeeUH.js → TextViewerPlugin-D1-T3aC7.js} +4 -4
  838. package/src/ui/dist/assets/branding/runner-claude.svg +107 -0
  839. package/src/ui/dist/assets/branding/runner-codex.svg +10 -0
  840. package/src/ui/dist/assets/branding/runner-kimi.svg +14 -0
  841. package/src/ui/dist/assets/branding/runner-opencode.svg +7 -0
  842. package/src/ui/dist/assets/cli-store-CoZ-x5Ip.js +1 -0
  843. package/src/ui/dist/assets/{code-WlFHE7z_.js → code-DbsmSd3Y.js} +1 -1
  844. package/src/ui/dist/assets/file-diff-panel-DsvyRz47.js +1 -0
  845. package/src/ui/dist/assets/{wrap-text-BC-Hltpd.js → file-jump-queue-DeQBikaw.js} +3 -3
  846. package/src/ui/dist/assets/{file-socket-CfQPKQKj.js → file-socket-DA5XIx88.js} +1 -1
  847. package/src/ui/dist/assets/fonts/ds-fonts.css +50 -4
  848. package/src/ui/dist/assets/images/deepxiv/register-guide.png +0 -0
  849. package/src/ui/dist/assets/index-39vY9LmZ.js +1 -0
  850. package/src/ui/dist/assets/{index-CwNu1aH4.js → index-BsO46tJA.js} +1 -1
  851. package/src/ui/dist/assets/index-CHzJ2xtB.js +3530 -0
  852. package/src/ui/dist/assets/index-DH-zxoZ3.css +33 -0
  853. package/src/ui/dist/assets/{plugin-notebook-HbW2K-1c.js → plugin-notebook-JRhysCqj.js} +2 -2
  854. package/src/ui/dist/assets/{project-sync-C9IdzdZW.js → project-sync-DPmWKmKD.js} +1 -1
  855. package/src/ui/dist/assets/{zoom-out-E_gaeAxL.js → zoom-out-DAukFWen.js} +3 -3
  856. package/src/ui/dist/index.html +3 -3
  857. package/src/skills/analysis-campaign/references/artifact-orchestration.md +0 -58
  858. package/src/skills/baseline/references/memory-playbook.md +0 -40
  859. package/src/skills/baseline/references/publishable-baseline-package.md +0 -30
  860. package/src/skills/write/references/outline-evidence-contract-example.md +0 -107
  861. package/src/skills/write/references/paper-experiment-matrix-template.md +0 -131
  862. package/src/skills/write/references/paper-section-playbook.md +0 -64
  863. package/src/skills/write/references/reviewer-first-writing.md +0 -64
  864. package/src/skills/write/references/revision-checklist.md +0 -70
  865. package/src/skills/write/references/section-contracts.md +0 -82
  866. package/src/skills/write/references/sentence-level-proofing.md +0 -49
  867. package/src/ui/dist/assets/AiManusChatView-Bv-Z8YpU.js +0 -204
  868. package/src/ui/dist/assets/CliPlugin-BCKcpc35.js +0 -109
  869. package/src/ui/dist/assets/CodeEditorPlugin-DbOfSJ8K.js +0 -2
  870. package/src/ui/dist/assets/GitCommitViewerPlugin-CIUqbUDO.js +0 -1
  871. package/src/ui/dist/assets/LabCopilotPanel-BHxOxF4z.js +0 -14
  872. package/src/ui/dist/assets/LabPlugin-BKoZGs95.js +0 -22
  873. package/src/ui/dist/assets/NotebookEditor-BEQhaQbt.js +0 -81
  874. package/src/ui/dist/assets/PdfViewerPlugin-c-RK9DLM.js +0 -17
  875. package/src/ui/dist/assets/SearchPlugin-CxF9ytAx.js +0 -16
  876. package/src/ui/dist/assets/VNCViewer-BoLGLnHz.js +0 -11
  877. package/src/ui/dist/assets/bot-DREQOxzP.js +0 -6
  878. package/src/ui/dist/assets/chevron-up-C9Qpx4DE.js +0 -6
  879. package/src/ui/dist/assets/file-content-BZMz3RYp.js +0 -1
  880. package/src/ui/dist/assets/file-diff-panel-CQhw0jS2.js +0 -1
  881. package/src/ui/dist/assets/file-jump-queue-DA-SdG__.js +0 -1
  882. package/src/ui/dist/assets/git-commit-horizontal-DxZ8DCZh.js +0 -6
  883. package/src/ui/dist/assets/image-Bgl4VIyx.js +0 -6
  884. package/src/ui/dist/assets/index-BpV6lusQ.css +0 -33
  885. package/src/ui/dist/assets/index-CBNVuWcP.js +0 -2496
  886. package/src/ui/dist/assets/index-DrUnlf6K.js +0 -1
  887. package/src/ui/dist/assets/index-NW-h8VzN.js +0 -1
  888. package/src/ui/dist/assets/pdf-effect-queue-J8OnM0jE.js +0 -6
  889. package/src/ui/dist/assets/popover-CLc0pPP8.js +0 -1
  890. package/src/ui/dist/assets/select-Cs2PmzwL.js +0 -11
  891. package/src/ui/dist/assets/sigma-ClKcHAXm.js +0 -6
  892. package/src/ui/dist/assets/trash-DwpbFr3w.js +0 -11
  893. package/src/ui/dist/assets/useCliAccess-NQ8m0Let.js +0 -1
  894. package/src/ui/dist/assets/useFileDiffOverlay-FuhcnKiw.js +0 -1
@@ -0,0 +1,1310 @@
1
+ from __future__ import annotations
2
+
3
+ from collections import Counter
4
+ from datetime import UTC, datetime, timedelta
5
+ from pathlib import Path
6
+ import platform
7
+ import subprocess
8
+ import sys
9
+ from typing import Any
10
+
11
+ from .. import __version__ as DEEPSCIENTIST_VERSION
12
+ from ..diagnostics import diagnose_runner_failure
13
+ from ..doctor import _read_runtime_failure_record
14
+ from ..runtime_tools import RuntimeToolService
15
+ from ..shared import read_json, read_jsonl_tail, read_text, utc_now, utf8_text_subprocess_kwargs, which, write_json
16
+ from .charts import AdminChartService, AdminMetricsCollector
17
+ from .logs import AdminLogService
18
+ from .system_info import AdminSystemMonitor, collect_system_hardware
19
+ from .tool_metrics import AdminToolMetricsService
20
+
21
+
22
+ _FAILURE_LOOKBACK = timedelta(days=7)
23
+
24
+
25
+ class AdminService:
26
+ def __init__(self, app: Any) -> None:
27
+ self.app = app
28
+ self.home = Path(app.home)
29
+ self.log_service = AdminLogService(self.home)
30
+ self.system_monitor = AdminSystemMonitor(self.home)
31
+ self.tool_metrics_service = AdminToolMetricsService(self.home, logger=getattr(app, "logger", None))
32
+ self.chart_service = AdminChartService(app, self)
33
+ self.metrics_collector = AdminMetricsCollector(
34
+ self.home,
35
+ build_fleet_snapshot=self.fleet_snapshot,
36
+ sync_tool_metrics=self.tool_metrics_service.catch_up,
37
+ prune_tool_metrics=lambda current=None: self.tool_metrics_service.prune(current=current),
38
+ logger=getattr(app, "logger", None),
39
+ )
40
+
41
+ def overview(self) -> dict[str, Any]:
42
+ quests = self.app.quest_service.list_quests()
43
+ connectors = self.app.list_connector_statuses()
44
+ quest_insights = self._quest_insights(quests)
45
+ connector_health = self._connector_health_summary(connectors)
46
+ tasks = self.app.admin_task_service.list_tasks(limit=50)
47
+ task_health = self._task_health_summary(tasks)
48
+ active_quests = [
49
+ item
50
+ for item in quests
51
+ if str(item.get("runtime_status") or item.get("status") or "").strip().lower() in {"running", "active"}
52
+ or str(item.get("active_run_id") or "").strip()
53
+ ]
54
+ pending_decisions = sum(int((item.get("counts") or {}).get("pending_decision_count") or 0) for item in quests)
55
+ queued_messages = sum(int((item.get("counts") or {}).get("pending_user_message_count") or 0) for item in quests)
56
+ running_bash = sum(int((item.get("counts") or {}).get("bash_running_count") or 0) for item in quests)
57
+ failures = self.failure_records(limit=200)
58
+ latest_failure = failures[0] if failures else None
59
+ hardware_summary = self.system_hardware(refresh=False)
60
+ return {
61
+ "ok": True,
62
+ "generated_at": utc_now(),
63
+ "daemon": self.app.handlers.health(),
64
+ "cli_health": self.app.handlers.cli_health(),
65
+ "system_update": self.app.admin_task_service.cached_result("system_update.json"),
66
+ "doctor": self.app.admin_task_service.cached_result("doctor.json"),
67
+ "totals": {
68
+ "quests_total": len(quests),
69
+ "quests_active": len(active_quests),
70
+ "pending_decisions_total": pending_decisions,
71
+ "queued_user_messages_total": queued_messages,
72
+ "running_bash_total": running_bash,
73
+ "connectors_total": len(connectors),
74
+ "connectors_enabled": sum(1 for item in connectors if bool(item.get("enabled"))),
75
+ "connectors_degraded": int(connector_health.get("degraded_total") or 0),
76
+ "open_repairs": sum(1 for item in self.app.admin_repair_service.list_repairs(limit=200) if str(item.get("status") or "") == "open"),
77
+ "tasks_running": int(task_health.get("running_total") or 0),
78
+ "tasks_failed": int(task_health.get("failed_total") or 0),
79
+ "runtime_failures_last_7d": len(failures),
80
+ "quests_updated_last_24h": int(
81
+ ((quest_insights.get("recent_activity") or {}) if isinstance(quest_insights.get("recent_activity"), dict) else {}).get("updated_last_24h")
82
+ or 0
83
+ ),
84
+ },
85
+ "latest_failure": latest_failure,
86
+ "latest_failure_scanned": True,
87
+ "system_hardware": hardware_summary,
88
+ "quest_insights": quest_insights,
89
+ "connector_health": connector_health,
90
+ "task_health": task_health,
91
+ "quests": quests[:12],
92
+ "connectors": connectors[:12],
93
+ "tasks": tasks[:12],
94
+ }
95
+
96
+ def fleet_snapshot(self) -> dict[str, Any]:
97
+ quests = self.app.quest_service.list_quests()
98
+ connectors = self.app.list_connector_statuses()
99
+ tasks = self.app.admin_task_service.list_tasks(limit=200)
100
+ quest_insights = self._quest_insights(quests)
101
+ connector_health = self._connector_health_summary(connectors)
102
+ task_health = self._task_health_summary(tasks)
103
+ return {
104
+ "recorded_at": utc_now(),
105
+ "sampling_version": 1,
106
+ "runtime": {
107
+ "quests_total": len(quests),
108
+ "active_quests": sum(
109
+ 1
110
+ for item in quests
111
+ if self._normalize_label(item.get("runtime_status") or item.get("status")) in {"running", "active"}
112
+ or str(item.get("active_run_id") or "").strip()
113
+ ),
114
+ "pending_decisions_total": sum(int((item.get("counts") or {}).get("pending_decision_count") or 0) for item in quests),
115
+ "queued_user_messages_total": sum(int((item.get("counts") or {}).get("pending_user_message_count") or 0) for item in quests),
116
+ "running_bash_total": sum(int((item.get("counts") or {}).get("bash_running_count") or 0) for item in quests),
117
+ "running_tasks": int(task_health.get("running_total") or 0),
118
+ "failed_tasks": int(task_health.get("failed_total") or 0),
119
+ "degraded_connectors": int(connector_health.get("degraded_total") or 0),
120
+ },
121
+ "distributions": {
122
+ "status_counts": quest_insights.get("status_counts") or {},
123
+ "anchor_counts": quest_insights.get("anchor_counts") or {},
124
+ "workspace_mode_counts": quest_insights.get("workspace_mode_counts") or {},
125
+ "runner_counts": quest_insights.get("runner_counts") or {},
126
+ "task_status_counts": task_health.get("status_counts") or {},
127
+ "task_kind_counts": task_health.get("kind_counts") or {},
128
+ "connector_state_counts": connector_health.get("state_counts") or {},
129
+ },
130
+ }
131
+
132
+ def quests(self, *, limit: int = 100) -> dict[str, Any]:
133
+ items = self.app.quest_service.list_quests()[: max(1, limit)]
134
+ return {
135
+ "ok": True,
136
+ "items": items,
137
+ "total": len(self.app.quest_service.list_quests()),
138
+ }
139
+
140
+ def quest_summary(self, quest_id: str) -> dict[str, Any]:
141
+ snapshot = self.app.quest_service.snapshot(quest_id)
142
+ return {
143
+ "ok": True,
144
+ "snapshot": snapshot,
145
+ "workflow_preview": self.app.quest_service.workflow(quest_id),
146
+ "recent_failures": [item for item in self.failure_records(limit=200) if str(item.get("quest_id") or "") == quest_id][:10],
147
+ }
148
+
149
+ def runtime_sessions(self, *, limit: int = 200) -> dict[str, Any]:
150
+ items: list[dict[str, Any]] = []
151
+ per_quest_limit = max(5, min(50, max(1, limit)))
152
+ for quest in self.app.quest_service.list_quests():
153
+ quest_id = str(quest.get("quest_id") or "").strip()
154
+ if not quest_id:
155
+ continue
156
+ quest_root = self.app.quest_service._quest_root(quest_id)
157
+ for session in self.app.bash_exec_service.list_sessions(quest_root, limit=per_quest_limit):
158
+ items.append(
159
+ {
160
+ "quest_id": quest_id,
161
+ "quest_title": quest.get("title"),
162
+ **dict(session),
163
+ }
164
+ )
165
+ items.sort(
166
+ key=lambda item: (
167
+ str(item.get("updated_at") or item.get("started_at") or ""),
168
+ str(item.get("quest_id") or ""),
169
+ ),
170
+ reverse=True,
171
+ )
172
+ return {
173
+ "ok": True,
174
+ "items": items[: max(1, limit)],
175
+ "total": len(items),
176
+ }
177
+
178
+ def log_sources(self) -> dict[str, Any]:
179
+ return {"ok": True, "items": self.log_service.list_sources()}
180
+
181
+ def log_tail(self, source: str, *, line_count: int = 200) -> dict[str, Any]:
182
+ return {"ok": True, **self.log_service.tail(source, line_count=line_count)}
183
+
184
+ @staticmethod
185
+ def _parse_iso(value: object) -> datetime | None:
186
+ normalized = str(value or "").strip()
187
+ if not normalized:
188
+ return None
189
+ candidate = normalized.replace("Z", "+00:00")
190
+ try:
191
+ parsed = datetime.fromisoformat(candidate)
192
+ except ValueError:
193
+ return None
194
+ if parsed.tzinfo is None:
195
+ parsed = parsed.replace(tzinfo=UTC)
196
+ return parsed.astimezone(UTC)
197
+
198
+ @staticmethod
199
+ def _normalize_label(value: object, *, default: str = "unknown") -> str:
200
+ normalized = str(value or "").strip().lower()
201
+ return normalized or default
202
+
203
+ @staticmethod
204
+ def _sorted_counter(counter: Counter[str]) -> dict[str, int]:
205
+ return {key: counter[key] for key, _value in sorted(counter.items(), key=lambda item: (-item[1], item[0]))}
206
+
207
+ def _quest_focus_item(self, quest: dict[str, Any], *, now: datetime | None = None) -> dict[str, Any]:
208
+ current = now or datetime.now(UTC)
209
+ counts = quest.get("counts") if isinstance(quest.get("counts"), dict) else {}
210
+ updated_at = self._parse_iso(quest.get("updated_at"))
211
+ age_hours = round(max((current - updated_at).total_seconds(), 0.0) / 3600.0, 1) if updated_at is not None else None
212
+ return {
213
+ "quest_id": str(quest.get("quest_id") or "").strip(),
214
+ "title": str(quest.get("title") or quest.get("quest_id") or "").strip() or None,
215
+ "runtime_status": str(quest.get("runtime_status") or quest.get("status") or "").strip() or None,
216
+ "active_anchor": str(quest.get("active_anchor") or "").strip() or None,
217
+ "workspace_mode": str(quest.get("workspace_mode") or "").strip() or None,
218
+ "runner": str(quest.get("runner") or "").strip() or None,
219
+ "updated_at": updated_at.isoformat() if updated_at is not None else (str(quest.get("updated_at") or "").strip() or None),
220
+ "pending_decisions": int(counts.get("pending_decision_count") or 0),
221
+ "pending_user_messages": int(counts.get("pending_user_message_count") or 0),
222
+ "running_bash": int(counts.get("bash_running_count") or 0),
223
+ "status_line": str(((quest.get("summary") or {}) if isinstance(quest.get("summary"), dict) else {}).get("status_line") or "").strip() or None,
224
+ "age_hours": age_hours,
225
+ }
226
+
227
+ def _quest_insights(self, quests: list[dict[str, Any]]) -> dict[str, Any]:
228
+ now = datetime.now(UTC)
229
+ status_counts = Counter(self._normalize_label(item.get("runtime_status") or item.get("status")) for item in quests)
230
+ anchor_counts = Counter(self._normalize_label(item.get("active_anchor")) for item in quests)
231
+ workspace_mode_counts = Counter(self._normalize_label(item.get("workspace_mode"), default="quest") for item in quests)
232
+ runner_counts = Counter(self._normalize_label(item.get("runner"), default="codex") for item in quests)
233
+
234
+ decision_backlog_buckets: Counter[str] = Counter()
235
+ message_backlog_buckets: Counter[str] = Counter()
236
+ updated_last_24h = 0
237
+ updated_last_7d = 0
238
+ created_last_7d = 0
239
+ stale_over_7d = 0
240
+ updated_by_day: Counter[str] = Counter()
241
+ created_by_day: Counter[str] = Counter()
242
+
243
+ focus_items = [self._quest_focus_item(item, now=now) for item in quests]
244
+ for quest in quests:
245
+ counts = quest.get("counts") if isinstance(quest.get("counts"), dict) else {}
246
+ pending_decisions = int(counts.get("pending_decision_count") or 0)
247
+ pending_messages = int(counts.get("pending_user_message_count") or 0)
248
+ updated_at = self._parse_iso(quest.get("updated_at"))
249
+ created_at = self._parse_iso(quest.get("created_at"))
250
+
251
+ if pending_decisions <= 0:
252
+ decision_backlog_buckets["none"] += 1
253
+ elif pending_decisions == 1:
254
+ decision_backlog_buckets["one"] += 1
255
+ elif pending_decisions <= 3:
256
+ decision_backlog_buckets["two_to_three"] += 1
257
+ else:
258
+ decision_backlog_buckets["four_plus"] += 1
259
+
260
+ if pending_messages <= 0:
261
+ message_backlog_buckets["none"] += 1
262
+ elif pending_messages == 1:
263
+ message_backlog_buckets["one"] += 1
264
+ elif pending_messages <= 3:
265
+ message_backlog_buckets["two_to_three"] += 1
266
+ else:
267
+ message_backlog_buckets["four_plus"] += 1
268
+
269
+ if updated_at is not None:
270
+ updated_by_day[updated_at.date().isoformat()] += 1
271
+ if updated_at >= now - timedelta(hours=24):
272
+ updated_last_24h += 1
273
+ if updated_at >= now - timedelta(days=7):
274
+ updated_last_7d += 1
275
+ else:
276
+ stale_over_7d += 1
277
+ if created_at is not None:
278
+ created_by_day[created_at.date().isoformat()] += 1
279
+ if created_at >= now - timedelta(days=7):
280
+ created_last_7d += 1
281
+
282
+ def _updated_rank(item: dict[str, Any]) -> str:
283
+ return str(item.get("updated_at") or "")
284
+
285
+ active_watchlist = [
286
+ item
287
+ for item in focus_items
288
+ if self._normalize_label(item.get("runtime_status")) in {"running", "active"}
289
+ or int(item.get("running_bash") or 0) > 0
290
+ ]
291
+ active_watchlist.sort(
292
+ key=lambda item: (
293
+ int(item.get("running_bash") or 0),
294
+ int(item.get("pending_decisions") or 0),
295
+ float(item.get("age_hours") or 0.0),
296
+ _updated_rank(item),
297
+ ),
298
+ reverse=True,
299
+ )
300
+
301
+ top_pending_decisions = [item for item in focus_items if int(item.get("pending_decisions") or 0) > 0]
302
+ top_pending_decisions.sort(
303
+ key=lambda item: (int(item.get("pending_decisions") or 0), int(item.get("pending_user_messages") or 0), _updated_rank(item)),
304
+ reverse=True,
305
+ )
306
+
307
+ top_waiting_messages = [item for item in focus_items if int(item.get("pending_user_messages") or 0) > 0]
308
+ top_waiting_messages.sort(
309
+ key=lambda item: (int(item.get("pending_user_messages") or 0), int(item.get("pending_decisions") or 0), _updated_rank(item)),
310
+ reverse=True,
311
+ )
312
+
313
+ recently_updated = sorted(focus_items, key=_updated_rank, reverse=True)
314
+
315
+ activity_timeline_7d = []
316
+ for offset in range(6, -1, -1):
317
+ day = (now - timedelta(days=offset)).date()
318
+ day_key = day.isoformat()
319
+ activity_timeline_7d.append(
320
+ {
321
+ "date": day_key,
322
+ "label": day.strftime("%m-%d"),
323
+ "quests_created": int(created_by_day.get(day_key) or 0),
324
+ "quests_updated": int(updated_by_day.get(day_key) or 0),
325
+ }
326
+ )
327
+
328
+ return {
329
+ "status_counts": self._sorted_counter(status_counts),
330
+ "anchor_counts": self._sorted_counter(anchor_counts),
331
+ "workspace_mode_counts": self._sorted_counter(workspace_mode_counts),
332
+ "runner_counts": self._sorted_counter(runner_counts),
333
+ "recent_activity": {
334
+ "updated_last_24h": updated_last_24h,
335
+ "updated_last_7d": updated_last_7d,
336
+ "created_last_7d": created_last_7d,
337
+ "stale_over_7d": stale_over_7d,
338
+ },
339
+ "decision_backlog_buckets": {
340
+ "none": int(decision_backlog_buckets.get("none") or 0),
341
+ "one": int(decision_backlog_buckets.get("one") or 0),
342
+ "two_to_three": int(decision_backlog_buckets.get("two_to_three") or 0),
343
+ "four_plus": int(decision_backlog_buckets.get("four_plus") or 0),
344
+ },
345
+ "message_backlog_buckets": {
346
+ "none": int(message_backlog_buckets.get("none") or 0),
347
+ "one": int(message_backlog_buckets.get("one") or 0),
348
+ "two_to_three": int(message_backlog_buckets.get("two_to_three") or 0),
349
+ "four_plus": int(message_backlog_buckets.get("four_plus") or 0),
350
+ },
351
+ "activity_timeline_7d": activity_timeline_7d,
352
+ "top_pending_decisions": top_pending_decisions[:6],
353
+ "top_waiting_messages": top_waiting_messages[:6],
354
+ "recently_updated": recently_updated[:6],
355
+ "active_watchlist": active_watchlist[:6],
356
+ }
357
+
358
+ def _connector_health_summary(self, connectors: list[dict[str, Any]]) -> dict[str, Any]:
359
+ state_counts = Counter()
360
+ degraded_items: list[dict[str, Any]] = []
361
+ for item in connectors:
362
+ state = self._normalize_label(item.get("connection_state"), default="unknown")
363
+ if state == "unknown":
364
+ state = "enabled" if bool(item.get("enabled")) else "disabled"
365
+ state_counts[state] += 1
366
+ if str(item.get("last_error") or "").strip() or state in {"error", "offline", "degraded"}:
367
+ degraded_items.append(
368
+ {
369
+ "name": item.get("name"),
370
+ "connection_state": state,
371
+ "enabled": bool(item.get("enabled")),
372
+ "last_error": item.get("last_error"),
373
+ }
374
+ )
375
+ return {
376
+ "state_counts": self._sorted_counter(state_counts),
377
+ "degraded_total": len(degraded_items),
378
+ "degraded_items": degraded_items[:8],
379
+ }
380
+
381
+ def _task_health_summary(self, tasks: list[dict[str, Any]]) -> dict[str, Any]:
382
+ status_counts = Counter(self._normalize_label(item.get("status")) for item in tasks)
383
+ kind_counts = Counter(self._normalize_label(item.get("kind"), default="unknown") for item in tasks)
384
+ active_items = [
385
+ item
386
+ for item in tasks
387
+ if self._normalize_label(item.get("status")) in {"queued", "running", "active"}
388
+ ]
389
+ failed_items = [
390
+ item
391
+ for item in tasks
392
+ if self._normalize_label(item.get("status")) in {"failed", "error"}
393
+ ]
394
+ return {
395
+ "total": len(tasks),
396
+ "status_counts": self._sorted_counter(status_counts),
397
+ "kind_counts": self._sorted_counter(kind_counts),
398
+ "queued_total": sum(1 for item in tasks if self._normalize_label(item.get("status")) == "queued"),
399
+ "running_total": sum(1 for item in tasks if self._normalize_label(item.get("status")) in {"running", "active"}),
400
+ "failed_total": len(failed_items),
401
+ "active_items": active_items[:8],
402
+ "failed_items": failed_items[:8],
403
+ }
404
+
405
+ def _latest_failure_candidate(self, *, cutoff: datetime) -> dict[str, Any] | None:
406
+ quests_root = self.home / "quests"
407
+ if not quests_root.exists():
408
+ return None
409
+ event_files: list[tuple[float, Path, Path]] = []
410
+ for quest_root in quests_root.glob("*/"):
411
+ event_path = quest_root / ".ds" / "events.jsonl"
412
+ if not event_path.exists():
413
+ continue
414
+ try:
415
+ mtime = event_path.stat().st_mtime
416
+ except OSError:
417
+ continue
418
+ event_files.append((mtime, quest_root, event_path))
419
+
420
+ best: dict[str, Any] | None = None
421
+ for mtime, quest_root, event_path in sorted(event_files, key=lambda item: item[0], reverse=True):
422
+ if best is not None and mtime <= float(best.get("_created_at_ts") or 0.0):
423
+ break
424
+ for event in reversed(read_jsonl_tail(event_path, 120)):
425
+ event_type = str(event.get("type") or "").strip()
426
+ if event_type not in {"runner.turn_error", "runner.turn_retry_exhausted", "quest.runtime_auto_resume_suppressed"}:
427
+ continue
428
+ created_at = self._parse_iso(event.get("created_at"))
429
+ if created_at is None or created_at < cutoff:
430
+ continue
431
+ candidate = {
432
+ "quest_id": quest_root.name,
433
+ "event_type": event_type,
434
+ "run_id": str(event.get("run_id") or "").strip() or None,
435
+ "created_at": created_at.isoformat(),
436
+ "summary": str(event.get("summary") or "").strip(),
437
+ "_quest_root": str(quest_root),
438
+ "_created_at_ts": created_at.timestamp(),
439
+ }
440
+ if best is None or float(candidate["_created_at_ts"]) > float(best.get("_created_at_ts") or 0.0):
441
+ best = candidate
442
+ break
443
+ return best
444
+
445
+ def failure_records(self, *, limit: int = 100) -> list[dict[str, Any]]:
446
+ cutoff = datetime.now(UTC) - _FAILURE_LOOKBACK
447
+ candidates: list[dict[str, Any]] = []
448
+ quests_root = self.home / "quests"
449
+ if not quests_root.exists():
450
+ return candidates
451
+
452
+ if limit == 1:
453
+ latest_candidate = self._latest_failure_candidate(cutoff=cutoff)
454
+ if latest_candidate is not None:
455
+ candidates = [latest_candidate]
456
+ else:
457
+ tail_limit = 120 if limit <= 20 else 400
458
+ for quest_root in sorted(quests_root.glob("*/")):
459
+ quest_id = quest_root.name
460
+ for event in reversed(read_jsonl_tail(quest_root / ".ds" / "events.jsonl", tail_limit)):
461
+ event_type = str(event.get("type") or "").strip()
462
+ if event_type not in {"runner.turn_error", "runner.turn_retry_exhausted", "quest.runtime_auto_resume_suppressed"}:
463
+ continue
464
+ created_at = self._parse_iso(event.get("created_at"))
465
+ if created_at is None or created_at < cutoff:
466
+ continue
467
+ candidates.append(
468
+ {
469
+ "quest_id": quest_id,
470
+ "event_type": event_type,
471
+ "run_id": str(event.get("run_id") or "").strip() or None,
472
+ "created_at": created_at.isoformat(),
473
+ "summary": str(event.get("summary") or "").strip(),
474
+ "_quest_root": str(quest_root),
475
+ }
476
+ )
477
+ break
478
+ candidates.sort(key=lambda item: str(item.get("created_at") or ""), reverse=True)
479
+ latest = _read_runtime_failure_record(self.home)
480
+ if latest and all(
481
+ str(item.get("quest_id") or "") != str(latest.get("quest_id") or "")
482
+ or str(item.get("run_id") or "") != str(latest.get("run_id") or "")
483
+ for item in candidates
484
+ ):
485
+ candidates.insert(0, latest)
486
+
487
+ selected = candidates[: max(1, limit)]
488
+ items: list[dict[str, Any]] = []
489
+ for item in selected:
490
+ if "diagnosis" in item:
491
+ items.append(item)
492
+ continue
493
+ quest_root_value = str(item.get("_quest_root") or "").strip()
494
+ quest_root = Path(quest_root_value) if quest_root_value else None
495
+ run_id = str(item.get("run_id") or "").strip() or None
496
+ stderr_text = ""
497
+ output_text = ""
498
+ if quest_root and run_id:
499
+ run_root = quest_root / ".ds" / "runs" / run_id
500
+ result_payload = read_json(run_root / "result.json", {})
501
+ if isinstance(result_payload, dict):
502
+ stderr_text = str(result_payload.get("stderr_text") or "").strip()
503
+ output_text = str(result_payload.get("output_text") or "").strip()
504
+ if not stderr_text:
505
+ stderr_text = read_text(run_root / "stderr.txt", "")
506
+ diagnosis = diagnose_runner_failure(
507
+ runner_name="codex",
508
+ summary=str(item.get("summary") or ""),
509
+ stderr_text=stderr_text,
510
+ output_text=output_text,
511
+ )
512
+ items.append(
513
+ {
514
+ "quest_id": item.get("quest_id"),
515
+ "event_type": item.get("event_type"),
516
+ "run_id": run_id,
517
+ "created_at": item.get("created_at"),
518
+ "summary": item.get("summary"),
519
+ "diagnosis": diagnosis.__dict__ if diagnosis is not None else None,
520
+ "stderr_excerpt": stderr_text[:800] if stderr_text else None,
521
+ "output_excerpt": output_text[:800] if output_text else None,
522
+ }
523
+ )
524
+ return items
525
+
526
+ def failures(self, *, limit: int = 100) -> dict[str, Any]:
527
+ return {"ok": True, "items": self.failure_records(limit=limit)}
528
+
529
+ def error_console(self, *, limit: int = 100) -> dict[str, Any]:
530
+ connectors = self.app.list_connector_statuses()
531
+ degraded_connectors = [
532
+ {
533
+ "name": item.get("name"),
534
+ "connection_state": item.get("connection_state"),
535
+ "last_error": item.get("last_error"),
536
+ }
537
+ for item in connectors
538
+ if str(item.get("last_error") or "").strip()
539
+ or str(item.get("connection_state") or "").strip().lower() in {"error", "offline", "degraded"}
540
+ ]
541
+ daemon_errors = [
542
+ item
543
+ for item in reversed(read_jsonl_tail(self.home / "logs" / "daemon.jsonl", max(50, limit * 2)))
544
+ if str(item.get("level") or "").strip().lower() in {"error", "warning"}
545
+ ][: max(1, limit)]
546
+ failed_tasks = [
547
+ item
548
+ for item in self.app.admin_task_service.list_tasks(limit=max(50, limit * 2))
549
+ if str(item.get("status") or "").strip().lower() == "failed"
550
+ ][: max(1, limit)]
551
+ failures = self.failure_records(limit=limit)
552
+ return {
553
+ "ok": True,
554
+ "generated_at": utc_now(),
555
+ "totals": {
556
+ "degraded_connectors": len(degraded_connectors),
557
+ "runtime_failures": len(failures),
558
+ "daemon_errors": len(daemon_errors),
559
+ "failed_tasks": len(failed_tasks),
560
+ },
561
+ "degraded_connectors": degraded_connectors,
562
+ "runtime_failures": failures,
563
+ "daemon_errors": daemon_errors,
564
+ "failed_tasks": failed_tasks,
565
+ }
566
+
567
+ def runtime_tools(self) -> dict[str, Any]:
568
+ return {"ok": True, "items": RuntimeToolService(self.home).all_statuses()}
569
+
570
+ @property
571
+ def system_hardware_cache_path(self) -> Path:
572
+ return self.home / "runtime" / "admin" / "cache" / "system_hardware.json"
573
+
574
+ def _hardware_preferences(self) -> dict[str, Any]:
575
+ config = self.app.config_manager.load_runtime_config()
576
+ hardware = config.get("hardware") if isinstance(config.get("hardware"), dict) else {}
577
+ selection_mode = str(hardware.get("gpu_selection_mode") or "all").strip().lower() or "all"
578
+ if selection_mode not in {"all", "selected"}:
579
+ selection_mode = "all"
580
+ selected_gpu_ids = []
581
+ for item in hardware.get("selected_gpu_ids") or []:
582
+ normalized = str(item or "").strip()
583
+ if normalized and normalized not in selected_gpu_ids:
584
+ selected_gpu_ids.append(normalized)
585
+ return {
586
+ "gpu_selection_mode": selection_mode,
587
+ "selected_gpu_ids": selected_gpu_ids,
588
+ "include_system_hardware_in_prompt": bool(hardware.get("include_system_hardware_in_prompt", True)),
589
+ }
590
+
591
+ def _memory_preferences(self) -> dict[str, Any]:
592
+ config = self.app.config_manager.load_runtime_config()
593
+ memory = config.get("memory") if isinstance(config.get("memory"), dict) else {}
594
+ return {
595
+ "read_visibility_mode": self.app.memory_service.normalize_read_visibility_mode(
596
+ memory.get("read_visibility_mode")
597
+ ),
598
+ }
599
+
600
+ @staticmethod
601
+ def _effective_gpu_ids(*, gpus: list[dict[str, Any]], selection_mode: str, selected_gpu_ids: list[str]) -> list[str]:
602
+ available_ids = [str(item.get("gpu_id") or "").strip() for item in gpus if str(item.get("gpu_id") or "").strip()]
603
+ if selection_mode != "selected":
604
+ return available_ids
605
+ return [item for item in selected_gpu_ids if item in available_ids]
606
+
607
+ @staticmethod
608
+ def _prompt_hardware_summary(*, system_payload: dict[str, Any], preferences: dict[str, Any]) -> str:
609
+ cpu = system_payload.get("cpu") if isinstance(system_payload.get("cpu"), dict) else {}
610
+ memory = system_payload.get("memory") if isinstance(system_payload.get("memory"), dict) else {}
611
+ disks = system_payload.get("disks") if isinstance(system_payload.get("disks"), list) else []
612
+ gpus = system_payload.get("gpus") if isinstance(system_payload.get("gpus"), list) else []
613
+ effective_gpu_ids = preferences.get("effective_gpu_ids") if isinstance(preferences.get("effective_gpu_ids"), list) else []
614
+ cpu_text = str(cpu.get("model") or "unknown cpu").strip()
615
+ core_text = str(cpu.get("logical_cores") or "unknown").strip()
616
+ memory_text = str(memory.get("total_gb") or "unknown").strip()
617
+ disk_text = "unknown"
618
+ if disks and isinstance(disks[0], dict):
619
+ disk_text = f"{disks[0].get('free_gb') or 'unknown'}GB free on {disks[0].get('mount') or '/'}"
620
+ if gpus:
621
+ gpu_parts = []
622
+ for item in gpus[:8]:
623
+ gpu_id = str(item.get("gpu_id") or "").strip()
624
+ name = str(item.get("name") or "GPU").strip()
625
+ memory_total = item.get("memory_total_gb")
626
+ gpu_parts.append(f"{gpu_id}:{name}{f' {memory_total}GB' if memory_total is not None else ''}")
627
+ gpu_summary = "; ".join(gpu_parts)
628
+ else:
629
+ gpu_summary = "no GPU detected"
630
+ selected = ",".join(effective_gpu_ids) if effective_gpu_ids else ("none" if preferences.get("gpu_selection_mode") == "selected" else "all")
631
+ return f"CPU: {cpu_text} ({core_text} logical cores) | Memory: {memory_text}GB | Disk: {disk_text} | GPUs: {gpu_summary} | Selected GPUs: {selected}"
632
+
633
+ def system_hardware(self, *, refresh: bool = True) -> dict[str, Any]:
634
+ system_payload = collect_system_hardware(self.home)
635
+ system_payload["generated_at"] = utc_now()
636
+ preferences = self._hardware_preferences()
637
+ memory_preferences = self._memory_preferences()
638
+ gpus = [dict(item) for item in (system_payload.get("gpus") or []) if isinstance(item, dict)]
639
+ effective_gpu_ids = self._effective_gpu_ids(
640
+ gpus=gpus,
641
+ selection_mode=str(preferences.get("gpu_selection_mode") or "all"),
642
+ selected_gpu_ids=list(preferences.get("selected_gpu_ids") or []),
643
+ )
644
+ available_gpu_ids = [str(item.get("gpu_id") or "").strip() for item in gpus if str(item.get("gpu_id") or "").strip()]
645
+ preferences = {
646
+ **preferences,
647
+ "available_gpu_ids": available_gpu_ids,
648
+ "available_gpu_count": len(available_gpu_ids),
649
+ "effective_gpu_ids": effective_gpu_ids,
650
+ "cuda_visible_devices": ",".join(effective_gpu_ids) if effective_gpu_ids else None,
651
+ }
652
+ prompt_summary = self._prompt_hardware_summary(system_payload=system_payload, preferences=preferences)
653
+ for item in gpus:
654
+ item["selected"] = str(item.get("gpu_id") or "").strip() in effective_gpu_ids
655
+ if refresh:
656
+ latest_sample = self.system_monitor.sample_now(persist=True)
657
+ recent_stats = self.system_monitor.latest_summary(window_minutes=60)
658
+ else:
659
+ recent_stats = self.system_monitor.latest_summary(window_minutes=60)
660
+ latest_sample = recent_stats.get("latest_sample")
661
+ if latest_sample is None:
662
+ latest_sample = self.system_monitor.sample_now(persist=True)
663
+ recent_stats = self.system_monitor.latest_summary(window_minutes=60)
664
+ payload = {
665
+ "ok": True,
666
+ "generated_at": system_payload.get("generated_at"),
667
+ "system": system_payload,
668
+ "preferences": preferences,
669
+ "memory_preferences": memory_preferences,
670
+ "prompt_hardware_summary": prompt_summary,
671
+ "latest_sample": latest_sample,
672
+ "recent_stats": recent_stats,
673
+ }
674
+ write_json(self.system_hardware_cache_path, payload)
675
+ return payload
676
+
677
+ def update_system_hardware_preferences(
678
+ self,
679
+ *,
680
+ gpu_selection_mode: str | None = None,
681
+ selected_gpu_ids: list[str] | None = None,
682
+ include_system_hardware_in_prompt: bool | None = None,
683
+ memory_read_visibility_mode: str | None = None,
684
+ ) -> dict[str, Any]:
685
+ config = self.app.config_manager.load_runtime_config()
686
+ hardware = config.get("hardware") if isinstance(config.get("hardware"), dict) else {}
687
+ memory = config.get("memory") if isinstance(config.get("memory"), dict) else {}
688
+ if gpu_selection_mode is not None:
689
+ normalized_mode = str(gpu_selection_mode or "all").strip().lower() or "all"
690
+ hardware["gpu_selection_mode"] = normalized_mode if normalized_mode in {"all", "selected"} else "all"
691
+ if selected_gpu_ids is not None:
692
+ deduped: list[str] = []
693
+ for item in selected_gpu_ids:
694
+ normalized = str(item or "").strip()
695
+ if normalized and normalized not in deduped:
696
+ deduped.append(normalized)
697
+ hardware["selected_gpu_ids"] = deduped
698
+ if include_system_hardware_in_prompt is not None:
699
+ hardware["include_system_hardware_in_prompt"] = bool(include_system_hardware_in_prompt)
700
+ if memory_read_visibility_mode is not None:
701
+ memory["read_visibility_mode"] = self.app.memory_service.normalize_read_visibility_mode(
702
+ memory_read_visibility_mode
703
+ )
704
+ config["hardware"] = hardware
705
+ config["memory"] = memory
706
+ save_result = self.app.config_manager.save_named_payload("config", config)
707
+ runtime_reload = self.app.reload_runtime_config()
708
+ payload = self.system_hardware()
709
+ payload["save_result"] = save_result
710
+ payload["runtime_reload"] = runtime_reload
711
+ return payload
712
+
713
+ def audit(self, *, limit: int = 200) -> dict[str, Any]:
714
+ audit_path = self.home / "logs" / "admin" / "audit.jsonl"
715
+ return {
716
+ "ok": True,
717
+ "items": list(reversed(read_jsonl_tail(audit_path, max(1, limit)))),
718
+ }
719
+
720
+ def write_audit(self, *, action: str, source: str = "admin-ui", **payload: Any) -> dict[str, Any]:
721
+ from ..shared import append_jsonl, ensure_dir
722
+
723
+ record = {
724
+ "action": str(action or "").strip(),
725
+ "source": str(source or "").strip() or "admin-ui",
726
+ "created_at": utc_now(),
727
+ "payload": payload,
728
+ }
729
+ audit_path = ensure_dir(self.home / "logs" / "admin") / "audit.jsonl"
730
+ append_jsonl(audit_path, record)
731
+ return record
732
+
733
+ def stats_summary(self) -> dict[str, Any]:
734
+ quests = self.app.quest_service.list_quests()
735
+ connectors = self.app.list_connector_statuses()
736
+ connector_health = self._connector_health_summary(connectors)
737
+ tasks = self.app.admin_task_service.list_tasks(limit=200)
738
+ task_health = self._task_health_summary(tasks)
739
+ quest_insights = self._quest_insights(quests)
740
+ failures = self.failure_records(limit=500)
741
+ failure_type_counts = Counter(self._normalize_label(item.get("event_type")) for item in failures)
742
+ return {
743
+ "ok": True,
744
+ "generated_at": utc_now(),
745
+ "totals": {
746
+ "quests": len(quests),
747
+ "active_quests": sum(
748
+ 1
749
+ for item in quests
750
+ if self._normalize_label(item.get("runtime_status") or item.get("status")) in {"running", "active"}
751
+ or str(item.get("active_run_id") or "").strip()
752
+ ),
753
+ "pending_decisions_total": sum(int((item.get("counts") or {}).get("pending_decision_count") or 0) for item in quests),
754
+ "queued_user_messages_total": sum(int((item.get("counts") or {}).get("pending_user_message_count") or 0) for item in quests),
755
+ "running_bash_total": sum(int((item.get("counts") or {}).get("bash_running_count") or 0) for item in quests),
756
+ "failures_last_7d": len(failures),
757
+ "repairs_total": len(self.app.admin_repair_service.list_repairs(limit=500)),
758
+ "degraded_connectors": int(connector_health.get("degraded_total") or 0),
759
+ "running_tasks": int(task_health.get("running_total") or 0),
760
+ "failed_tasks": int(task_health.get("failed_total") or 0),
761
+ },
762
+ "status_counts": quest_insights.get("status_counts"),
763
+ "anchor_counts": quest_insights.get("anchor_counts"),
764
+ "workspace_mode_counts": quest_insights.get("workspace_mode_counts"),
765
+ "runner_counts": quest_insights.get("runner_counts"),
766
+ "connector_state_counts": connector_health.get("state_counts"),
767
+ "task_status_counts": task_health.get("status_counts"),
768
+ "task_kind_counts": task_health.get("kind_counts"),
769
+ "failure_type_counts": self._sorted_counter(failure_type_counts),
770
+ "decision_backlog_buckets": quest_insights.get("decision_backlog_buckets"),
771
+ "message_backlog_buckets": quest_insights.get("message_backlog_buckets"),
772
+ "recent_activity": quest_insights.get("recent_activity"),
773
+ "activity_timeline_7d": quest_insights.get("activity_timeline_7d"),
774
+ "top_pending_decisions": quest_insights.get("top_pending_decisions"),
775
+ "top_waiting_messages": quest_insights.get("top_waiting_messages"),
776
+ "recently_updated": quest_insights.get("recently_updated"),
777
+ "active_watchlist": quest_insights.get("active_watchlist"),
778
+ }
779
+
780
+ def chart_catalog(self) -> dict[str, Any]:
781
+ return self.chart_service.catalog()
782
+
783
+ def chart_query(self, items: list[dict[str, Any]]) -> dict[str, Any]:
784
+ return self.chart_service.query(items)
785
+
786
+ def search(self, query: str, *, limit: int = 100) -> dict[str, Any]:
787
+ term = str(query or "").strip().lower()
788
+ if not term:
789
+ return {"ok": True, "items": []}
790
+ items: list[dict[str, Any]] = []
791
+ for quest in self.app.quest_service.list_quests():
792
+ quest_id = str(quest.get("quest_id") or "").strip()
793
+ title = str(quest.get("title") or "").strip()
794
+ summary_line = str(((quest.get("summary") or {}) if isinstance(quest.get("summary"), dict) else {}).get("status_line") or "").strip()
795
+ haystacks = [quest_id, title, summary_line]
796
+ if any(term in value.lower() for value in haystacks if value):
797
+ items.append(
798
+ {
799
+ "kind": "quest",
800
+ "quest_id": quest_id,
801
+ "title": title,
802
+ "summary": summary_line,
803
+ }
804
+ )
805
+ quest_root = self.home / "quests" / quest_id
806
+ for event in reversed(read_jsonl_tail(quest_root / ".ds" / "events.jsonl", 120)):
807
+ summary = str(event.get("summary") or event.get("message") or "").strip()
808
+ if summary and term in summary.lower():
809
+ items.append(
810
+ {
811
+ "kind": "event",
812
+ "quest_id": quest_id,
813
+ "event_type": str(event.get("type") or ""),
814
+ "summary": summary,
815
+ "created_at": event.get("created_at"),
816
+ }
817
+ )
818
+ break
819
+ return {"ok": True, "items": items[: max(1, limit)]}
820
+
821
+ @staticmethod
822
+ def _origin_repo_url(repo_root: Path) -> str:
823
+ fallback = "https://github.com/ResearAI/DeepScientist"
824
+ try:
825
+ result = subprocess.run(
826
+ ["git", "remote", "get-url", "origin"],
827
+ cwd=str(repo_root),
828
+ capture_output=True,
829
+ timeout=2,
830
+ check=False,
831
+ **utf8_text_subprocess_kwargs(),
832
+ )
833
+ except Exception:
834
+ return fallback
835
+ if result.returncode != 0:
836
+ return fallback
837
+ raw = (result.stdout or result.stderr or "").strip()
838
+ if raw.startswith("git@github.com:"):
839
+ path = raw[len("git@github.com:") :]
840
+ if path.endswith(".git"):
841
+ path = path[:-4]
842
+ return f"https://github.com/{path}"
843
+ if raw.startswith("https://github.com/") or raw.startswith("http://github.com/"):
844
+ return raw[:-4] if raw.endswith(".git") else raw
845
+ return fallback
846
+
847
+ @staticmethod
848
+ def _node_version() -> str | None:
849
+ node = which("node")
850
+ if not node:
851
+ return None
852
+ try:
853
+ result = subprocess.run(
854
+ [node, "--version"],
855
+ capture_output=True,
856
+ timeout=2,
857
+ check=False,
858
+ **utf8_text_subprocess_kwargs(),
859
+ )
860
+ except Exception:
861
+ return None
862
+ if result.returncode != 0:
863
+ return None
864
+ return (result.stdout or result.stderr or "").strip() or None
865
+
866
+ @staticmethod
867
+ def _dedupe_issue_lines(lines: list[str], *, limit: int = 10) -> list[str]:
868
+ seen: set[str] = set()
869
+ deduped: list[str] = []
870
+ for item in lines:
871
+ text = str(item or "").strip()
872
+ if not text:
873
+ continue
874
+ normalized = text.casefold()
875
+ if normalized in seen:
876
+ continue
877
+ seen.add(normalized)
878
+ deduped.append(text)
879
+ if len(deduped) >= max(1, limit):
880
+ break
881
+ return deduped
882
+
883
+ @classmethod
884
+ def _recommended_issue_actions(
885
+ cls,
886
+ *,
887
+ degraded_connectors: list[dict[str, Any]],
888
+ runtime_failures: list[dict[str, Any]],
889
+ doctor_cache: dict[str, Any] | None,
890
+ ) -> list[str]:
891
+ suggestions: list[str] = []
892
+
893
+ for item in runtime_failures[:4]:
894
+ diagnosis = item.get("diagnosis") if isinstance(item.get("diagnosis"), dict) else {}
895
+ for guidance in diagnosis.get("guidance") or []:
896
+ text = str(guidance or "").strip()
897
+ if text:
898
+ suggestions.append(text)
899
+
900
+ for item in degraded_connectors[:4]:
901
+ name = str(item.get("name") or "connector").strip() or "connector"
902
+ state = str(item.get("connection_state") or "degraded").strip() or "degraded"
903
+ last_error = str(item.get("last_error") or "").strip()
904
+ if last_error:
905
+ suggestions.append(
906
+ f"Reconnect or reconfigure `{name}`. Current state is `{state}` and the latest reported error is `{last_error[:180]}`."
907
+ )
908
+ else:
909
+ suggestions.append(f"Reconnect or reconfigure `{name}` because its current state is `{state}`.")
910
+
911
+ report = doctor_cache.get("report") if isinstance(doctor_cache, dict) and isinstance(doctor_cache.get("report"), dict) else None
912
+ if isinstance(report, dict):
913
+ for item in (report.get("checks") or [])[:8]:
914
+ if not isinstance(item, dict):
915
+ continue
916
+ status = str(item.get("status") or "").strip().lower()
917
+ if status in {"ok", "pass", "healthy", "success"}:
918
+ continue
919
+ check_id = str(item.get("id") or "doctor-check").strip() or "doctor-check"
920
+ summary = str(item.get("summary") or "").strip()
921
+ if summary:
922
+ suggestions.append(f"Review doctor check `{check_id}`: {summary}")
923
+
924
+ if not suggestions:
925
+ suggestions.append("Review the attached runtime evidence and compare it with the expected behavior before retrying.")
926
+ suggestions.append("If this is reproducible, add exact reproduction steps and expected/actual behavior before submitting.")
927
+
928
+ return cls._dedupe_issue_lines(suggestions, limit=10)
929
+
930
+ @staticmethod
931
+ def _hardware_issue_snapshot(hardware_payload: dict[str, Any] | None) -> list[str]:
932
+ if not isinstance(hardware_payload, dict):
933
+ return ["- Hardware summary unavailable."]
934
+ prompt_summary = str(hardware_payload.get("prompt_hardware_summary") or "").strip()
935
+ system_payload = hardware_payload.get("system") if isinstance(hardware_payload.get("system"), dict) else {}
936
+ host = system_payload.get("host") if isinstance(system_payload.get("host"), dict) else {}
937
+ cpu = system_payload.get("cpu") if isinstance(system_payload.get("cpu"), dict) else {}
938
+ memory = system_payload.get("memory") if isinstance(system_payload.get("memory"), dict) else {}
939
+ gpus = system_payload.get("gpus") if isinstance(system_payload.get("gpus"), list) else []
940
+ lines: list[str] = []
941
+ hostname = str(host.get("hostname") or "").strip()
942
+ platform_text = str(host.get("platform") or "").strip()
943
+ if hostname or platform_text:
944
+ lines.append(f"- Host: `{hostname or 'unknown'}` platform=`{platform_text or 'unknown'}`")
945
+ if prompt_summary:
946
+ lines.append(f"- Summary: {prompt_summary}")
947
+ cpu_model = str(cpu.get("model") or "").strip()
948
+ logical_cores = cpu.get("logical_cores")
949
+ if cpu_model or logical_cores is not None:
950
+ lines.append(f"- CPU: `{cpu_model or 'unknown cpu'}` logical_cores=`{logical_cores if logical_cores is not None else 'unknown'}`")
951
+ total_memory_gb = memory.get("total_gb")
952
+ available_memory_gb = memory.get("available_gb")
953
+ if total_memory_gb is not None or available_memory_gb is not None:
954
+ lines.append(f"- Memory: total_gb=`{total_memory_gb if total_memory_gb is not None else 'unknown'}` available_gb=`{available_memory_gb if available_memory_gb is not None else 'unknown'}`")
955
+ if gpus:
956
+ gpu_descriptions = []
957
+ for item in gpus[:6]:
958
+ if not isinstance(item, dict):
959
+ continue
960
+ gpu_id = str(item.get("gpu_id") or "").strip()
961
+ name = str(item.get("name") or "GPU").strip()
962
+ memory_total_gb = item.get("memory_total_gb")
963
+ gpu_descriptions.append(f"{gpu_id or '?'}:{name}{f' {memory_total_gb}GB' if memory_total_gb is not None else ''}")
964
+ if gpu_descriptions:
965
+ lines.append(f"- GPUs: {'; '.join(gpu_descriptions)}")
966
+ if not lines:
967
+ return ["- Hardware summary unavailable."]
968
+ return lines
969
+
970
+ def _system_quirks_issue_snapshot(self) -> list[str]:
971
+ path = self.home / "system_quirks.md"
972
+ if not path.exists():
973
+ return ["_No system quirks file exists yet._"]
974
+ try:
975
+ content = path.read_text(encoding="utf-8")
976
+ except OSError as exc:
977
+ return [f"_Unable to read system quirks: {exc}_"]
978
+
979
+ lines = [line.rstrip() for line in content.splitlines()]
980
+ has_entry = any(
981
+ line.lstrip().startswith("## ")
982
+ or line.lstrip().startswith("- Expected behavior:")
983
+ or line.lstrip().startswith("- Actual behavior:")
984
+ for line in lines
985
+ )
986
+ if not has_entry:
987
+ return ["_No system quirks have been recorded yet._"]
988
+ if len(lines) > 200:
989
+ lines = lines[:200] + ["", "_Truncated to the first 200 lines. Review and redact before submitting._"]
990
+ return lines
991
+
992
+ def issue_draft(
993
+ self,
994
+ *,
995
+ summary: str | None = None,
996
+ user_notes: str | None = None,
997
+ include_doctor: bool = True,
998
+ include_logs: bool = True,
999
+ include_system_settings: bool = True,
1000
+ include_system_quirks: bool = False,
1001
+ ) -> dict[str, Any]:
1002
+ error_console = self.error_console(limit=10)
1003
+ doctor_cache = self.app.admin_task_service.cached_result("doctor.json") if include_doctor else None
1004
+ health = self.app.handlers.health()
1005
+ cli_health = self.app.handlers.cli_health()
1006
+ repo_url = self._origin_repo_url(self.app.repo_root)
1007
+ issue_url_base = f"{repo_url}/issues/new"
1008
+ degraded_connectors = error_console.get("degraded_connectors") or []
1009
+ runtime_failures = error_console.get("runtime_failures") or []
1010
+ daemon_errors = error_console.get("daemon_errors") or []
1011
+ failed_tasks = error_console.get("failed_tasks") or []
1012
+ hardware = self.system_hardware(refresh=False) if include_system_settings else None
1013
+ title = str(summary or "").strip()
1014
+ if not title:
1015
+ if runtime_failures:
1016
+ title = f"Admin report: {str((runtime_failures[0] or {}).get('summary') or 'runtime failure').strip()[:90]}"
1017
+ elif degraded_connectors:
1018
+ title = f"Admin report: connector degradation in {str((degraded_connectors[0] or {}).get('name') or 'connector')}"
1019
+ elif failed_tasks:
1020
+ title = f"Admin report: failed admin task `{str((failed_tasks[0] or {}).get('kind') or 'task')}`"
1021
+ else:
1022
+ title = "Admin report: runtime issue investigation"
1023
+
1024
+ recommended_actions = self._recommended_issue_actions(
1025
+ degraded_connectors=degraded_connectors,
1026
+ runtime_failures=runtime_failures,
1027
+ doctor_cache=doctor_cache if isinstance(doctor_cache, dict) else None,
1028
+ )
1029
+
1030
+ lines = [
1031
+ "# Summary",
1032
+ "",
1033
+ title,
1034
+ "",
1035
+ "## Operator Notes",
1036
+ "",
1037
+ str(user_notes or "").strip() or "_Add any extra observations here before submitting._",
1038
+ "",
1039
+ "## Detected Problems",
1040
+ "",
1041
+ ]
1042
+ if runtime_failures:
1043
+ for item in runtime_failures[:5]:
1044
+ diagnosis = item.get("diagnosis") if isinstance(item.get("diagnosis"), dict) else {}
1045
+ diagnosis_problem = str(diagnosis.get("problem") or "").strip()
1046
+ diagnosis_code = str(diagnosis.get("code") or "").strip()
1047
+ diagnosis_suffix = f" diagnosis=`{diagnosis_code}` {diagnosis_problem}" if diagnosis_problem or diagnosis_code else ""
1048
+ lines.append(
1049
+ f"- Runtime failure: quest=`{item.get('quest_id')}` run=`{item.get('run_id')}` type=`{item.get('event_type')}` summary={item.get('summary')}{diagnosis_suffix}"
1050
+ )
1051
+ else:
1052
+ lines.append("- No runtime failure records were detected in the recent window.")
1053
+ if degraded_connectors:
1054
+ for item in degraded_connectors[:5]:
1055
+ lines.append(
1056
+ f"- Connector issue: `{item.get('name')}` state=`{item.get('connection_state')}` error=`{item.get('last_error')}`"
1057
+ )
1058
+ else:
1059
+ lines.append("- No degraded connectors were detected.")
1060
+ if failed_tasks:
1061
+ for item in failed_tasks[:5]:
1062
+ lines.append(
1063
+ f"- Failed admin task: kind=`{item.get('kind')}` task_id=`{item.get('task_id')}` error=`{item.get('error')}`"
1064
+ )
1065
+ else:
1066
+ lines.append("- No failed admin tasks were detected.")
1067
+ lines.extend(
1068
+ [
1069
+ "",
1070
+ "## Recommended Fixes / Workarounds",
1071
+ "",
1072
+ ]
1073
+ )
1074
+ for item in recommended_actions:
1075
+ lines.append(f"- {item}")
1076
+ if include_system_settings:
1077
+ lines.extend(
1078
+ [
1079
+ "",
1080
+ "## Environment",
1081
+ "",
1082
+ ]
1083
+ )
1084
+ lines.extend(self._hardware_issue_snapshot(hardware))
1085
+ lines.extend(
1086
+ [
1087
+ "",
1088
+ f"- DeepScientist version: `{DEEPSCIENTIST_VERSION}`",
1089
+ f"- Platform: `{platform.platform()}`",
1090
+ f"- Python: `{sys.version.split()[0]}`",
1091
+ f"- Node: `{self._node_version() or 'unavailable'}`",
1092
+ f"- Repo: `{repo_url}`",
1093
+ f"- Home: `{self.home}`",
1094
+ ]
1095
+ )
1096
+ lines.extend(
1097
+ [
1098
+ "",
1099
+ "## Runtime Health",
1100
+ "",
1101
+ f"- Daemon status: `{health.get('status')}`",
1102
+ f"- Daemon id: `{health.get('daemon_id')}`",
1103
+ f"- Browser auth enabled: `{health.get('auth_enabled')}`",
1104
+ f"- CLI checks: `{cli_health.get('checks')}`",
1105
+ ]
1106
+ )
1107
+ if include_system_quirks:
1108
+ lines.extend(
1109
+ [
1110
+ "",
1111
+ "## System Quirks",
1112
+ "",
1113
+ ]
1114
+ )
1115
+ lines.extend(self._system_quirks_issue_snapshot())
1116
+ lines.extend(
1117
+ [
1118
+ "",
1119
+ "_Review and redact this section before submitting a public issue._",
1120
+ ]
1121
+ )
1122
+ lines.extend(["", "## Recent Runtime Failures", ""])
1123
+ if runtime_failures:
1124
+ for item in runtime_failures[:5]:
1125
+ lines.append(
1126
+ f"- quest=`{item.get('quest_id')}` run=`{item.get('run_id')}` type=`{item.get('event_type')}` summary={item.get('summary')}"
1127
+ )
1128
+ else:
1129
+ lines.append("- None detected.")
1130
+ lines.extend(["", "## Failed Admin Tasks", ""])
1131
+ if failed_tasks:
1132
+ for item in failed_tasks[:5]:
1133
+ lines.append(
1134
+ f"- kind=`{item.get('kind')}` task_id=`{item.get('task_id')}` error=`{item.get('error')}`"
1135
+ )
1136
+ else:
1137
+ lines.append("- None detected.")
1138
+ if include_logs:
1139
+ lines.extend(["", "## Daemon Error Excerpts", ""])
1140
+ if daemon_errors:
1141
+ for item in daemon_errors[:8]:
1142
+ payload = item.get("payload") if isinstance(item.get("payload"), dict) else {}
1143
+ message = str(payload.get("message") or item.get("event") or "").strip()
1144
+ lines.append(f"- level=`{item.get('level')}` event=`{item.get('event')}` message={message[:240]}")
1145
+ else:
1146
+ lines.append("- None captured in recent daemon tail.")
1147
+ if include_doctor:
1148
+ lines.extend(["", "## Cached Doctor Summary", ""])
1149
+ report = doctor_cache.get("report") if isinstance(doctor_cache, dict) and isinstance(doctor_cache.get("report"), dict) else None
1150
+ if isinstance(report, dict):
1151
+ lines.append(f"- ok: `{report.get('ok')}`")
1152
+ for item in (report.get("checks") or [])[:12]:
1153
+ if not isinstance(item, dict):
1154
+ continue
1155
+ lines.append(
1156
+ f"- `{item.get('id')}` status=`{item.get('status')}` summary={item.get('summary')}"
1157
+ )
1158
+ else:
1159
+ lines.append("- No cached doctor report available.")
1160
+ body_markdown = "\n".join(lines).strip() + "\n"
1161
+ return {
1162
+ "ok": True,
1163
+ "title": title,
1164
+ "body_markdown": body_markdown,
1165
+ "issue_url_base": issue_url_base,
1166
+ "repo_url": repo_url,
1167
+ "generated_at": utc_now(),
1168
+ "context": {
1169
+ "error_console": error_console,
1170
+ "doctor_cached": doctor_cache,
1171
+ },
1172
+ }
1173
+
1174
+ def built_in_controllers(self) -> list[dict[str, Any]]:
1175
+ state = self._controller_state()
1176
+ entries = state.get("controllers") if isinstance(state, dict) else {}
1177
+ if not isinstance(entries, dict):
1178
+ entries = {}
1179
+ catalog = [
1180
+ ("stale_running_quest_guard", "Detect quests stuck in running state without fresh tool activity."),
1181
+ ("repeated_runner_error_guard", "Detect quests repeatedly hitting runner turn failures."),
1182
+ ("connector_degraded_guard", "Detect connectors with last_error or unhealthy connection state."),
1183
+ ]
1184
+ items: list[dict[str, Any]] = []
1185
+ for controller_id, description in catalog:
1186
+ current = dict(entries.get(controller_id) or {}) if isinstance(entries.get(controller_id), dict) else {}
1187
+ items.append(
1188
+ {
1189
+ "controller_id": controller_id,
1190
+ "description": description,
1191
+ "enabled": bool(current.get("enabled")),
1192
+ "last_run_at": current.get("last_run_at"),
1193
+ "last_result": current.get("last_result"),
1194
+ }
1195
+ )
1196
+ return items
1197
+
1198
+ def controllers(self) -> dict[str, Any]:
1199
+ return {"ok": True, "items": self.built_in_controllers()}
1200
+
1201
+ def _controller_state_path(self) -> Path:
1202
+ return self.home / "runtime" / "admin" / "controllers.json"
1203
+
1204
+ def _controller_state(self) -> dict[str, Any]:
1205
+ payload = read_json(self._controller_state_path(), default={})
1206
+ if not isinstance(payload, dict):
1207
+ payload = {}
1208
+ payload.setdefault("controllers", {})
1209
+ return payload
1210
+
1211
+ def _write_controller_state(self, payload: dict[str, Any]) -> dict[str, Any]:
1212
+ from ..shared import write_json
1213
+
1214
+ write_json(self._controller_state_path(), payload)
1215
+ return payload
1216
+
1217
+ def controller_toggle(self, controller_id: str, *, enabled: bool) -> dict[str, Any]:
1218
+ payload = self._controller_state()
1219
+ controllers = payload.setdefault("controllers", {})
1220
+ current = dict(controllers.get(controller_id) or {}) if isinstance(controllers.get(controller_id), dict) else {}
1221
+ current["enabled"] = bool(enabled)
1222
+ current["updated_at"] = utc_now()
1223
+ controllers[controller_id] = current
1224
+ self._write_controller_state(payload)
1225
+ return next(item for item in self.built_in_controllers() if item["controller_id"] == controller_id)
1226
+
1227
+ def controller_run(self, controller_id: str) -> dict[str, Any]:
1228
+ handlers = {
1229
+ "stale_running_quest_guard": self._run_stale_running_quest_guard,
1230
+ "repeated_runner_error_guard": self._run_repeated_runner_error_guard,
1231
+ "connector_degraded_guard": self._run_connector_degraded_guard,
1232
+ }
1233
+ if controller_id not in handlers:
1234
+ raise FileNotFoundError(f"Unknown controller `{controller_id}`.")
1235
+ result = handlers[controller_id]()
1236
+ payload = self._controller_state()
1237
+ controllers = payload.setdefault("controllers", {})
1238
+ current = dict(controllers.get(controller_id) or {}) if isinstance(controllers.get(controller_id), dict) else {}
1239
+ current["last_run_at"] = utc_now()
1240
+ current["last_result"] = result
1241
+ controllers[controller_id] = current
1242
+ self._write_controller_state(payload)
1243
+ return result
1244
+
1245
+ def _run_stale_running_quest_guard(self) -> dict[str, Any]:
1246
+ hits: list[dict[str, Any]] = []
1247
+ threshold = datetime.now(UTC) - timedelta(minutes=30)
1248
+ for item in self.app.quest_service.list_quests():
1249
+ runtime_status = str(item.get("runtime_status") or item.get("status") or "").strip().lower()
1250
+ active_run = str(item.get("active_run_id") or "").strip()
1251
+ if runtime_status not in {"running", "active"} and not active_run:
1252
+ continue
1253
+ last_tool_activity = self._parse_iso(item.get("last_tool_activity_at"))
1254
+ if last_tool_activity is None or last_tool_activity <= threshold:
1255
+ hits.append(
1256
+ {
1257
+ "quest_id": item.get("quest_id"),
1258
+ "title": item.get("title"),
1259
+ "runtime_status": runtime_status,
1260
+ "active_run_id": active_run or None,
1261
+ "last_tool_activity_at": item.get("last_tool_activity_at"),
1262
+ }
1263
+ )
1264
+ return {
1265
+ "status": "warning" if hits else "ok",
1266
+ "hit_count": len(hits),
1267
+ "hits": hits,
1268
+ }
1269
+
1270
+ def _run_repeated_runner_error_guard(self) -> dict[str, Any]:
1271
+ grouped = Counter()
1272
+ details: dict[str, list[dict[str, Any]]] = {}
1273
+ for item in self.failure_records(limit=500):
1274
+ grouped[str(item.get("quest_id") or "")] += 1
1275
+ details.setdefault(str(item.get("quest_id") or ""), []).append(item)
1276
+ hits = [
1277
+ {
1278
+ "quest_id": quest_id,
1279
+ "failure_count": count,
1280
+ "latest_failure": (details.get(quest_id) or [None])[0],
1281
+ }
1282
+ for quest_id, count in grouped.items()
1283
+ if quest_id and count >= 2
1284
+ ]
1285
+ return {
1286
+ "status": "warning" if hits else "ok",
1287
+ "hit_count": len(hits),
1288
+ "hits": hits,
1289
+ }
1290
+
1291
+ def _run_connector_degraded_guard(self) -> dict[str, Any]:
1292
+ hits: list[dict[str, Any]] = []
1293
+ for item in self.app.list_connector_statuses():
1294
+ if not bool(item.get("enabled")):
1295
+ continue
1296
+ connection_state = str(item.get("connection_state") or "").strip().lower()
1297
+ last_error = str(item.get("last_error") or "").strip()
1298
+ if last_error or connection_state in {"error", "degraded", "offline"}:
1299
+ hits.append(
1300
+ {
1301
+ "name": item.get("name"),
1302
+ "connection_state": item.get("connection_state"),
1303
+ "last_error": item.get("last_error"),
1304
+ }
1305
+ )
1306
+ return {
1307
+ "status": "warning" if hits else "ok",
1308
+ "hit_count": len(hits),
1309
+ "hits": hits,
1310
+ }