@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.
- package/AGENTS.md +309 -130
- package/AISB/catalog/aisb.b1.agentic_coding.yaml +244 -0
- package/AISB/catalog/aisb.b10.climate_earth.yaml +235 -0
- package/AISB/catalog/aisb.b11.model_efficiency.yaml +231 -0
- package/AISB/catalog/aisb.b12.embodied_ai.yaml +238 -0
- package/AISB/catalog/aisb.b2.agent_systems.yaml +229 -0
- package/AISB/catalog/aisb.b3.self_evolving_rl.yaml +237 -0
- package/AISB/catalog/aisb.b4.lm_reasoning.yaml +240 -0
- package/AISB/catalog/aisb.b5.math_proof.yaml +235 -0
- package/AISB/catalog/aisb.b6.research_process.yaml +243 -0
- package/AISB/catalog/aisb.b7.multimodal_fusion.yaml +232 -0
- package/AISB/catalog/aisb.b8.lifesci_drug.yaml +275 -0
- package/AISB/catalog/aisb.b9.material_science.yaml +237 -0
- package/AISB/catalog/aisb.t3.001_savvy.yaml +159 -0
- package/AISB/catalog/aisb.t3.001_savvy.zh.yaml +121 -0
- package/AISB/catalog/aisb.t3.002_pinet.yaml +189 -0
- package/AISB/catalog/aisb.t3.002_pinet.zh.yaml +130 -0
- package/AISB/catalog/aisb.t3.004_decentralattn.yaml +184 -0
- package/AISB/catalog/aisb.t3.004_decentralattn.zh.yaml +153 -0
- package/AISB/catalog/aisb.t3.005_tsae.yaml +193 -0
- package/AISB/catalog/aisb.t3.005_tsae.zh.yaml +139 -0
- package/AISB/catalog/aisb.t3.006_physense.yaml +194 -0
- package/AISB/catalog/aisb.t3.006_physense.zh.yaml +118 -0
- package/AISB/catalog/aisb.t3.007_reasoningiqa.yaml +169 -0
- package/AISB/catalog/aisb.t3.007_reasoningiqa.zh.yaml +133 -0
- package/AISB/catalog/aisb.t3.008_meanflows.yaml +188 -0
- package/AISB/catalog/aisb.t3.008_meanflows.zh.yaml +140 -0
- package/AISB/catalog/aisb.t3.009_scoremissing.yaml +179 -0
- package/AISB/catalog/aisb.t3.009_scoremissing.zh.yaml +119 -0
- package/AISB/catalog/aisb.t3.010_suitabilityfilter.yaml +221 -0
- package/AISB/catalog/aisb.t3.010_suitabilityfilter.zh.yaml +141 -0
- package/AISB/catalog/aisb.t3.011_osd.yaml +206 -0
- package/AISB/catalog/aisb.t3.011_osd.zh.yaml +163 -0
- package/AISB/catalog/aisb.t3.012_efficientqat.yaml +206 -0
- package/AISB/catalog/aisb.t3.012_efficientqat.zh.yaml +159 -0
- package/AISB/catalog/aisb.t3.013_appl.yaml +152 -0
- package/AISB/catalog/aisb.t3.013_appl.zh.yaml +126 -0
- package/AISB/catalog/aisb.t3.014_piguard.yaml +207 -0
- package/AISB/catalog/aisb.t3.014_piguard.zh.yaml +164 -0
- package/AISB/catalog/aisb.t3.015_frspec.yaml +209 -0
- package/AISB/catalog/aisb.t3.015_frspec.zh.yaml +163 -0
- package/AISB/catalog/aisb.t3.016_mathfusion.yaml +166 -0
- package/AISB/catalog/aisb.t3.016_mathfusion.zh.yaml +145 -0
- package/AISB/catalog/aisb.t3.017_multimodalglp.yaml +171 -0
- package/AISB/catalog/aisb.t3.017_multimodalglp.zh.yaml +122 -0
- package/AISB/catalog/aisb.t3.018_cotsynth.yaml +206 -0
- package/AISB/catalog/aisb.t3.018_cotsynth.zh.yaml +162 -0
- package/AISB/catalog/aisb.t3.019_dyscaleut.yaml +211 -0
- package/AISB/catalog/aisb.t3.019_dyscaleut.zh.yaml +148 -0
- package/AISB/catalog/aisb.t3.020_aristotle.yaml +173 -0
- package/AISB/catalog/aisb.t3.020_aristotle.zh.yaml +119 -0
- package/AISB/catalog/aisb.t3.021_tokenrecycling.yaml +160 -0
- package/AISB/catalog/aisb.t3.021_tokenrecycling.zh.yaml +129 -0
- package/AISB/catalog/aisb.t3.022_chainofreasoning.yaml +204 -0
- package/AISB/catalog/aisb.t3.022_chainofreasoning.zh.yaml +161 -0
- package/AISB/catalog/aisb.t3.023_guidedembed.yaml +211 -0
- package/AISB/catalog/aisb.t3.023_guidedembed.zh.yaml +189 -0
- package/AISB/catalog/aisb.t3.024_outputcentric.yaml +148 -0
- package/AISB/catalog/aisb.t3.024_outputcentric.zh.yaml +131 -0
- package/AISB/catalog/aisb.t3.025_deeper.yaml +143 -0
- package/AISB/catalog/aisb.t3.025_deeper.zh.yaml +116 -0
- package/AISB/catalog/aisb.t3.026_gartkg.yaml +195 -0
- package/AISB/catalog/aisb.t3.026_gartkg.zh.yaml +127 -0
- package/AISB/catalog/aisb.t3.027_citeeval.yaml +182 -0
- package/AISB/catalog/aisb.t3.027_citeeval.zh.yaml +135 -0
- package/AISB/catalog/aisb.t3.028_sbam.yaml +206 -0
- package/AISB/catalog/aisb.t3.028_sbam.zh.yaml +166 -0
- package/AISB/catalog/aisb.t3.029_cdqgeoembed.yaml +224 -0
- package/AISB/catalog/aisb.t3.029_cdqgeoembed.zh.yaml +142 -0
- package/AISB/catalog/aisb.t3.030_processrm.yaml +211 -0
- package/AISB/catalog/aisb.t3.030_processrm.zh.yaml +166 -0
- package/AISB/catalog/aisb.t3.031_circuitstability.yaml +172 -0
- package/AISB/catalog/aisb.t3.031_circuitstability.zh.yaml +134 -0
- package/AISB/catalog/aisb.t3.032_ptsolver.yaml +169 -0
- package/AISB/catalog/aisb.t3.032_ptsolver.zh.yaml +135 -0
- package/AISB/catalog/aisb.t3.033_gcse.yaml +144 -0
- package/AISB/catalog/aisb.t3.033_gcse.zh.yaml +126 -0
- package/AISB/catalog/aisb.t3.034_ensemblewm.yaml +183 -0
- package/AISB/catalog/aisb.t3.034_ensemblewm.zh.yaml +146 -0
- package/AISB/catalog/aisb.t3.035_moralvalueswa.yaml +207 -0
- package/AISB/catalog/aisb.t3.035_moralvalueswa.zh.yaml +165 -0
- package/AISB/catalog/aisb.t3.036_weakstrongpref.yaml +210 -0
- package/AISB/catalog/aisb.t3.036_weakstrongpref.zh.yaml +194 -0
- package/AISB/catalog/aisb.t3.037_dementiamask.yaml +172 -0
- package/AISB/catalog/aisb.t3.037_dementiamask.zh.yaml +132 -0
- package/AISB/catalog/aisb.t3.038_tinysam.yaml +284 -0
- package/AISB/catalog/aisb.t3.038_tinysam.zh.yaml +240 -0
- package/AISB/catalog/aisb.t3.039_calf.yaml +224 -0
- package/AISB/catalog/aisb.t3.039_calf.zh.yaml +194 -0
- package/AISB/catalog/aisb.t3.040_graniteguardian.yaml +199 -0
- package/AISB/catalog/aisb.t3.040_graniteguardian.zh.yaml +174 -0
- package/AISB/catalog/aisb.t3.041_amdm.yaml +149 -0
- package/AISB/catalog/aisb.t3.041_amdm.zh.yaml +137 -0
- package/AISB/catalog/aisb.t3.042_xpatch.yaml +216 -0
- package/AISB/catalog/aisb.t3.042_xpatch.zh.yaml +182 -0
- package/AISB/catalog/aisb.t3.043_vhm.yaml +268 -0
- package/AISB/catalog/aisb.t3.043_vhm.zh.yaml +193 -0
- package/AISB/catalog/aisb.t3.044_rgvi.yaml +224 -0
- package/AISB/catalog/aisb.t3.044_rgvi.zh.yaml +176 -0
- package/AISB/catalog/aisb.t3.045_pslstm.yaml +203 -0
- package/AISB/catalog/aisb.t3.045_pslstm.zh.yaml +179 -0
- package/AISB/catalog/aisb.t3.046_nonstatts.yaml +208 -0
- package/AISB/catalog/aisb.t3.046_nonstatts.zh.yaml +194 -0
- package/AISB/catalog/aisb.t3.047_timepfn.yaml +156 -0
- package/AISB/catalog/aisb.t3.047_timepfn.zh.yaml +124 -0
- package/AISB/catalog/aisb.t3.048_proxyspex.yaml +148 -0
- package/AISB/catalog/aisb.t3.048_proxyspex.zh.yaml +125 -0
- package/AISB/catalog/aisb.t3.049_hogwildinference.yaml +183 -0
- package/AISB/catalog/aisb.t3.049_hogwildinference.zh.yaml +138 -0
- package/AISB/catalog/aisb.t3.050_causalpfn.yaml +214 -0
- package/AISB/catalog/aisb.t3.050_causalpfn.zh.yaml +190 -0
- package/AISB/catalog/aisb.t3.051_flashtp.yaml +169 -0
- package/AISB/catalog/aisb.t3.051_flashtp.zh.yaml +124 -0
- package/AISB/catalog/aisb.t3.052_nsdiff.yaml +155 -0
- package/AISB/catalog/aisb.t3.052_nsdiff.zh.yaml +138 -0
- package/AISB/catalog/aisb.t3.053_k2vae.yaml +158 -0
- package/AISB/catalog/aisb.t3.053_k2vae.zh.yaml +132 -0
- package/AISB/catalog/aisb.t3.054_timebase.yaml +178 -0
- package/AISB/catalog/aisb.t3.054_timebase.zh.yaml +158 -0
- package/AISB/catalog/aisb.t3.055_csbrain.yaml +238 -0
- package/AISB/catalog/aisb.t3.055_csbrain.zh.yaml +184 -0
- package/AISB/catalog/aisb.t3.056_infosam.yaml +224 -0
- package/AISB/catalog/aisb.t3.056_infosam.zh.yaml +189 -0
- package/AISB/catalog/aisb.t3.057_mdreid.yaml +129 -0
- package/AISB/catalog/aisb.t3.057_mdreid.zh.yaml +117 -0
- package/AISB/catalog/aisb.t3.058_mindglitch.yaml +171 -0
- package/AISB/catalog/aisb.t3.058_mindglitch.zh.yaml +145 -0
- package/AISB/catalog/aisb.t3.059_selfsupervised.yaml +154 -0
- package/AISB/catalog/aisb.t3.059_selfsupervised.zh.yaml +125 -0
- package/AISB/catalog/aisb.t3.060_iaggad.yaml +121 -0
- package/AISB/catalog/aisb.t3.060_iaggad.zh.yaml +100 -0
- package/AISB/catalog/aisb.t3.061_hsgkn.yaml +136 -0
- package/AISB/catalog/aisb.t3.061_hsgkn.zh.yaml +113 -0
- package/AISB/catalog/aisb.t3.062_visionts.yaml +237 -0
- package/AISB/catalog/aisb.t3.062_visionts.zh.yaml +216 -0
- package/AISB/catalog/aisb.t3.063_tsrag.yaml +162 -0
- package/AISB/catalog/aisb.t3.063_tsrag.zh.yaml +138 -0
- package/AISB/catalog/aisb.t3.064_pir.yaml +221 -0
- package/AISB/catalog/aisb.t3.064_pir.zh.yaml +197 -0
- package/AISB/catalog/aisb.t3.065_proteinbinding.yaml +234 -0
- package/AISB/catalog/aisb.t3.065_proteinbinding.zh.yaml +167 -0
- package/AISB/catalog/aisb.t3.066_tropicalattention.yaml +267 -0
- package/AISB/catalog/aisb.t3.066_tropicalattention.zh.yaml +229 -0
- package/AISB/catalog/aisb.t3.067_kanad.yaml +193 -0
- package/AISB/catalog/aisb.t3.067_kanad.zh.yaml +167 -0
- package/AISB/catalog/aisb.t3.068_sempo.yaml +187 -0
- package/AISB/catalog/aisb.t3.068_sempo.zh.yaml +148 -0
- package/AISB/catalog/aisb.t3.069_treehfd.yaml +129 -0
- package/AISB/catalog/aisb.t3.069_treehfd.zh.yaml +111 -0
- package/AISB/catalog/aisb.t3.070_certifiedunlearning.yaml +224 -0
- package/AISB/catalog/aisb.t3.070_certifiedunlearning.zh.yaml +171 -0
- package/AISB/catalog/aisb.t3.071_neuralmjd.yaml +142 -0
- package/AISB/catalog/aisb.t3.071_neuralmjd.zh.yaml +120 -0
- package/AISB/catalog/aisb.t3.072_fedgmt.yaml +181 -0
- package/AISB/catalog/aisb.t3.072_fedgmt.zh.yaml +158 -0
- package/AISB/catalog/aisb.t3.073_rld.yaml +161 -0
- package/AISB/catalog/aisb.t3.073_rld.zh.yaml +129 -0
- package/AISB/catalog/aisb.t3.074_lsvi.yaml +163 -0
- package/AISB/catalog/aisb.t3.074_lsvi.zh.yaml +129 -0
- package/AISB/catalog/aisb.t3.075_treeslicedentropy.yaml +201 -0
- package/AISB/catalog/aisb.t3.075_treeslicedentropy.zh.yaml +148 -0
- package/AISB/catalog/aisb.t3.076_aanet.yaml +169 -0
- package/AISB/catalog/aisb.t3.076_aanet.zh.yaml +129 -0
- package/AISB/catalog/aisb.t3.077_cmnn.yaml +199 -0
- package/AISB/catalog/aisb.t3.077_cmnn.zh.yaml +165 -0
- package/AISB/catalog/aisb.t3.078_conformalanomaly.yaml +146 -0
- package/AISB/catalog/aisb.t3.078_conformalanomaly.zh.yaml +117 -0
- package/AISB/catalog/aisb.t3.079_dpfkmeans.yaml +131 -0
- package/AISB/catalog/aisb.t3.079_dpfkmeans.zh.yaml +104 -0
- package/AISB/catalog/aisb.t3.080_latentscorereweight.yaml +169 -0
- package/AISB/catalog/aisb.t3.080_latentscorereweight.zh.yaml +123 -0
- package/AISB/catalog/aisb.t3.081_qmamba.yaml +150 -0
- package/AISB/catalog/aisb.t3.081_qmamba.zh.yaml +117 -0
- package/AISB/catalog/aisb.t3.082_onlinellmrouting.yaml +160 -0
- package/AISB/catalog/aisb.t3.082_onlinellmrouting.zh.yaml +133 -0
- package/AISB/catalog/aisb.t3.083_starformer.yaml +178 -0
- package/AISB/catalog/aisb.t3.083_starformer.zh.yaml +140 -0
- package/AISB/catalog/aisb.t3.084_ift.yaml +139 -0
- package/AISB/catalog/aisb.t3.084_ift.zh.yaml +111 -0
- package/AISB/catalog/aisb.t3.085_neuralsurv.yaml +183 -0
- package/AISB/catalog/aisb.t3.085_neuralsurv.zh.yaml +143 -0
- package/AISB/catalog/aisb.t3.086_stella.yaml +197 -0
- package/AISB/catalog/aisb.t3.086_stella.zh.yaml +142 -0
- package/AISB/catalog/aisb.t3.087_moses.yaml +167 -0
- package/AISB/catalog/aisb.t3.087_moses.zh.yaml +132 -0
- package/AISB/catalog/aisb.t3.088_channelnorm.yaml +140 -0
- package/AISB/catalog/aisb.t3.088_channelnorm.zh.yaml +109 -0
- package/AISB/catalog/aisb.t3.089_causalvelocity.yaml +730 -0
- package/AISB/catalog/aisb.t3.089_causalvelocity.zh.yaml +668 -0
- package/AISB/catalog/aisb.t3.090_rstib.yaml +144 -0
- package/AISB/catalog/aisb.t3.090_rstib.zh.yaml +109 -0
- package/AISB/catalog/aisb.t3.091_timeawarecausal.yaml +132 -0
- package/AISB/catalog/aisb.t3.091_timeawarecausal.zh.yaml +107 -0
- package/AISB/catalog/aisb.t3.092_kmeanslocalopt.yaml +138 -0
- package/AISB/catalog/aisb.t3.092_kmeanslocalopt.zh.yaml +110 -0
- package/AISB/catalog/aisb.t3.093_fedwmsam.yaml +134 -0
- package/AISB/catalog/aisb.t3.093_fedwmsam.zh.yaml +106 -0
- package/AISB/catalog/aisb.t3.094_boundre.yaml +147 -0
- package/AISB/catalog/aisb.t3.094_boundre.zh.yaml +114 -0
- package/AISB/catalog/aisb.t3.095_fastfeaturecp.yaml +153 -0
- package/AISB/catalog/aisb.t3.095_fastfeaturecp.zh.yaml +118 -0
- package/AISB/catalog/aisb.t3.096_m3svm.yaml +189 -0
- package/AISB/catalog/aisb.t3.096_m3svm.zh.yaml +149 -0
- package/AISB/catalog/aisb.t3.097_wassersteintl.yaml +212 -0
- package/AISB/catalog/aisb.t3.097_wassersteintl.zh.yaml +169 -0
- package/AISB/catalog/aisb.t3.098_xmahalanobis.yaml +171 -0
- package/AISB/catalog/aisb.t3.098_xmahalanobis.zh.yaml +127 -0
- package/AISB/catalog/aisb.t3.099_ollalanding.yaml +248 -0
- package/AISB/catalog/aisb.t3.099_ollalanding.zh.yaml +182 -0
- package/AISB/catalog/aisb.t3.100_invmissingdata.yaml +179 -0
- package/AISB/catalog/aisb.t3.100_invmissingdata.zh.yaml +150 -0
- package/AISB/catalog/aisb.t3.101_acia.yaml +164 -0
- package/AISB/catalog/aisb.t3.101_acia.zh.yaml +109 -0
- package/AISB/catalog/aisb.t3.102_stochasticff.yaml +178 -0
- package/AISB/catalog/aisb.t3.102_stochasticff.zh.yaml +130 -0
- package/AISB/catalog/aisb.t3.103_qdcp.yaml +150 -0
- package/AISB/catalog/aisb.t3.103_qdcp.zh.yaml +116 -0
- package/AISB/catalog/aisb.t3.104_balancedactiveinf.yaml +137 -0
- package/AISB/catalog/aisb.t3.104_balancedactiveinf.zh.yaml +104 -0
- package/AISB/catalog/aisb.t3.105_binaryclasseval.yaml +161 -0
- package/AISB/catalog/aisb.t3.105_binaryclasseval.zh.yaml +130 -0
- package/AISB/image/001_aisb.t3.001_savvy.jpg +0 -0
- package/AISB/image/002_aisb.t3.002_pinet.jpg +0 -0
- package/AISB/image/003_aisb.t3.003_dmsqd.jpg +0 -0
- package/AISB/image/004_aisb.t3.004_decentralattn.jpg +0 -0
- package/AISB/image/005_aisb.t3.005_tsae.jpg +0 -0
- package/AISB/image/006_aisb.t3.006_physense.jpg +0 -0
- package/AISB/image/007_aisb.t3.007_reasoningiqa.jpg +0 -0
- package/AISB/image/008_aisb.t3.008_meanflows.jpg +0 -0
- package/AISB/image/009_aisb.t3.009_scoremissing.jpg +0 -0
- package/AISB/image/010_aisb.t3.010_suitabilityfilter.jpg +0 -0
- package/AISB/image/011_aisb.t3.011_osd.jpg +0 -0
- package/AISB/image/012_aisb.t3.012_efficientqat.jpg +0 -0
- package/AISB/image/013_aisb.t3.013_appl.jpg +0 -0
- package/AISB/image/014_aisb.t3.014_piguard.jpg +0 -0
- package/AISB/image/015_aisb.t3.015_frspec.jpg +0 -0
- package/AISB/image/016_aisb.t3.016_mathfusion.jpg +0 -0
- package/AISB/image/017_aisb.t3.017_multimodalglp.jpg +0 -0
- package/AISB/image/018_aisb.t3.018_cotsynth.jpg +0 -0
- package/AISB/image/019_aisb.t3.019_dyscaleut.jpg +0 -0
- package/AISB/image/020_aisb.t3.020_aristotle.jpg +0 -0
- package/AISB/image/021_aisb.t3.021_tokenrecycling.jpg +0 -0
- package/AISB/image/022_aisb.t3.022_chainofreasoning.jpg +0 -0
- package/AISB/image/023_aisb.t3.023_guidedembed.jpg +0 -0
- package/AISB/image/024_aisb.t3.024_outputcentric.jpg +0 -0
- package/AISB/image/025_aisb.t3.025_deeper.jpg +0 -0
- package/AISB/image/026_aisb.t3.026_gartkg.jpg +0 -0
- package/AISB/image/027_aisb.t3.027_citeeval.jpg +0 -0
- package/AISB/image/028_aisb.t3.028_sbam.jpg +0 -0
- package/AISB/image/029_aisb.t3.029_cdqgeoembed.jpg +0 -0
- package/AISB/image/030_aisb.t3.030_processrm.jpg +0 -0
- package/AISB/image/031_aisb.t3.031_circuitstability.jpg +0 -0
- package/AISB/image/032_aisb.t3.032_ptsolver.jpg +0 -0
- package/AISB/image/033_aisb.t3.033_gcse.jpg +0 -0
- package/AISB/image/034_aisb.t3.034_ensemblewm.jpg +0 -0
- package/AISB/image/035_aisb.t3.035_moralvalueswa.jpg +0 -0
- package/AISB/image/036_aisb.t3.036_weakstrongpref.jpg +0 -0
- package/AISB/image/037_aisb.t3.037_dementiamask.jpg +0 -0
- package/AISB/image/038_aisb.t3.038_tinysam.jpg +0 -0
- package/AISB/image/039_aisb.t3.039_calf.jpg +0 -0
- package/AISB/image/040_aisb.t3.040_graniteguardian.jpg +0 -0
- package/AISB/image/041_aisb.t3.041_amdm.jpg +0 -0
- package/AISB/image/042_aisb.t3.042_xpatch.jpg +0 -0
- package/AISB/image/043_aisb.t3.043_vhm.jpg +0 -0
- package/AISB/image/044_aisb.t3.044_rgvi.jpg +0 -0
- package/AISB/image/045_aisb.t3.045_pslstm.jpg +0 -0
- package/AISB/image/046_aisb.t3.046_nonstatts.jpg +0 -0
- package/AISB/image/047_aisb.t3.047_timepfn.jpg +0 -0
- package/AISB/image/048_aisb.t3.048_proxyspex.jpg +0 -0
- package/AISB/image/049_aisb.t3.049_hogwildinference.jpg +0 -0
- package/AISB/image/050_aisb.t3.050_causalpfn.jpg +0 -0
- package/AISB/image/051_aisb.t3.051_flashtp.jpg +0 -0
- package/AISB/image/052_aisb.t3.052_nsdiff.jpg +0 -0
- package/AISB/image/053_aisb.t3.053_k2vae.jpg +0 -0
- package/AISB/image/054_aisb.t3.054_timebase.jpg +0 -0
- package/AISB/image/055_aisb.t3.055_csbrain.jpg +0 -0
- package/AISB/image/056_aisb.t3.056_infosam.jpg +0 -0
- package/AISB/image/057_aisb.t3.057_mdreid.jpg +0 -0
- package/AISB/image/058_aisb.t3.058_mindglitch.jpg +0 -0
- package/AISB/image/059_aisb.t3.059_selfsupervised.jpg +0 -0
- package/AISB/image/060_aisb.t3.060_iaggad.jpg +0 -0
- package/AISB/image/061_aisb.t3.061_hsgkn.jpg +0 -0
- package/AISB/image/062_aisb.t3.062_visionts.jpg +0 -0
- package/AISB/image/063_aisb.t3.063_tsrag.jpg +0 -0
- package/AISB/image/064_aisb.t3.064_pir.jpg +0 -0
- package/AISB/image/065_aisb.t3.065_proteinbinding.jpg +0 -0
- package/AISB/image/066_aisb.t3.066_tropicalattention.jpg +0 -0
- package/AISB/image/067_aisb.t3.067_kanad.jpg +0 -0
- package/AISB/image/068_aisb.t3.068_sempo.jpg +0 -0
- package/AISB/image/069_aisb.t3.069_treehfd.jpg +0 -0
- package/AISB/image/070_aisb.t3.070_certifiedunlearning.jpg +0 -0
- package/AISB/image/071_aisb.t3.071_neuralmjd.jpg +0 -0
- package/AISB/image/072_aisb.t3.072_fedgmt.jpg +0 -0
- package/AISB/image/073_aisb.t3.073_rld.jpg +0 -0
- package/AISB/image/074_aisb.t3.074_lsvi.jpg +0 -0
- package/AISB/image/075_aisb.t3.075_treeslicedentropy.jpg +0 -0
- package/AISB/image/076_aisb.t3.076_aanet.jpg +0 -0
- package/AISB/image/077_aisb.t3.077_cmnn.jpg +0 -0
- package/AISB/image/078_aisb.t3.078_conformalanomaly.jpg +0 -0
- package/AISB/image/079_aisb.t3.079_dpfkmeans.jpg +0 -0
- package/AISB/image/080_aisb.t3.080_latentscorereweight.jpg +0 -0
- package/AISB/image/081_aisb.t3.081_qmamba.jpg +0 -0
- package/AISB/image/082_aisb.t3.082_onlinellmrouting.jpg +0 -0
- package/AISB/image/083_aisb.t3.083_starformer.jpg +0 -0
- package/AISB/image/084_aisb.t3.084_ift.jpg +0 -0
- package/AISB/image/085_aisb.t3.085_neuralsurv.jpg +0 -0
- package/AISB/image/086_aisb.t3.086_stella.jpg +0 -0
- package/AISB/image/087_aisb.t3.087_moses.jpg +0 -0
- package/AISB/image/088_aisb.t3.088_channelnorm.jpg +0 -0
- package/AISB/image/089_aisb.t3.089_causalvelocity.jpg +0 -0
- package/AISB/image/090_aisb.t3.090_rstib.jpg +0 -0
- package/AISB/image/091_aisb.t3.091_timeawarecausal.jpg +0 -0
- package/AISB/image/092_aisb.t3.092_kmeanslocalopt.jpg +0 -0
- package/AISB/image/093_aisb.t3.093_fedwmsam.jpg +0 -0
- package/AISB/image/094_aisb.t3.094_boundre.jpg +0 -0
- package/AISB/image/095_aisb.t3.095_fastfeaturecp.jpg +0 -0
- package/AISB/image/096_aisb.t3.096_m3svm.jpg +0 -0
- package/AISB/image/097_aisb.t3.097_wassersteintl.jpg +0 -0
- package/AISB/image/098_aisb.t3.098_xmahalanobis.jpg +0 -0
- package/AISB/image/099_aisb.t3.099_ollalanding.jpg +0 -0
- package/AISB/image/100_aisb.t3.100_invmissingdata.jpg +0 -0
- package/AISB/image/101_aisb.t3.101_acia.jpg +0 -0
- package/AISB/image/102_aisb.t3.102_stochasticff.jpg +0 -0
- package/AISB/image/103_aisb.t3.103_qdcp.jpg +0 -0
- package/AISB/image/104_aisb.t3.104_balancedactiveinf.jpg +0 -0
- package/AISB/image/105_aisb.t3.105_binaryclasseval.jpg +0 -0
- package/AISB/image/106_aisb.t1.reasoning_lite.jpg +0 -0
- package/AISB/image/107_aisb.t2.paper_audit.jpg +0 -0
- package/AISB/image/108_aisb.t3.multi_gpu_search.jpg +0 -0
- package/AISB/image/109_aisb.t3.tdc_admet.jpg +0 -0
- package/AISB/image/aisb.b1.agentic_coding.svg +16 -0
- package/AISB/image/aisb.b10.climate_earth.svg +16 -0
- package/AISB/image/aisb.b11.model_efficiency.svg +16 -0
- package/AISB/image/aisb.b12.embodied_ai.svg +16 -0
- package/AISB/image/aisb.b2.agent_systems.svg +16 -0
- package/AISB/image/aisb.b3.self_evolving_rl.svg +16 -0
- package/AISB/image/aisb.b4.lm_reasoning.svg +16 -0
- package/AISB/image/aisb.b5.math_proof.svg +16 -0
- package/AISB/image/aisb.b6.research_process.svg +16 -0
- package/AISB/image/aisb.b7.multimodal_fusion.svg +16 -0
- package/AISB/image/aisb.b8.lifesci_drug.svg +16 -0
- package/AISB/image/aisb.b9.material_science.svg +16 -0
- package/README.md +132 -11
- package/bin/ds.js +376 -49
- package/docs/en/00_QUICK_START.md +135 -18
- package/docs/en/01_SETTINGS_REFERENCE.md +468 -96
- package/docs/en/02_START_RESEARCH_GUIDE.md +26 -5
- package/docs/en/03_QQ_CONNECTOR_GUIDE.md +14 -3
- package/docs/en/04_LINGZHU_CONNECTOR_GUIDE.md +2 -0
- package/docs/en/05_TUI_GUIDE.md +171 -2
- package/docs/en/07_MEMORY_AND_MCP.md +38 -2
- package/docs/en/09_DOCTOR.md +64 -4
- package/docs/en/10_WEIXIN_CONNECTOR_GUIDE.md +38 -1
- package/docs/en/11_LICENSE_AND_RISK.md +4 -0
- package/docs/en/12_GUIDED_WORKFLOW_TOUR.md +15 -0
- package/docs/en/14_PROMPT_SKILLS_AND_MCP_GUIDE.md +9 -0
- package/docs/en/15_CODEX_PROVIDER_SETUP.md +622 -187
- package/docs/en/16_TELEGRAM_CONNECTOR_GUIDE.md +14 -0
- package/docs/en/17_WHATSAPP_CONNECTOR_GUIDE.md +14 -0
- package/docs/en/18_FEISHU_CONNECTOR_GUIDE.md +14 -0
- package/docs/en/21_LOCAL_MODEL_BACKENDS_GUIDE.md +105 -2
- package/docs/en/22_BENCHSTORE_YAML_REFERENCE.md +469 -0
- package/docs/en/23_BENCHSTORE_GITHUB_RELEASES_SPEC.md +316 -0
- package/docs/en/24_CLAUDE_CODE_PROVIDER_SETUP.md +469 -0
- package/docs/en/25_OPENCODE_PROVIDER_SETUP.md +653 -0
- package/docs/en/26_CITATION_AND_ATTRIBUTION.md +119 -0
- package/docs/en/27_KIMI_CODE_PROVIDER_SETUP.md +180 -0
- package/docs/en/28_DISCORD_CONNECTOR_GUIDE.md +61 -0
- package/docs/en/29_SLACK_CONNECTOR_GUIDE.md +60 -0
- package/docs/en/30_SETTINGS_CONTROL_CENTER_GUIDE.md +371 -0
- package/docs/en/{19_LOCAL_BROWSER_AUTH.md → 31_LOCAL_BROWSER_AUTH.md} +1 -1
- package/docs/en/32_WINDOWS_WSL2_DEPLOYMENT_GUIDE.md +273 -0
- package/docs/en/33_WORKSPACE_EXPLORER_QA.md +121 -0
- package/docs/en/91_DEVELOPMENT.md +29 -0
- package/docs/en/99_ACKNOWLEDGEMENTS.md +24 -19
- package/docs/en/README.md +44 -7
- package/docs/images/admin/admin-connectors-health-en.png +0 -0
- package/docs/images/admin/admin-controllers-en.png +0 -0
- package/docs/images/admin/admin-diagnostics-en.png +0 -0
- package/docs/images/admin/admin-errors-en.png +0 -0
- package/docs/images/admin/admin-issues-en.png +0 -0
- package/docs/images/admin/admin-logs-en.png +0 -0
- package/docs/images/admin/admin-quest-detail-en.png +0 -0
- package/docs/images/admin/admin-quests-en.png +0 -0
- package/docs/images/admin/admin-repairs-en.png +0 -0
- package/docs/images/admin/admin-runtime-en.png +0 -0
- package/docs/images/admin/admin-search-en.png +0 -0
- package/docs/images/admin/admin-stats-en.png +0 -0
- package/docs/images/admin/admin-summary-en.png +0 -0
- package/docs/images/connectors/connector-discord-en.png +0 -0
- package/docs/images/connectors/connector-feishu-en.png +0 -0
- package/docs/images/connectors/connector-lingzhu-en.png +0 -0
- package/docs/images/connectors/connector-qq-en.png +0 -0
- package/docs/images/connectors/connector-slack-en.png +0 -0
- package/docs/images/connectors/connector-telegram-en.png +0 -0
- package/docs/images/connectors/connector-weixin-en.png +0 -0
- package/docs/images/connectors/connector-whatsapp-en.png +0 -0
- package/docs/images/settings/settings-baselines-en.png +0 -0
- package/docs/images/settings/settings-config-en.png +0 -0
- package/docs/images/settings/settings-connectors-overview-en.png +0 -0
- package/docs/images/settings/settings-deepxiv-en.png +0 -0
- package/docs/images/settings/settings-mcp-servers-en.png +0 -0
- package/docs/images/settings/settings-plugins-en.png +0 -0
- package/docs/images/settings/settings-runners-en.png +0 -0
- package/docs/zh/00_QUICK_START.md +92 -17
- package/docs/zh/01_SETTINGS_REFERENCE.md +219 -98
- package/docs/zh/02_START_RESEARCH_GUIDE.md +26 -5
- package/docs/zh/05_TUI_GUIDE.md +171 -2
- package/docs/zh/07_MEMORY_AND_MCP.md +29 -2
- package/docs/zh/09_DOCTOR.md +39 -4
- package/docs/zh/10_WEIXIN_CONNECTOR_GUIDE.md +24 -1
- package/docs/zh/11_LICENSE_AND_RISK.md +4 -0
- package/docs/zh/12_GUIDED_WORKFLOW_TOUR.md +15 -0
- package/docs/zh/14_PROMPT_SKILLS_AND_MCP_GUIDE.md +9 -0
- package/docs/zh/15_CODEX_PROVIDER_SETUP.md +550 -188
- package/docs/zh/21_LOCAL_MODEL_BACKENDS_GUIDE.md +105 -2
- package/docs/zh/22_BENCHSTORE_YAML_REFERENCE.md +459 -0
- package/docs/zh/23_BENCHSTORE_GITHUB_RELEASES_SPEC.md +287 -0
- package/docs/zh/23_CLAUDE_RUNNER_GUIDE.md +103 -0
- package/docs/zh/24_CLAUDE_CODE_PROVIDER_SETUP.md +460 -0
- package/docs/zh/25_OPENCODE_PROVIDER_SETUP.md +660 -0
- package/docs/zh/26_CITATION_AND_ATTRIBUTION.md +102 -0
- package/docs/zh/27_KIMI_CODE_PROVIDER_SETUP.md +51 -0
- package/docs/zh/{19_LOCAL_BROWSER_AUTH.md → 31_LOCAL_BROWSER_AUTH.md} +1 -1
- package/docs/zh/32_WINDOWS_WSL2_DEPLOYMENT_GUIDE.md +264 -0
- package/docs/zh/33_WORKSPACE_EXPLORER_QA.md +127 -0
- package/docs/zh/99_ACKNOWLEDGEMENTS.md +23 -19
- package/docs/zh/README.md +29 -7
- package/install.sh +122 -16
- package/package.json +4 -1
- package/pyproject.toml +2 -1
- package/src/deepscientist/__init__.py +1 -1
- package/src/deepscientist/acp/envelope.py +13 -0
- package/src/deepscientist/admin/__init__.py +3 -0
- package/src/deepscientist/admin/charts.py +681 -0
- package/src/deepscientist/admin/logs.py +119 -0
- package/src/deepscientist/admin/repairs.py +217 -0
- package/src/deepscientist/admin/service.py +1310 -0
- package/src/deepscientist/admin/system_info.py +700 -0
- package/src/deepscientist/admin/tasks.py +465 -0
- package/src/deepscientist/admin/tool_metrics.py +600 -0
- package/src/deepscientist/artifact/guidance.py +8 -4
- package/src/deepscientist/artifact/schemas.py +115 -0
- package/src/deepscientist/artifact/service.py +4268 -260
- package/src/deepscientist/bash_exec/monitor.py +30 -3
- package/src/deepscientist/bash_exec/service.py +134 -1
- package/src/deepscientist/benchstore/__init__.py +4 -0
- package/src/deepscientist/benchstore/prompt_builder.py +224 -0
- package/src/deepscientist/benchstore/service.py +1716 -0
- package/src/deepscientist/channels/weixin_ilink.py +8 -1
- package/src/deepscientist/cli.py +92 -17
- package/src/deepscientist/codex_cli_compat.py +2 -2
- package/src/deepscientist/config/models.py +82 -11
- package/src/deepscientist/config/service.py +927 -91
- package/src/deepscientist/connector/weixin_support.py +48 -17
- package/src/deepscientist/daemon/api/handlers.py +697 -210
- package/src/deepscientist/daemon/api/router.py +76 -1
- package/src/deepscientist/daemon/app.py +1054 -51
- package/src/deepscientist/diagnostics/runner_failures.py +147 -0
- package/src/deepscientist/doctor.py +212 -65
- package/src/deepscientist/evidence_packets.py +590 -0
- package/src/deepscientist/home.py +52 -4
- package/src/deepscientist/kimi_cli_compat.py +50 -0
- package/src/deepscientist/latex_runtime.py +2 -2
- package/src/deepscientist/mcp/context.py +2 -0
- package/src/deepscientist/mcp/schemas.py +114 -0
- package/src/deepscientist/mcp/server.py +1566 -126
- package/src/deepscientist/memory/service.py +203 -16
- package/src/deepscientist/process_control.py +8 -1
- package/src/deepscientist/prompts/builder.py +836 -92
- package/src/deepscientist/quest/__init__.py +2 -2
- package/src/deepscientist/quest/layout.py +12 -1
- package/src/deepscientist/quest/node_traces.py +10 -0
- package/src/deepscientist/quest/service.py +1430 -139
- package/src/deepscientist/quest/stage_views.py +1 -1
- package/src/deepscientist/runners/__init__.py +18 -0
- package/src/deepscientist/runners/base.py +89 -1
- package/src/deepscientist/runners/builtins.py +13 -1
- package/src/deepscientist/runners/claude.py +391 -0
- package/src/deepscientist/runners/codex.py +421 -21
- package/src/deepscientist/runners/codex_telemetry.py +127 -0
- package/src/deepscientist/runners/kimi.py +334 -0
- package/src/deepscientist/runners/metadata.py +68 -0
- package/src/deepscientist/runners/opencode.py +414 -0
- package/src/deepscientist/runners/runtime_overrides.py +100 -0
- package/src/deepscientist/runners/simple_cli.py +538 -0
- package/src/deepscientist/runtime_storage.py +303 -0
- package/src/deepscientist/shared.py +61 -16
- package/src/deepscientist/skills/installer.py +37 -0
- package/src/deepscientist/skills/registry.py +2 -0
- package/src/deepscientist/tinytex.py +2 -2
- package/src/deepscientist/tui.py +10 -3
- package/src/prompts/benchstore/system.md +77 -0
- package/src/prompts/connectors/qq.md +33 -2
- package/src/prompts/connectors/weixin.md +208 -23
- package/src/prompts/contracts/admin_ops.md +74 -0
- package/src/prompts/contracts/admin_ops_knowledge.md +138 -0
- package/src/prompts/contracts/shared_interaction.md +5 -11
- package/src/prompts/start_setup/system.md +422 -0
- package/src/prompts/system.md +409 -315
- package/src/prompts/system_copilot.md +88 -12
- package/src/skills/analysis-campaign/SKILL.md +239 -578
- package/src/skills/analysis-campaign/references/artifact-flow-examples.md +102 -0
- package/src/skills/analysis-campaign/references/boundary-cases.md +98 -0
- package/src/skills/analysis-campaign/references/campaign-checklist-template.md +39 -24
- package/src/skills/analysis-campaign/references/campaign-design.md +26 -10
- package/src/skills/analysis-campaign/references/campaign-plan-template.md +53 -54
- package/src/skills/analysis-campaign/references/operational-guidance.md +97 -0
- package/src/skills/analysis-campaign/references/writing-facing-slice-examples.md +10 -20
- package/src/skills/baseline/SKILL.md +183 -461
- package/src/skills/baseline/references/artifact-flow-examples.md +106 -0
- package/src/skills/baseline/references/artifact-payload-examples.md +1 -1
- package/src/skills/baseline/references/baseline-checklist-template.md +27 -35
- package/src/skills/baseline/references/baseline-plan-template.md +37 -76
- package/src/skills/baseline/references/boundary-cases.md +86 -0
- package/src/skills/baseline/references/codebase-audit-checklist.md +2 -6
- package/src/skills/baseline/references/comparability-contract.md +7 -12
- package/src/skills/baseline/references/operational-guidance.md +56 -0
- package/src/skills/baseline/references/route-selection.md +5 -25
- package/src/skills/decision/SKILL.md +113 -306
- package/src/skills/decision/references/checkpoint-memory-template.md +47 -0
- package/src/skills/decision/references/operational-guidance.md +94 -0
- package/src/skills/decision/references/research-route-criteria.md +7 -8
- package/src/skills/decision/references/strategic-decision-template.md +13 -26
- package/src/skills/experiment/SKILL.md +132 -670
- package/src/skills/experiment/references/execution-playbook.md +374 -0
- package/src/skills/experiment/references/main-experiment-checklist-template.md +26 -2
- package/src/skills/experiment/references/main-experiment-plan-template.md +28 -17
- package/src/skills/experiment/references/operational-guidance.md +108 -0
- package/src/skills/finalize/SKILL.md +62 -0
- package/src/skills/finalize/references/checkpoint-memory-template.md +49 -0
- package/src/skills/finalize/references/resume-packet-template.md +7 -0
- package/src/skills/idea/SKILL.md +228 -15
- package/src/skills/idea/references/controlled-brainstorming-playbook.md +78 -0
- package/src/skills/idea/references/current-board-packet-template.md +61 -0
- package/src/skills/idea/references/high-value-idea-sourcing.md +119 -0
- package/src/skills/idea/references/idea-generation-playbook.md +21 -0
- package/src/skills/idea/references/idea-thinking-flow.md +6 -0
- package/src/skills/idea/references/literature-survey-template.md +3 -0
- package/src/skills/idea/references/objective-contract-template.md +54 -0
- package/src/skills/idea/references/outline-seeding-example.md +56 -0
- package/src/skills/idea/references/pre-idea-draft-template.md +105 -0
- package/src/skills/idea/references/related-work-playbook.md +75 -2
- package/src/skills/idea/references/research-history-playbook.md +114 -0
- package/src/skills/idea/references/selection-gate.md +58 -6
- package/src/skills/intake-audit/SKILL.md +43 -2
- package/src/skills/intake-audit/references/state-audit-template.md +10 -0
- package/src/skills/nature-data/SKILL.md +128 -0
- package/src/skills/nature-data/UPSTREAM_LICENSE.txt +21 -0
- package/src/skills/nature-data/agents/openai.yaml +4 -0
- package/src/skills/nature-data/references/chinese-author-alignment.md +84 -0
- package/src/skills/nature-data/references/fair-metadata-checklist.md +105 -0
- package/src/skills/nature-data/references/policy-principles.md +103 -0
- package/src/skills/nature-data/references/repository-and-identifiers.md +96 -0
- package/src/skills/nature-data/references/source-basis.md +54 -0
- package/src/skills/nature-data/references/statement-patterns.md +153 -0
- package/src/skills/nature-figure/SKILL.md +197 -0
- package/src/skills/nature-figure/UPSTREAM_LICENSE.txt +21 -0
- package/src/skills/nature-figure/agents/openai.yaml +4 -0
- package/src/skills/nature-figure/evals/evals.json +37 -0
- package/src/skills/nature-figure/references/api.md +428 -0
- package/src/skills/nature-figure/references/backend-selection.md +100 -0
- package/src/skills/nature-figure/references/chart-types.md +281 -0
- package/src/skills/nature-figure/references/common-patterns.md +349 -0
- package/src/skills/nature-figure/references/design-theory.md +436 -0
- package/src/skills/nature-figure/references/figure-contract.md +93 -0
- package/src/skills/nature-figure/references/nature-2026-observations.md +112 -0
- package/src/skills/nature-figure/references/qa-contract.md +119 -0
- package/src/skills/nature-figure/references/r-template-index.md +66 -0
- package/src/skills/nature-figure/references/r-workflow.md +161 -0
- package/src/skills/nature-figure/references/tutorials.md +250 -0
- package/src/skills/nature-paper2ppt/SKILL.md +507 -0
- package/src/skills/nature-paper2ppt/UPSTREAM_LICENSE.txt +21 -0
- package/src/skills/nature-paper2ppt/agents/openai.yaml +4 -0
- package/src/skills/nature-polishing/SKILL.md +385 -0
- package/src/skills/nature-polishing/UPSTREAM_LICENSE.txt +21 -0
- package/src/skills/nature-polishing/agents/openai.yaml +4 -0
- package/src/skills/nature-polishing/references/phrasebank-playbook.md +162 -0
- package/src/skills/nature-polishing/references/section-moves.md +240 -0
- package/src/skills/nature-polishing/references/style-guardrails.md +94 -0
- package/src/skills/nature-polishing/references/writing-strategy.md +148 -0
- package/src/skills/optimize/SKILL.md +177 -1568
- package/src/skills/optimize/references/brief-shaping-playbook.md +95 -0
- package/src/skills/optimize/references/candidate-board-template.md +13 -0
- package/src/skills/optimize/references/candidate-ranking-template.md +51 -0
- package/src/skills/optimize/references/codegen-route-playbook.md +50 -0
- package/src/skills/optimize/references/debug-response-template.md +29 -0
- package/src/skills/optimize/references/frontier-review-template.md +32 -0
- package/src/skills/optimize/references/fusion-playbook.md +36 -0
- package/src/skills/optimize/references/method-brief-template.md +73 -0
- package/src/skills/optimize/references/operational-guidance.md +621 -0
- package/src/skills/optimize/references/optimization-memory-template.md +30 -0
- package/src/skills/optimize/references/optimize-checklist-template.md +18 -0
- package/src/skills/optimize/references/plateau-response-playbook.md +28 -0
- package/src/skills/optimize/references/prompt-patterns.md +49 -0
- package/src/skills/paper-outline/SKILL.md +227 -0
- package/src/skills/paper-outline/references/outline-patterns.md +87 -0
- package/src/skills/paper-plot/SKILL.md +79 -0
- package/src/skills/paper-plot/agents/openai.yaml +4 -0
- package/src/skills/paper-plot/references/bar_grouped_hatch.md +96 -0
- package/src/skills/paper-plot/references/bar_paired_delta.md +72 -0
- package/src/skills/paper-plot/references/line_confidence_band.md +75 -0
- package/src/skills/paper-plot/references/line_loss_with_inset.md +65 -0
- package/src/skills/paper-plot/references/line_training_curve.md +44 -0
- package/src/skills/paper-plot/references/radar_dual_series.md +59 -0
- package/src/skills/paper-plot/references/scatter_broken_axis.md +59 -0
- package/src/skills/paper-plot/references/scatter_tsne_cluster.md +72 -0
- package/src/skills/paper-plot/scripts/bar_memevolve.py +109 -0
- package/src/skills/paper-plot/scripts/bar_spice.py +166 -0
- package/src/skills/paper-plot/scripts/line_aime.py +94 -0
- package/src/skills/paper-plot/scripts/line_loss_inset.py +157 -0
- package/src/skills/paper-plot/scripts/line_selfdistill.py +168 -0
- package/src/skills/paper-plot/scripts/radar_dora.py +151 -0
- package/src/skills/paper-plot/scripts/scatter_break.py +169 -0
- package/src/skills/paper-plot/scripts/scatter_tsne.py +133 -0
- package/src/skills/rebuttal/SKILL.md +9 -0
- package/src/skills/references/tool-usage-by-stage.md +438 -0
- package/src/skills/review/SKILL.md +105 -7
- package/src/skills/science/PROVENANCE.md +44 -0
- package/src/skills/science/SKILL.md +137 -0
- package/src/skills/science/references/artifact-science-tool.md +110 -0
- package/src/skills/science/references/claim-type-discipline.md +56 -0
- package/src/skills/science/references/domain-index.md +422 -0
- package/src/skills/science/references/hpc-via-bash-exec.md +42 -0
- package/src/skills/science/references/package-check-playbook.md +64 -0
- package/src/skills/science/references/package-index.min.json +3616 -0
- package/src/skills/science/references/packages/abinit.md +80 -0
- package/src/skills/science/references/packages/acts.md +73 -0
- package/src/skills/science/references/packages/aiida-core.md +80 -0
- package/src/skills/science/references/packages/alamode.md +80 -0
- package/src/skills/science/references/packages/amuse.md +88 -0
- package/src/skills/science/references/packages/anndata.md +88 -0
- package/src/skills/science/references/packages/arbor.md +80 -0
- package/src/skills/science/references/packages/arc.md +73 -0
- package/src/skills/science/references/packages/astropy.md +88 -0
- package/src/skills/science/references/packages/astroquery.md +88 -0
- package/src/skills/science/references/packages/atomate2.md +80 -0
- package/src/skills/science/references/packages/atomsmltr.md +73 -0
- package/src/skills/science/references/packages/awkward.md +73 -0
- package/src/skills/science/references/packages/batman.md +88 -0
- package/src/skills/science/references/packages/biopython.md +88 -0
- package/src/skills/science/references/packages/bloqade.md +73 -0
- package/src/skills/science/references/packages/brian2.md +73 -0
- package/src/skills/science/references/packages/bullet3.md +73 -0
- package/src/skills/science/references/packages/calculix.md +80 -0
- package/src/skills/science/references/packages/cantera.md +73 -0
- package/src/skills/science/references/packages/cavity-md-ipi.md +80 -0
- package/src/skills/science/references/packages/ccdproc.md +88 -0
- package/src/skills/science/references/packages/celerite2.md +88 -0
- package/src/skills/science/references/packages/cellrank.md +73 -0
- package/src/skills/science/references/packages/cesm.md +80 -0
- package/src/skills/science/references/packages/chemicals.md +73 -0
- package/src/skills/science/references/packages/chempy.md +73 -0
- package/src/skills/science/references/packages/cirq.md +73 -0
- package/src/skills/science/references/packages/coffea.md +73 -0
- package/src/skills/science/references/packages/cp2k.md +88 -0
- package/src/skills/science/references/packages/custodian.md +80 -0
- package/src/skills/science/references/packages/dart.md +73 -0
- package/src/skills/science/references/packages/datamol.md +88 -0
- package/src/skills/science/references/packages/dd4hep.md +73 -0
- package/src/skills/science/references/packages/dealii.md +80 -0
- package/src/skills/science/references/packages/deepchem.md +88 -0
- package/src/skills/science/references/packages/delphes.md +73 -0
- package/src/skills/science/references/packages/devito.md +80 -0
- package/src/skills/science/references/packages/dftb.md +88 -0
- package/src/skills/science/references/packages/dftd4.md +88 -0
- package/src/skills/science/references/packages/dftk-jl.md +80 -0
- package/src/skills/science/references/packages/dolfinx.md +80 -0
- package/src/skills/science/references/packages/drake.md +73 -0
- package/src/skills/science/references/packages/dumux.md +73 -0
- package/src/skills/science/references/packages/elk.md +80 -0
- package/src/skills/science/references/packages/elmerfem.md +80 -0
- package/src/skills/science/references/packages/enzo-e.md +88 -0
- package/src/skills/science/references/packages/espresso.md +80 -0
- package/src/skills/science/references/packages/exoplanet.md +88 -0
- package/src/skills/science/references/packages/fairroot.md +73 -0
- package/src/skills/science/references/packages/fbpic.md +80 -0
- package/src/skills/science/references/packages/fdtdbath-meep.md +80 -0
- package/src/skills/science/references/packages/geant4.md +73 -0
- package/src/skills/science/references/packages/geosx.md +80 -0
- package/src/skills/science/references/packages/gprmax.md +80 -0
- package/src/skills/science/references/packages/gromacs.md +80 -0
- package/src/skills/science/references/packages/gwaslab.md +73 -0
- package/src/skills/science/references/packages/gz-sim.md +73 -0
- package/src/skills/science/references/packages/hail.md +88 -0
- package/src/skills/science/references/packages/hiphive.md +80 -0
- package/src/skills/science/references/packages/hoomd-blue.md +80 -0
- package/src/skills/science/references/packages/itensor.md +73 -0
- package/src/skills/science/references/packages/itensors-jl.md +73 -0
- package/src/skills/science/references/packages/jdftx.md +73 -0
- package/src/skills/science/references/packages/jobflow.md +80 -0
- package/src/skills/science/references/packages/kadanoffbaym-jl.md +73 -0
- package/src/skills/science/references/packages/kite.md +80 -0
- package/src/skills/science/references/packages/kratos.md +80 -0
- package/src/skills/science/references/packages/kwant.md +73 -0
- package/src/skills/science/references/packages/lammps.md +80 -0
- package/src/skills/science/references/packages/lightkurve.md +88 -0
- package/src/skills/science/references/packages/limix.md +73 -0
- package/src/skills/science/references/packages/maxwelllink.md +80 -0
- package/src/skills/science/references/packages/mcdc.md +73 -0
- package/src/skills/science/references/packages/meep.md +80 -0
- package/src/skills/science/references/packages/mfem.md +80 -0
- package/src/skills/science/references/packages/mitgcm.md +73 -0
- package/src/skills/science/references/packages/modflow6.md +73 -0
- package/src/skills/science/references/packages/molecool.md +73 -0
- package/src/skills/science/references/packages/mom6.md +73 -0
- package/src/skills/science/references/packages/moose.md +80 -0
- package/src/skills/science/references/packages/mpas-model.md +73 -0
- package/src/skills/science/references/packages/mujoco.md +73 -0
- package/src/skills/science/references/packages/mumax3.md +73 -0
- package/src/skills/science/references/packages/nekrs.md +80 -0
- package/src/skills/science/references/packages/nessi.md +73 -0
- package/src/skills/science/references/packages/nest-simulator.md +73 -0
- package/src/skills/science/references/packages/netket.md +73 -0
- package/src/skills/science/references/packages/neuron.md +73 -0
- package/src/skills/science/references/packages/nextflow.md +88 -0
- package/src/skills/science/references/packages/nwchem.md +88 -0
- package/src/skills/science/references/packages/openbabel.md +88 -0
- package/src/skills/science/references/packages/openems.md +80 -0
- package/src/skills/science/references/packages/openff-toolkit.md +88 -0
- package/src/skills/science/references/packages/openfoam-dev.md +80 -0
- package/src/skills/science/references/packages/openmc.md +73 -0
- package/src/skills/science/references/packages/openmm.md +80 -0
- package/src/skills/science/references/packages/openmoc.md +73 -0
- package/src/skills/science/references/packages/openmx.md +80 -0
- package/src/skills/science/references/packages/opensees.md +80 -0
- package/src/skills/science/references/packages/opensn.md +80 -0
- package/src/skills/science/references/packages/opm-simulators.md +73 -0
- package/src/skills/science/references/packages/oqupy.md +73 -0
- package/src/skills/science/references/packages/packmol.md +80 -0
- package/src/skills/science/references/packages/palabos.md +80 -0
- package/src/skills/science/references/packages/parflow.md +80 -0
- package/src/skills/science/references/packages/pennylane.md +88 -0
- package/src/skills/science/references/packages/perceval.md +73 -0
- package/src/skills/science/references/packages/phono3py.md +73 -0
- package/src/skills/science/references/packages/phonopy.md +73 -0
- package/src/skills/science/references/packages/photutils.md +88 -0
- package/src/skills/science/references/packages/picongpu.md +80 -0
- package/src/skills/science/references/packages/plink-ng.md +88 -0
- package/src/skills/science/references/packages/precice.md +73 -0
- package/src/skills/science/references/packages/psc.md +80 -0
- package/src/skills/science/references/packages/psi4.md +88 -0
- package/src/skills/science/references/packages/pybinding.md +73 -0
- package/src/skills/science/references/packages/pyfr.md +80 -0
- package/src/skills/science/references/packages/pyhf.md +73 -0
- package/src/skills/science/references/packages/pyiron_base.md +80 -0
- package/src/skills/science/references/packages/pylcp.md +73 -0
- package/src/skills/science/references/packages/pylith.md +80 -0
- package/src/skills/science/references/packages/pynbody.md +88 -0
- package/src/skills/science/references/packages/pysam.md +88 -0
- package/src/skills/science/references/packages/pyscf.md +88 -0
- package/src/skills/science/references/packages/q-e.md +73 -0
- package/src/skills/science/references/packages/qibo.md +73 -0
- package/src/skills/science/references/packages/qiskit.md +73 -0
- package/src/skills/science/references/packages/quantica-jl.md +73 -0
- package/src/skills/science/references/packages/quantumoptics-jl.md +73 -0
- package/src/skills/science/references/packages/quimb.md +73 -0
- package/src/skills/science/references/packages/qulacs.md +73 -0
- package/src/skills/science/references/packages/qutip.md +73 -0
- package/src/skills/science/references/packages/rdkit.md +88 -0
- package/src/skills/science/references/packages/rmg-py.md +73 -0
- package/src/skills/science/references/packages/root.md +73 -0
- package/src/skills/science/references/packages/scanpy.md +88 -0
- package/src/skills/science/references/packages/scikit-allel.md +88 -0
- package/src/skills/science/references/packages/scikit-bio.md +88 -0
- package/src/skills/science/references/packages/scqubits.md +73 -0
- package/src/skills/science/references/packages/scuff-em.md +80 -0
- package/src/skills/science/references/packages/scvi-tools.md +73 -0
- package/src/skills/science/references/packages/seissol.md +73 -0
- package/src/skills/science/references/packages/sfepy.md +80 -0
- package/src/skills/science/references/packages/sisl.md +73 -0
- package/src/skills/science/references/packages/smilei.md +80 -0
- package/src/skills/science/references/packages/snakemake.md +88 -0
- package/src/skills/science/references/packages/specfem3d-globe.md +80 -0
- package/src/skills/science/references/packages/specutils.md +88 -0
- package/src/skills/science/references/packages/spglib.md +80 -0
- package/src/skills/science/references/packages/squidpy.md +88 -0
- package/src/skills/science/references/packages/starry.md +88 -0
- package/src/skills/science/references/packages/strawberryfields.md +73 -0
- package/src/skills/science/references/packages/su2.md +80 -0
- package/src/skills/science/references/packages/sunny-jl.md +73 -0
- package/src/skills/science/references/packages/sw4.md +73 -0
- package/src/skills/science/references/packages/swift.md +88 -0
- package/src/skills/science/references/packages/tdnegf.md +73 -0
- package/src/skills/science/references/packages/tenpy.md +73 -0
- package/src/skills/science/references/packages/thermo.md +73 -0
- package/src/skills/science/references/packages/tkwant.md +73 -0
- package/src/skills/science/references/packages/tvb-root.md +73 -0
- package/src/skills/science/references/packages/uproot5.md +73 -0
- package/src/skills/science/references/packages/vampire.md +80 -0
- package/src/skills/science/references/packages/wannier_tools.md +73 -0
- package/src/skills/science/references/packages/warpx.md +80 -0
- package/src/skills/science/references/packages/wrf.md +73 -0
- package/src/skills/science/references/packages/xtb.md +88 -0
- package/src/skills/science/references/packages/yt.md +73 -0
- package/src/skills/science/references/science-task-brief-template.md +71 -0
- package/src/skills/scout/SKILL.md +83 -425
- package/src/skills/scout/references/literature-scout-template.md +5 -24
- package/src/skills/scout/references/operational-guidance.md +191 -0
- package/src/skills/scout/references/paper-triage-playbook.md +11 -35
- package/src/skills/write/SKILL.md +744 -1246
- package/src/skills/write/references/experiments_analysis_patterns.md +129 -0
- package/src/skills/write/references/oral_package_patterns.md +252 -0
- package/src/skills/write/references/oral_writing_principles.md +291 -0
- package/src/skills/write/references/section_rewrite_checklist.md +234 -0
- package/src/tui/dist/app/AppContainer.js +1314 -27
- package/src/tui/dist/components/Composer.js +26 -1
- package/src/tui/dist/components/ConfigScreen.js +2 -1
- package/src/tui/dist/components/InputPrompt.js +25 -9
- package/src/tui/dist/components/MainContent.js +18 -3
- package/src/tui/dist/components/QuestScreen.js +3 -2
- package/src/tui/dist/components/UtilityScreen.js +37 -0
- package/src/tui/dist/hooks/useSafeInput.js +10 -0
- package/src/tui/dist/index.js +13 -1
- package/src/tui/dist/layouts/DefaultAppLayout.js +11 -8
- package/src/tui/dist/lib/api.js +89 -1
- package/src/tui/package.json +1 -1
- package/src/ui/dist/assets/{AnalysisPlugin-BCKAfjba.js → AnalysisPlugin-CA94NGmI.js} +1 -1
- package/src/ui/dist/assets/CliPlugin-DHBzphZU.js +79 -0
- package/src/ui/dist/assets/CodeEditorPlugin-BOFwD2rn.js +2 -0
- package/src/ui/dist/assets/{CodeViewerPlugin-CbaFRrUU.js → CodeViewerPlugin-CqDpgjik.js} +4 -4
- package/src/ui/dist/assets/{DocViewerPlugin-DAjLVeQD.js → DocViewerPlugin-UDBgt8-4.js} +3 -3
- package/src/ui/dist/assets/GitCommitViewerPlugin-BmHtZ0bZ.js +6 -0
- package/src/ui/dist/assets/{GitDiffViewerPlugin-CQACjoAA.js → GitDiffViewerPlugin-CAxjNorQ.js} +2 -2
- package/src/ui/dist/assets/{GitSnapshotViewer-0r4nLPke.js → GitSnapshotViewer-CweA6VON.js} +2 -2
- package/src/ui/dist/assets/{ImageViewerPlugin-nBOmI2v_.js → ImageViewerPlugin-C8wHGvGN.js} +5 -5
- package/src/ui/dist/assets/LabPlugin-COyyLUol.js +32 -0
- package/src/ui/dist/assets/{LatexPlugin-ZwtV8pIp.js → LatexPlugin-BQjAaA5J.js} +4 -4
- package/src/ui/dist/assets/{MarkdownViewerPlugin-DKqVfKyW.js → MarkdownViewerPlugin-Dy1NE2dI.js} +3 -3
- package/src/ui/dist/assets/{MarketplacePlugin-BwxStZ9D.js → MarketplacePlugin-DMIZtEJ2.js} +2 -2
- package/src/ui/dist/assets/NotebookEditor-CFHMq_Qt.js +91 -0
- package/src/ui/dist/assets/{NotebookEditor-DB9N_T9q.js → NotebookEditor-WFyd8Ybt.js} +3 -3
- package/src/ui/dist/assets/{PdfLoader-eWBONbQP.js → PdfLoader-CLE5u5TS.js} +3 -3
- package/src/ui/dist/assets/{PdfMarkdownPlugin-D22YOZL3.js → PdfMarkdownPlugin-_iNK_H83.js} +1 -1
- package/src/ui/dist/assets/PdfViewerPlugin-DgWsbInT.js +22 -0
- package/src/ui/dist/assets/SearchPlugin-DrZmn5iw.js +11 -0
- package/src/ui/dist/assets/{TextViewerPlugin-C5xqeeUH.js → TextViewerPlugin-D1-T3aC7.js} +4 -4
- package/src/ui/dist/assets/branding/runner-claude.svg +107 -0
- package/src/ui/dist/assets/branding/runner-codex.svg +10 -0
- package/src/ui/dist/assets/branding/runner-kimi.svg +14 -0
- package/src/ui/dist/assets/branding/runner-opencode.svg +7 -0
- package/src/ui/dist/assets/cli-store-CoZ-x5Ip.js +1 -0
- package/src/ui/dist/assets/{code-WlFHE7z_.js → code-DbsmSd3Y.js} +1 -1
- package/src/ui/dist/assets/file-diff-panel-DsvyRz47.js +1 -0
- package/src/ui/dist/assets/{wrap-text-BC-Hltpd.js → file-jump-queue-DeQBikaw.js} +3 -3
- package/src/ui/dist/assets/{file-socket-CfQPKQKj.js → file-socket-DA5XIx88.js} +1 -1
- package/src/ui/dist/assets/fonts/ds-fonts.css +50 -4
- package/src/ui/dist/assets/images/deepxiv/register-guide.png +0 -0
- package/src/ui/dist/assets/index-39vY9LmZ.js +1 -0
- package/src/ui/dist/assets/{index-CwNu1aH4.js → index-BsO46tJA.js} +1 -1
- package/src/ui/dist/assets/index-CHzJ2xtB.js +3530 -0
- package/src/ui/dist/assets/index-DH-zxoZ3.css +33 -0
- package/src/ui/dist/assets/{plugin-notebook-HbW2K-1c.js → plugin-notebook-JRhysCqj.js} +2 -2
- package/src/ui/dist/assets/{project-sync-C9IdzdZW.js → project-sync-DPmWKmKD.js} +1 -1
- package/src/ui/dist/assets/{zoom-out-E_gaeAxL.js → zoom-out-DAukFWen.js} +3 -3
- package/src/ui/dist/index.html +3 -3
- package/src/skills/analysis-campaign/references/artifact-orchestration.md +0 -58
- package/src/skills/baseline/references/memory-playbook.md +0 -40
- package/src/skills/baseline/references/publishable-baseline-package.md +0 -30
- package/src/skills/write/references/outline-evidence-contract-example.md +0 -107
- package/src/skills/write/references/paper-experiment-matrix-template.md +0 -131
- package/src/skills/write/references/paper-section-playbook.md +0 -64
- package/src/skills/write/references/reviewer-first-writing.md +0 -64
- package/src/skills/write/references/revision-checklist.md +0 -70
- package/src/skills/write/references/section-contracts.md +0 -82
- package/src/skills/write/references/sentence-level-proofing.md +0 -49
- package/src/ui/dist/assets/AiManusChatView-Bv-Z8YpU.js +0 -204
- package/src/ui/dist/assets/CliPlugin-BCKcpc35.js +0 -109
- package/src/ui/dist/assets/CodeEditorPlugin-DbOfSJ8K.js +0 -2
- package/src/ui/dist/assets/GitCommitViewerPlugin-CIUqbUDO.js +0 -1
- package/src/ui/dist/assets/LabCopilotPanel-BHxOxF4z.js +0 -14
- package/src/ui/dist/assets/LabPlugin-BKoZGs95.js +0 -22
- package/src/ui/dist/assets/NotebookEditor-BEQhaQbt.js +0 -81
- package/src/ui/dist/assets/PdfViewerPlugin-c-RK9DLM.js +0 -17
- package/src/ui/dist/assets/SearchPlugin-CxF9ytAx.js +0 -16
- package/src/ui/dist/assets/VNCViewer-BoLGLnHz.js +0 -11
- package/src/ui/dist/assets/bot-DREQOxzP.js +0 -6
- package/src/ui/dist/assets/chevron-up-C9Qpx4DE.js +0 -6
- package/src/ui/dist/assets/file-content-BZMz3RYp.js +0 -1
- package/src/ui/dist/assets/file-diff-panel-CQhw0jS2.js +0 -1
- package/src/ui/dist/assets/file-jump-queue-DA-SdG__.js +0 -1
- package/src/ui/dist/assets/git-commit-horizontal-DxZ8DCZh.js +0 -6
- package/src/ui/dist/assets/image-Bgl4VIyx.js +0 -6
- package/src/ui/dist/assets/index-BpV6lusQ.css +0 -33
- package/src/ui/dist/assets/index-CBNVuWcP.js +0 -2496
- package/src/ui/dist/assets/index-DrUnlf6K.js +0 -1
- package/src/ui/dist/assets/index-NW-h8VzN.js +0 -1
- package/src/ui/dist/assets/pdf-effect-queue-J8OnM0jE.js +0 -6
- package/src/ui/dist/assets/popover-CLc0pPP8.js +0 -1
- package/src/ui/dist/assets/select-Cs2PmzwL.js +0 -11
- package/src/ui/dist/assets/sigma-ClKcHAXm.js +0 -6
- package/src/ui/dist/assets/trash-DwpbFr3w.js +0 -11
- package/src/ui/dist/assets/useCliAccess-NQ8m0Let.js +0 -1
- 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
|
+
}
|