PraisonAI 3.0.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (393) hide show
  1. praisonai/__init__.py +54 -0
  2. praisonai/__main__.py +15 -0
  3. praisonai/acp/__init__.py +54 -0
  4. praisonai/acp/config.py +159 -0
  5. praisonai/acp/server.py +587 -0
  6. praisonai/acp/session.py +219 -0
  7. praisonai/adapters/__init__.py +50 -0
  8. praisonai/adapters/readers.py +395 -0
  9. praisonai/adapters/rerankers.py +315 -0
  10. praisonai/adapters/retrievers.py +394 -0
  11. praisonai/adapters/vector_stores.py +409 -0
  12. praisonai/agent_scheduler.py +337 -0
  13. praisonai/agents_generator.py +903 -0
  14. praisonai/api/call.py +292 -0
  15. praisonai/auto.py +1197 -0
  16. praisonai/capabilities/__init__.py +275 -0
  17. praisonai/capabilities/a2a.py +140 -0
  18. praisonai/capabilities/assistants.py +283 -0
  19. praisonai/capabilities/audio.py +320 -0
  20. praisonai/capabilities/batches.py +469 -0
  21. praisonai/capabilities/completions.py +336 -0
  22. praisonai/capabilities/container_files.py +155 -0
  23. praisonai/capabilities/containers.py +93 -0
  24. praisonai/capabilities/embeddings.py +158 -0
  25. praisonai/capabilities/files.py +467 -0
  26. praisonai/capabilities/fine_tuning.py +293 -0
  27. praisonai/capabilities/guardrails.py +182 -0
  28. praisonai/capabilities/images.py +330 -0
  29. praisonai/capabilities/mcp.py +190 -0
  30. praisonai/capabilities/messages.py +270 -0
  31. praisonai/capabilities/moderations.py +154 -0
  32. praisonai/capabilities/ocr.py +217 -0
  33. praisonai/capabilities/passthrough.py +204 -0
  34. praisonai/capabilities/rag.py +207 -0
  35. praisonai/capabilities/realtime.py +160 -0
  36. praisonai/capabilities/rerank.py +165 -0
  37. praisonai/capabilities/responses.py +266 -0
  38. praisonai/capabilities/search.py +109 -0
  39. praisonai/capabilities/skills.py +133 -0
  40. praisonai/capabilities/vector_store_files.py +334 -0
  41. praisonai/capabilities/vector_stores.py +304 -0
  42. praisonai/capabilities/videos.py +141 -0
  43. praisonai/chainlit_ui.py +304 -0
  44. praisonai/chat/__init__.py +106 -0
  45. praisonai/chat/app.py +125 -0
  46. praisonai/cli/__init__.py +26 -0
  47. praisonai/cli/app.py +213 -0
  48. praisonai/cli/commands/__init__.py +75 -0
  49. praisonai/cli/commands/acp.py +70 -0
  50. praisonai/cli/commands/completion.py +333 -0
  51. praisonai/cli/commands/config.py +166 -0
  52. praisonai/cli/commands/debug.py +142 -0
  53. praisonai/cli/commands/diag.py +55 -0
  54. praisonai/cli/commands/doctor.py +166 -0
  55. praisonai/cli/commands/environment.py +179 -0
  56. praisonai/cli/commands/lsp.py +112 -0
  57. praisonai/cli/commands/mcp.py +210 -0
  58. praisonai/cli/commands/profile.py +457 -0
  59. praisonai/cli/commands/run.py +228 -0
  60. praisonai/cli/commands/schedule.py +150 -0
  61. praisonai/cli/commands/serve.py +97 -0
  62. praisonai/cli/commands/session.py +212 -0
  63. praisonai/cli/commands/traces.py +145 -0
  64. praisonai/cli/commands/version.py +101 -0
  65. praisonai/cli/configuration/__init__.py +18 -0
  66. praisonai/cli/configuration/loader.py +353 -0
  67. praisonai/cli/configuration/paths.py +114 -0
  68. praisonai/cli/configuration/schema.py +164 -0
  69. praisonai/cli/features/__init__.py +268 -0
  70. praisonai/cli/features/acp.py +236 -0
  71. praisonai/cli/features/action_orchestrator.py +546 -0
  72. praisonai/cli/features/agent_scheduler.py +773 -0
  73. praisonai/cli/features/agent_tools.py +474 -0
  74. praisonai/cli/features/agents.py +375 -0
  75. praisonai/cli/features/at_mentions.py +471 -0
  76. praisonai/cli/features/auto_memory.py +182 -0
  77. praisonai/cli/features/autonomy_mode.py +490 -0
  78. praisonai/cli/features/background.py +356 -0
  79. praisonai/cli/features/base.py +168 -0
  80. praisonai/cli/features/capabilities.py +1326 -0
  81. praisonai/cli/features/checkpoints.py +338 -0
  82. praisonai/cli/features/code_intelligence.py +652 -0
  83. praisonai/cli/features/compaction.py +294 -0
  84. praisonai/cli/features/compare.py +534 -0
  85. praisonai/cli/features/cost_tracker.py +514 -0
  86. praisonai/cli/features/debug.py +810 -0
  87. praisonai/cli/features/deploy.py +517 -0
  88. praisonai/cli/features/diag.py +289 -0
  89. praisonai/cli/features/doctor/__init__.py +63 -0
  90. praisonai/cli/features/doctor/checks/__init__.py +24 -0
  91. praisonai/cli/features/doctor/checks/acp_checks.py +240 -0
  92. praisonai/cli/features/doctor/checks/config_checks.py +366 -0
  93. praisonai/cli/features/doctor/checks/db_checks.py +366 -0
  94. praisonai/cli/features/doctor/checks/env_checks.py +543 -0
  95. praisonai/cli/features/doctor/checks/lsp_checks.py +199 -0
  96. praisonai/cli/features/doctor/checks/mcp_checks.py +349 -0
  97. praisonai/cli/features/doctor/checks/memory_checks.py +268 -0
  98. praisonai/cli/features/doctor/checks/network_checks.py +251 -0
  99. praisonai/cli/features/doctor/checks/obs_checks.py +328 -0
  100. praisonai/cli/features/doctor/checks/performance_checks.py +235 -0
  101. praisonai/cli/features/doctor/checks/permissions_checks.py +259 -0
  102. praisonai/cli/features/doctor/checks/selftest_checks.py +322 -0
  103. praisonai/cli/features/doctor/checks/serve_checks.py +426 -0
  104. praisonai/cli/features/doctor/checks/skills_checks.py +231 -0
  105. praisonai/cli/features/doctor/checks/tools_checks.py +371 -0
  106. praisonai/cli/features/doctor/engine.py +266 -0
  107. praisonai/cli/features/doctor/formatters.py +310 -0
  108. praisonai/cli/features/doctor/handler.py +397 -0
  109. praisonai/cli/features/doctor/models.py +264 -0
  110. praisonai/cli/features/doctor/registry.py +239 -0
  111. praisonai/cli/features/endpoints.py +1019 -0
  112. praisonai/cli/features/eval.py +560 -0
  113. praisonai/cli/features/external_agents.py +231 -0
  114. praisonai/cli/features/fast_context.py +410 -0
  115. praisonai/cli/features/flow_display.py +566 -0
  116. praisonai/cli/features/git_integration.py +651 -0
  117. praisonai/cli/features/guardrail.py +171 -0
  118. praisonai/cli/features/handoff.py +185 -0
  119. praisonai/cli/features/hooks.py +583 -0
  120. praisonai/cli/features/image.py +384 -0
  121. praisonai/cli/features/interactive_runtime.py +585 -0
  122. praisonai/cli/features/interactive_tools.py +380 -0
  123. praisonai/cli/features/interactive_tui.py +603 -0
  124. praisonai/cli/features/jobs.py +632 -0
  125. praisonai/cli/features/knowledge.py +531 -0
  126. praisonai/cli/features/lite.py +244 -0
  127. praisonai/cli/features/lsp_cli.py +225 -0
  128. praisonai/cli/features/mcp.py +169 -0
  129. praisonai/cli/features/message_queue.py +587 -0
  130. praisonai/cli/features/metrics.py +211 -0
  131. praisonai/cli/features/n8n.py +673 -0
  132. praisonai/cli/features/observability.py +293 -0
  133. praisonai/cli/features/ollama.py +361 -0
  134. praisonai/cli/features/output_style.py +273 -0
  135. praisonai/cli/features/package.py +631 -0
  136. praisonai/cli/features/performance.py +308 -0
  137. praisonai/cli/features/persistence.py +636 -0
  138. praisonai/cli/features/profile.py +226 -0
  139. praisonai/cli/features/profiler/__init__.py +81 -0
  140. praisonai/cli/features/profiler/core.py +558 -0
  141. praisonai/cli/features/profiler/optimizations.py +652 -0
  142. praisonai/cli/features/profiler/suite.py +386 -0
  143. praisonai/cli/features/profiling.py +350 -0
  144. praisonai/cli/features/queue/__init__.py +73 -0
  145. praisonai/cli/features/queue/manager.py +395 -0
  146. praisonai/cli/features/queue/models.py +286 -0
  147. praisonai/cli/features/queue/persistence.py +564 -0
  148. praisonai/cli/features/queue/scheduler.py +484 -0
  149. praisonai/cli/features/queue/worker.py +372 -0
  150. praisonai/cli/features/recipe.py +1723 -0
  151. praisonai/cli/features/recipes.py +449 -0
  152. praisonai/cli/features/registry.py +229 -0
  153. praisonai/cli/features/repo_map.py +860 -0
  154. praisonai/cli/features/router.py +466 -0
  155. praisonai/cli/features/sandbox_executor.py +515 -0
  156. praisonai/cli/features/serve.py +829 -0
  157. praisonai/cli/features/session.py +222 -0
  158. praisonai/cli/features/skills.py +856 -0
  159. praisonai/cli/features/slash_commands.py +650 -0
  160. praisonai/cli/features/telemetry.py +179 -0
  161. praisonai/cli/features/templates.py +1384 -0
  162. praisonai/cli/features/thinking.py +305 -0
  163. praisonai/cli/features/todo.py +334 -0
  164. praisonai/cli/features/tools.py +680 -0
  165. praisonai/cli/features/tui/__init__.py +83 -0
  166. praisonai/cli/features/tui/app.py +580 -0
  167. praisonai/cli/features/tui/cli.py +566 -0
  168. praisonai/cli/features/tui/debug.py +511 -0
  169. praisonai/cli/features/tui/events.py +99 -0
  170. praisonai/cli/features/tui/mock_provider.py +328 -0
  171. praisonai/cli/features/tui/orchestrator.py +652 -0
  172. praisonai/cli/features/tui/screens/__init__.py +50 -0
  173. praisonai/cli/features/tui/screens/main.py +245 -0
  174. praisonai/cli/features/tui/screens/queue.py +174 -0
  175. praisonai/cli/features/tui/screens/session.py +124 -0
  176. praisonai/cli/features/tui/screens/settings.py +148 -0
  177. praisonai/cli/features/tui/widgets/__init__.py +56 -0
  178. praisonai/cli/features/tui/widgets/chat.py +261 -0
  179. praisonai/cli/features/tui/widgets/composer.py +224 -0
  180. praisonai/cli/features/tui/widgets/queue_panel.py +200 -0
  181. praisonai/cli/features/tui/widgets/status.py +167 -0
  182. praisonai/cli/features/tui/widgets/tool_panel.py +248 -0
  183. praisonai/cli/features/workflow.py +720 -0
  184. praisonai/cli/legacy.py +236 -0
  185. praisonai/cli/main.py +5559 -0
  186. praisonai/cli/schedule_cli.py +54 -0
  187. praisonai/cli/state/__init__.py +31 -0
  188. praisonai/cli/state/identifiers.py +161 -0
  189. praisonai/cli/state/sessions.py +313 -0
  190. praisonai/code/__init__.py +93 -0
  191. praisonai/code/agent_tools.py +344 -0
  192. praisonai/code/diff/__init__.py +21 -0
  193. praisonai/code/diff/diff_strategy.py +432 -0
  194. praisonai/code/tools/__init__.py +27 -0
  195. praisonai/code/tools/apply_diff.py +221 -0
  196. praisonai/code/tools/execute_command.py +275 -0
  197. praisonai/code/tools/list_files.py +274 -0
  198. praisonai/code/tools/read_file.py +206 -0
  199. praisonai/code/tools/search_replace.py +248 -0
  200. praisonai/code/tools/write_file.py +217 -0
  201. praisonai/code/utils/__init__.py +46 -0
  202. praisonai/code/utils/file_utils.py +307 -0
  203. praisonai/code/utils/ignore_utils.py +308 -0
  204. praisonai/code/utils/text_utils.py +276 -0
  205. praisonai/db/__init__.py +64 -0
  206. praisonai/db/adapter.py +531 -0
  207. praisonai/deploy/__init__.py +62 -0
  208. praisonai/deploy/api.py +231 -0
  209. praisonai/deploy/docker.py +454 -0
  210. praisonai/deploy/doctor.py +367 -0
  211. praisonai/deploy/main.py +327 -0
  212. praisonai/deploy/models.py +179 -0
  213. praisonai/deploy/providers/__init__.py +33 -0
  214. praisonai/deploy/providers/aws.py +331 -0
  215. praisonai/deploy/providers/azure.py +358 -0
  216. praisonai/deploy/providers/base.py +101 -0
  217. praisonai/deploy/providers/gcp.py +314 -0
  218. praisonai/deploy/schema.py +208 -0
  219. praisonai/deploy.py +185 -0
  220. praisonai/endpoints/__init__.py +53 -0
  221. praisonai/endpoints/a2u_server.py +410 -0
  222. praisonai/endpoints/discovery.py +165 -0
  223. praisonai/endpoints/providers/__init__.py +28 -0
  224. praisonai/endpoints/providers/a2a.py +253 -0
  225. praisonai/endpoints/providers/a2u.py +208 -0
  226. praisonai/endpoints/providers/agents_api.py +171 -0
  227. praisonai/endpoints/providers/base.py +231 -0
  228. praisonai/endpoints/providers/mcp.py +263 -0
  229. praisonai/endpoints/providers/recipe.py +206 -0
  230. praisonai/endpoints/providers/tools_mcp.py +150 -0
  231. praisonai/endpoints/registry.py +131 -0
  232. praisonai/endpoints/server.py +161 -0
  233. praisonai/inbuilt_tools/__init__.py +24 -0
  234. praisonai/inbuilt_tools/autogen_tools.py +117 -0
  235. praisonai/inc/__init__.py +2 -0
  236. praisonai/inc/config.py +96 -0
  237. praisonai/inc/models.py +155 -0
  238. praisonai/integrations/__init__.py +56 -0
  239. praisonai/integrations/base.py +303 -0
  240. praisonai/integrations/claude_code.py +270 -0
  241. praisonai/integrations/codex_cli.py +255 -0
  242. praisonai/integrations/cursor_cli.py +195 -0
  243. praisonai/integrations/gemini_cli.py +222 -0
  244. praisonai/jobs/__init__.py +67 -0
  245. praisonai/jobs/executor.py +425 -0
  246. praisonai/jobs/models.py +230 -0
  247. praisonai/jobs/router.py +314 -0
  248. praisonai/jobs/server.py +186 -0
  249. praisonai/jobs/store.py +203 -0
  250. praisonai/llm/__init__.py +66 -0
  251. praisonai/llm/registry.py +382 -0
  252. praisonai/mcp_server/__init__.py +152 -0
  253. praisonai/mcp_server/adapters/__init__.py +74 -0
  254. praisonai/mcp_server/adapters/agents.py +128 -0
  255. praisonai/mcp_server/adapters/capabilities.py +168 -0
  256. praisonai/mcp_server/adapters/cli_tools.py +568 -0
  257. praisonai/mcp_server/adapters/extended_capabilities.py +462 -0
  258. praisonai/mcp_server/adapters/knowledge.py +93 -0
  259. praisonai/mcp_server/adapters/memory.py +104 -0
  260. praisonai/mcp_server/adapters/prompts.py +306 -0
  261. praisonai/mcp_server/adapters/resources.py +124 -0
  262. praisonai/mcp_server/adapters/tools_bridge.py +280 -0
  263. praisonai/mcp_server/auth/__init__.py +48 -0
  264. praisonai/mcp_server/auth/api_key.py +291 -0
  265. praisonai/mcp_server/auth/oauth.py +460 -0
  266. praisonai/mcp_server/auth/oidc.py +289 -0
  267. praisonai/mcp_server/auth/scopes.py +260 -0
  268. praisonai/mcp_server/cli.py +852 -0
  269. praisonai/mcp_server/elicitation.py +445 -0
  270. praisonai/mcp_server/icons.py +302 -0
  271. praisonai/mcp_server/recipe_adapter.py +573 -0
  272. praisonai/mcp_server/recipe_cli.py +824 -0
  273. praisonai/mcp_server/registry.py +703 -0
  274. praisonai/mcp_server/sampling.py +422 -0
  275. praisonai/mcp_server/server.py +490 -0
  276. praisonai/mcp_server/tasks.py +443 -0
  277. praisonai/mcp_server/transports/__init__.py +18 -0
  278. praisonai/mcp_server/transports/http_stream.py +376 -0
  279. praisonai/mcp_server/transports/stdio.py +132 -0
  280. praisonai/persistence/__init__.py +84 -0
  281. praisonai/persistence/config.py +238 -0
  282. praisonai/persistence/conversation/__init__.py +25 -0
  283. praisonai/persistence/conversation/async_mysql.py +427 -0
  284. praisonai/persistence/conversation/async_postgres.py +410 -0
  285. praisonai/persistence/conversation/async_sqlite.py +371 -0
  286. praisonai/persistence/conversation/base.py +151 -0
  287. praisonai/persistence/conversation/json_store.py +250 -0
  288. praisonai/persistence/conversation/mysql.py +387 -0
  289. praisonai/persistence/conversation/postgres.py +401 -0
  290. praisonai/persistence/conversation/singlestore.py +240 -0
  291. praisonai/persistence/conversation/sqlite.py +341 -0
  292. praisonai/persistence/conversation/supabase.py +203 -0
  293. praisonai/persistence/conversation/surrealdb.py +287 -0
  294. praisonai/persistence/factory.py +301 -0
  295. praisonai/persistence/hooks/__init__.py +18 -0
  296. praisonai/persistence/hooks/agent_hooks.py +297 -0
  297. praisonai/persistence/knowledge/__init__.py +26 -0
  298. praisonai/persistence/knowledge/base.py +144 -0
  299. praisonai/persistence/knowledge/cassandra.py +232 -0
  300. praisonai/persistence/knowledge/chroma.py +295 -0
  301. praisonai/persistence/knowledge/clickhouse.py +242 -0
  302. praisonai/persistence/knowledge/cosmosdb_vector.py +438 -0
  303. praisonai/persistence/knowledge/couchbase.py +286 -0
  304. praisonai/persistence/knowledge/lancedb.py +216 -0
  305. praisonai/persistence/knowledge/langchain_adapter.py +291 -0
  306. praisonai/persistence/knowledge/lightrag_adapter.py +212 -0
  307. praisonai/persistence/knowledge/llamaindex_adapter.py +256 -0
  308. praisonai/persistence/knowledge/milvus.py +277 -0
  309. praisonai/persistence/knowledge/mongodb_vector.py +306 -0
  310. praisonai/persistence/knowledge/pgvector.py +335 -0
  311. praisonai/persistence/knowledge/pinecone.py +253 -0
  312. praisonai/persistence/knowledge/qdrant.py +301 -0
  313. praisonai/persistence/knowledge/redis_vector.py +291 -0
  314. praisonai/persistence/knowledge/singlestore_vector.py +299 -0
  315. praisonai/persistence/knowledge/surrealdb_vector.py +309 -0
  316. praisonai/persistence/knowledge/upstash_vector.py +266 -0
  317. praisonai/persistence/knowledge/weaviate.py +223 -0
  318. praisonai/persistence/migrations/__init__.py +10 -0
  319. praisonai/persistence/migrations/manager.py +251 -0
  320. praisonai/persistence/orchestrator.py +406 -0
  321. praisonai/persistence/state/__init__.py +21 -0
  322. praisonai/persistence/state/async_mongodb.py +200 -0
  323. praisonai/persistence/state/base.py +107 -0
  324. praisonai/persistence/state/dynamodb.py +226 -0
  325. praisonai/persistence/state/firestore.py +175 -0
  326. praisonai/persistence/state/gcs.py +155 -0
  327. praisonai/persistence/state/memory.py +245 -0
  328. praisonai/persistence/state/mongodb.py +158 -0
  329. praisonai/persistence/state/redis.py +190 -0
  330. praisonai/persistence/state/upstash.py +144 -0
  331. praisonai/persistence/tests/__init__.py +3 -0
  332. praisonai/persistence/tests/test_all_backends.py +633 -0
  333. praisonai/profiler.py +1214 -0
  334. praisonai/recipe/__init__.py +134 -0
  335. praisonai/recipe/bridge.py +278 -0
  336. praisonai/recipe/core.py +893 -0
  337. praisonai/recipe/exceptions.py +54 -0
  338. praisonai/recipe/history.py +402 -0
  339. praisonai/recipe/models.py +266 -0
  340. praisonai/recipe/operations.py +440 -0
  341. praisonai/recipe/policy.py +422 -0
  342. praisonai/recipe/registry.py +849 -0
  343. praisonai/recipe/runtime.py +214 -0
  344. praisonai/recipe/security.py +711 -0
  345. praisonai/recipe/serve.py +859 -0
  346. praisonai/recipe/server.py +613 -0
  347. praisonai/scheduler/__init__.py +45 -0
  348. praisonai/scheduler/agent_scheduler.py +552 -0
  349. praisonai/scheduler/base.py +124 -0
  350. praisonai/scheduler/daemon_manager.py +225 -0
  351. praisonai/scheduler/state_manager.py +155 -0
  352. praisonai/scheduler/yaml_loader.py +193 -0
  353. praisonai/scheduler.py +194 -0
  354. praisonai/setup/__init__.py +1 -0
  355. praisonai/setup/build.py +21 -0
  356. praisonai/setup/post_install.py +23 -0
  357. praisonai/setup/setup_conda_env.py +25 -0
  358. praisonai/setup.py +16 -0
  359. praisonai/templates/__init__.py +116 -0
  360. praisonai/templates/cache.py +364 -0
  361. praisonai/templates/dependency_checker.py +358 -0
  362. praisonai/templates/discovery.py +391 -0
  363. praisonai/templates/loader.py +564 -0
  364. praisonai/templates/registry.py +511 -0
  365. praisonai/templates/resolver.py +206 -0
  366. praisonai/templates/security.py +327 -0
  367. praisonai/templates/tool_override.py +498 -0
  368. praisonai/templates/tools_doctor.py +256 -0
  369. praisonai/test.py +105 -0
  370. praisonai/train.py +562 -0
  371. praisonai/train_vision.py +306 -0
  372. praisonai/ui/agents.py +824 -0
  373. praisonai/ui/callbacks.py +57 -0
  374. praisonai/ui/chainlit_compat.py +246 -0
  375. praisonai/ui/chat.py +532 -0
  376. praisonai/ui/code.py +717 -0
  377. praisonai/ui/colab.py +474 -0
  378. praisonai/ui/colab_chainlit.py +81 -0
  379. praisonai/ui/components/aicoder.py +284 -0
  380. praisonai/ui/context.py +283 -0
  381. praisonai/ui/database_config.py +56 -0
  382. praisonai/ui/db.py +294 -0
  383. praisonai/ui/realtime.py +488 -0
  384. praisonai/ui/realtimeclient/__init__.py +756 -0
  385. praisonai/ui/realtimeclient/tools.py +242 -0
  386. praisonai/ui/sql_alchemy.py +710 -0
  387. praisonai/upload_vision.py +140 -0
  388. praisonai/version.py +1 -0
  389. praisonai-3.0.0.dist-info/METADATA +3493 -0
  390. praisonai-3.0.0.dist-info/RECORD +393 -0
  391. praisonai-3.0.0.dist-info/WHEEL +5 -0
  392. praisonai-3.0.0.dist-info/entry_points.txt +4 -0
  393. praisonai-3.0.0.dist-info/top_level.txt +1 -0
@@ -0,0 +1,543 @@
1
+ """
2
+ Environment checks for the Doctor CLI module.
3
+
4
+ Validates Python version, packages, API keys, and environment variables.
5
+ """
6
+
7
+ import os
8
+ import platform
9
+ import shutil
10
+ import sys
11
+ from typing import List, Optional
12
+
13
+ from ..models import (
14
+ CheckCategory,
15
+ CheckResult,
16
+ CheckStatus,
17
+ CheckSeverity,
18
+ DoctorConfig,
19
+ )
20
+ from ..registry import register_check
21
+
22
+
23
+ @register_check(
24
+ id="python_version",
25
+ title="Python Version",
26
+ description="Check Python version is 3.9+",
27
+ category=CheckCategory.ENVIRONMENT,
28
+ severity=CheckSeverity.CRITICAL,
29
+ )
30
+ def check_python_version(config: DoctorConfig) -> CheckResult:
31
+ """Check Python version is 3.9+."""
32
+ version = sys.version_info
33
+ version_str = f"{version.major}.{version.minor}.{version.micro}"
34
+
35
+ if version.major >= 3 and version.minor >= 9:
36
+ return CheckResult(
37
+ id="python_version",
38
+ title="Python Version",
39
+ category=CheckCategory.ENVIRONMENT,
40
+ status=CheckStatus.PASS,
41
+ message=f"Python {version_str} (>= 3.9 required)",
42
+ metadata={"version": version_str, "executable": sys.executable},
43
+ )
44
+ else:
45
+ return CheckResult(
46
+ id="python_version",
47
+ title="Python Version",
48
+ category=CheckCategory.ENVIRONMENT,
49
+ status=CheckStatus.FAIL,
50
+ message=f"Python {version_str} found, but 3.9+ required",
51
+ remediation="Upgrade Python to version 3.9 or higher",
52
+ severity=CheckSeverity.CRITICAL,
53
+ )
54
+
55
+
56
+ @register_check(
57
+ id="praisonai_package",
58
+ title="PraisonAI Package",
59
+ description="Check praisonai package is installed",
60
+ category=CheckCategory.ENVIRONMENT,
61
+ severity=CheckSeverity.CRITICAL,
62
+ )
63
+ def check_praisonai_package(config: DoctorConfig) -> CheckResult:
64
+ """Check praisonai package is installed."""
65
+ try:
66
+ from praisonai.version import __version__
67
+ return CheckResult(
68
+ id="praisonai_package",
69
+ title="PraisonAI Package",
70
+ category=CheckCategory.ENVIRONMENT,
71
+ status=CheckStatus.PASS,
72
+ message=f"praisonai {__version__} installed",
73
+ metadata={"version": __version__},
74
+ )
75
+ except ImportError as e:
76
+ return CheckResult(
77
+ id="praisonai_package",
78
+ title="PraisonAI Package",
79
+ category=CheckCategory.ENVIRONMENT,
80
+ status=CheckStatus.FAIL,
81
+ message="praisonai package not found",
82
+ details=str(e),
83
+ remediation="Install with: pip install praisonai",
84
+ severity=CheckSeverity.CRITICAL,
85
+ )
86
+
87
+
88
+ @register_check(
89
+ id="praisonaiagents_package",
90
+ title="PraisonAI Agents Package",
91
+ description="Check praisonaiagents package is installed",
92
+ category=CheckCategory.ENVIRONMENT,
93
+ severity=CheckSeverity.CRITICAL,
94
+ )
95
+ def check_praisonaiagents_package(config: DoctorConfig) -> CheckResult:
96
+ """Check praisonaiagents package is installed."""
97
+ try:
98
+ import praisonaiagents
99
+ version = getattr(praisonaiagents, "__version__", "unknown")
100
+ return CheckResult(
101
+ id="praisonaiagents_package",
102
+ title="PraisonAI Agents Package",
103
+ category=CheckCategory.ENVIRONMENT,
104
+ status=CheckStatus.PASS,
105
+ message=f"praisonaiagents {version} installed",
106
+ metadata={"version": version},
107
+ )
108
+ except ImportError as e:
109
+ return CheckResult(
110
+ id="praisonaiagents_package",
111
+ title="PraisonAI Agents Package",
112
+ category=CheckCategory.ENVIRONMENT,
113
+ status=CheckStatus.FAIL,
114
+ message="praisonaiagents package not found",
115
+ details=str(e),
116
+ remediation="Install with: pip install praisonaiagents",
117
+ severity=CheckSeverity.CRITICAL,
118
+ )
119
+
120
+
121
+ @register_check(
122
+ id="openai_api_key",
123
+ title="OpenAI API Key",
124
+ description="Check OPENAI_API_KEY is configured",
125
+ category=CheckCategory.ENVIRONMENT,
126
+ severity=CheckSeverity.HIGH,
127
+ )
128
+ def check_openai_api_key(config: DoctorConfig) -> CheckResult:
129
+ """Check OpenAI API key is configured."""
130
+ key = os.environ.get("OPENAI_API_KEY", "")
131
+
132
+ if key:
133
+ # Mask the key for display
134
+ if config.show_keys and len(key) > 12:
135
+ masked = f"{key[:4]}...{key[-4:]}"
136
+ else:
137
+ masked = "***configured***"
138
+
139
+ # Basic validation
140
+ if key.startswith("sk-") or key.startswith("sk-proj-"):
141
+ return CheckResult(
142
+ id="openai_api_key",
143
+ title="OpenAI API Key",
144
+ category=CheckCategory.ENVIRONMENT,
145
+ status=CheckStatus.PASS,
146
+ message=f"OPENAI_API_KEY configured ({masked})",
147
+ metadata={"masked_key": masked},
148
+ )
149
+ else:
150
+ return CheckResult(
151
+ id="openai_api_key",
152
+ title="OpenAI API Key",
153
+ category=CheckCategory.ENVIRONMENT,
154
+ status=CheckStatus.WARN,
155
+ message=f"OPENAI_API_KEY set but format unexpected ({masked})",
156
+ remediation="Verify the API key format is correct",
157
+ )
158
+ else:
159
+ # Check for alternative providers
160
+ alternatives = []
161
+ if os.environ.get("ANTHROPIC_API_KEY"):
162
+ alternatives.append("ANTHROPIC_API_KEY")
163
+ if os.environ.get("GOOGLE_API_KEY") or os.environ.get("GEMINI_API_KEY"):
164
+ alternatives.append("GOOGLE/GEMINI_API_KEY")
165
+ if os.environ.get("OLLAMA_HOST") or shutil.which("ollama"):
166
+ alternatives.append("Ollama")
167
+
168
+ if alternatives:
169
+ return CheckResult(
170
+ id="openai_api_key",
171
+ title="OpenAI API Key",
172
+ category=CheckCategory.ENVIRONMENT,
173
+ status=CheckStatus.WARN,
174
+ message=f"OPENAI_API_KEY not set, but alternatives found: {', '.join(alternatives)}",
175
+ details="OpenAI is the default provider. Other providers require explicit configuration.",
176
+ )
177
+ else:
178
+ return CheckResult(
179
+ id="openai_api_key",
180
+ title="OpenAI API Key",
181
+ category=CheckCategory.ENVIRONMENT,
182
+ status=CheckStatus.FAIL,
183
+ message="OPENAI_API_KEY not configured and no alternative providers found",
184
+ remediation="Set OPENAI_API_KEY environment variable or configure an alternative provider",
185
+ severity=CheckSeverity.HIGH,
186
+ )
187
+
188
+
189
+ @register_check(
190
+ id="anthropic_api_key",
191
+ title="Anthropic API Key",
192
+ description="Check ANTHROPIC_API_KEY is configured",
193
+ category=CheckCategory.ENVIRONMENT,
194
+ severity=CheckSeverity.LOW,
195
+ )
196
+ def check_anthropic_api_key(config: DoctorConfig) -> CheckResult:
197
+ """Check Anthropic API key is configured."""
198
+ key = os.environ.get("ANTHROPIC_API_KEY", "")
199
+
200
+ if key:
201
+ if config.show_keys and len(key) > 12:
202
+ masked = f"{key[:4]}...{key[-4:]}"
203
+ else:
204
+ masked = "***configured***"
205
+
206
+ return CheckResult(
207
+ id="anthropic_api_key",
208
+ title="Anthropic API Key",
209
+ category=CheckCategory.ENVIRONMENT,
210
+ status=CheckStatus.PASS,
211
+ message=f"ANTHROPIC_API_KEY configured ({masked})",
212
+ )
213
+ else:
214
+ return CheckResult(
215
+ id="anthropic_api_key",
216
+ title="Anthropic API Key",
217
+ category=CheckCategory.ENVIRONMENT,
218
+ status=CheckStatus.SKIP,
219
+ message="ANTHROPIC_API_KEY not set (optional)",
220
+ )
221
+
222
+
223
+ @register_check(
224
+ id="google_api_key",
225
+ title="Google/Gemini API Key",
226
+ description="Check GOOGLE_API_KEY or GEMINI_API_KEY is configured",
227
+ category=CheckCategory.ENVIRONMENT,
228
+ severity=CheckSeverity.LOW,
229
+ )
230
+ def check_google_api_key(config: DoctorConfig) -> CheckResult:
231
+ """Check Google/Gemini API key is configured."""
232
+ key = os.environ.get("GOOGLE_API_KEY") or os.environ.get("GEMINI_API_KEY", "")
233
+
234
+ if key:
235
+ if config.show_keys and len(key) > 12:
236
+ masked = f"{key[:4]}...{key[-4:]}"
237
+ else:
238
+ masked = "***configured***"
239
+
240
+ return CheckResult(
241
+ id="google_api_key",
242
+ title="Google/Gemini API Key",
243
+ category=CheckCategory.ENVIRONMENT,
244
+ status=CheckStatus.PASS,
245
+ message=f"Google/Gemini API key configured ({masked})",
246
+ )
247
+ else:
248
+ return CheckResult(
249
+ id="google_api_key",
250
+ title="Google/Gemini API Key",
251
+ category=CheckCategory.ENVIRONMENT,
252
+ status=CheckStatus.SKIP,
253
+ message="GOOGLE_API_KEY/GEMINI_API_KEY not set (optional)",
254
+ )
255
+
256
+
257
+ @register_check(
258
+ id="os_info",
259
+ title="Operating System",
260
+ description="Check operating system information",
261
+ category=CheckCategory.ENVIRONMENT,
262
+ severity=CheckSeverity.INFO,
263
+ )
264
+ def check_os_info(config: DoctorConfig) -> CheckResult:
265
+ """Check operating system information."""
266
+ os_name = platform.system()
267
+ os_version = platform.release()
268
+ arch = platform.machine()
269
+
270
+ return CheckResult(
271
+ id="os_info",
272
+ title="Operating System",
273
+ category=CheckCategory.ENVIRONMENT,
274
+ status=CheckStatus.PASS,
275
+ message=f"{os_name} {os_version} ({arch})",
276
+ metadata={
277
+ "os_name": os_name,
278
+ "os_version": os_version,
279
+ "architecture": arch,
280
+ },
281
+ )
282
+
283
+
284
+ @register_check(
285
+ id="virtual_env",
286
+ title="Virtual Environment",
287
+ description="Check if running in a virtual environment",
288
+ category=CheckCategory.ENVIRONMENT,
289
+ severity=CheckSeverity.INFO,
290
+ )
291
+ def check_virtual_env(config: DoctorConfig) -> CheckResult:
292
+ """Check if running in a virtual environment."""
293
+ venv = os.environ.get("VIRTUAL_ENV")
294
+ conda_env = os.environ.get("CONDA_DEFAULT_ENV")
295
+
296
+ if venv:
297
+ return CheckResult(
298
+ id="virtual_env",
299
+ title="Virtual Environment",
300
+ category=CheckCategory.ENVIRONMENT,
301
+ status=CheckStatus.PASS,
302
+ message=f"Running in venv: {os.path.basename(venv)}",
303
+ metadata={"venv_path": venv},
304
+ )
305
+ elif conda_env:
306
+ return CheckResult(
307
+ id="virtual_env",
308
+ title="Virtual Environment",
309
+ category=CheckCategory.ENVIRONMENT,
310
+ status=CheckStatus.PASS,
311
+ message=f"Running in conda env: {conda_env}",
312
+ metadata={"conda_env": conda_env},
313
+ )
314
+ else:
315
+ return CheckResult(
316
+ id="virtual_env",
317
+ title="Virtual Environment",
318
+ category=CheckCategory.ENVIRONMENT,
319
+ status=CheckStatus.WARN,
320
+ message="Not running in a virtual environment",
321
+ remediation="Consider using a virtual environment for isolation",
322
+ )
323
+
324
+
325
+ @register_check(
326
+ id="git_available",
327
+ title="Git",
328
+ description="Check if git is available",
329
+ category=CheckCategory.ENVIRONMENT,
330
+ severity=CheckSeverity.LOW,
331
+ )
332
+ def check_git_available(config: DoctorConfig) -> CheckResult:
333
+ """Check if git is available."""
334
+ git_path = shutil.which("git")
335
+
336
+ if git_path:
337
+ import subprocess
338
+ try:
339
+ result = subprocess.run(
340
+ ["git", "--version"],
341
+ capture_output=True,
342
+ text=True,
343
+ timeout=5,
344
+ )
345
+ version = result.stdout.strip() if result.returncode == 0 else "unknown"
346
+ return CheckResult(
347
+ id="git_available",
348
+ title="Git",
349
+ category=CheckCategory.ENVIRONMENT,
350
+ status=CheckStatus.PASS,
351
+ message=f"Git available: {version}",
352
+ metadata={"path": git_path},
353
+ )
354
+ except Exception:
355
+ return CheckResult(
356
+ id="git_available",
357
+ title="Git",
358
+ category=CheckCategory.ENVIRONMENT,
359
+ status=CheckStatus.PASS,
360
+ message=f"Git available at {git_path}",
361
+ )
362
+ else:
363
+ return CheckResult(
364
+ id="git_available",
365
+ title="Git",
366
+ category=CheckCategory.ENVIRONMENT,
367
+ status=CheckStatus.WARN,
368
+ message="Git not found in PATH",
369
+ remediation="Install git for version control features",
370
+ )
371
+
372
+
373
+ @register_check(
374
+ id="docker_available",
375
+ title="Docker",
376
+ description="Check if Docker is available",
377
+ category=CheckCategory.ENVIRONMENT,
378
+ severity=CheckSeverity.LOW,
379
+ )
380
+ def check_docker_available(config: DoctorConfig) -> CheckResult:
381
+ """Check if Docker is available."""
382
+ docker_path = shutil.which("docker")
383
+
384
+ if docker_path:
385
+ import subprocess
386
+ try:
387
+ result = subprocess.run(
388
+ ["docker", "--version"],
389
+ capture_output=True,
390
+ text=True,
391
+ timeout=5,
392
+ )
393
+ if result.returncode == 0:
394
+ version = result.stdout.strip()
395
+ return CheckResult(
396
+ id="docker_available",
397
+ title="Docker",
398
+ category=CheckCategory.ENVIRONMENT,
399
+ status=CheckStatus.PASS,
400
+ message=f"Docker available: {version}",
401
+ metadata={"path": docker_path},
402
+ )
403
+ except Exception:
404
+ pass
405
+
406
+ return CheckResult(
407
+ id="docker_available",
408
+ title="Docker",
409
+ category=CheckCategory.ENVIRONMENT,
410
+ status=CheckStatus.WARN,
411
+ message="Docker found but not responding",
412
+ remediation="Ensure Docker daemon is running",
413
+ )
414
+ else:
415
+ return CheckResult(
416
+ id="docker_available",
417
+ title="Docker",
418
+ category=CheckCategory.ENVIRONMENT,
419
+ status=CheckStatus.SKIP,
420
+ message="Docker not found (optional)",
421
+ )
422
+
423
+
424
+ @register_check(
425
+ id="npx_available",
426
+ title="npx (Node.js)",
427
+ description="Check if npx is available for MCP servers",
428
+ category=CheckCategory.ENVIRONMENT,
429
+ severity=CheckSeverity.LOW,
430
+ )
431
+ def check_npx_available(config: DoctorConfig) -> CheckResult:
432
+ """Check if npx is available for MCP servers."""
433
+ npx_path = shutil.which("npx")
434
+
435
+ if npx_path:
436
+ return CheckResult(
437
+ id="npx_available",
438
+ title="npx (Node.js)",
439
+ category=CheckCategory.ENVIRONMENT,
440
+ status=CheckStatus.PASS,
441
+ message=f"npx available at {npx_path}",
442
+ metadata={"path": npx_path},
443
+ )
444
+ else:
445
+ return CheckResult(
446
+ id="npx_available",
447
+ title="npx (Node.js)",
448
+ category=CheckCategory.ENVIRONMENT,
449
+ status=CheckStatus.WARN,
450
+ message="npx not found (required for some MCP servers)",
451
+ remediation="Install Node.js to use MCP servers that require npx",
452
+ )
453
+
454
+
455
+ @register_check(
456
+ id="optional_deps",
457
+ title="Optional Dependencies",
458
+ description="Check availability of optional dependencies",
459
+ category=CheckCategory.ENVIRONMENT,
460
+ severity=CheckSeverity.INFO,
461
+ requires_deep=True,
462
+ )
463
+ def check_optional_deps(config: DoctorConfig) -> CheckResult:
464
+ """Check availability of optional dependencies."""
465
+ optional_packages = [
466
+ ("chromadb", "Knowledge/RAG features"),
467
+ ("mem0ai", "Memory features"),
468
+ ("litellm", "Multi-provider LLM support"),
469
+ ("chainlit", "Chat UI"),
470
+ ("gradio", "Gradio UI"),
471
+ ("crawl4ai", "Web crawling"),
472
+ ("tavily", "Tavily search"),
473
+ ("duckduckgo_search", "DuckDuckGo search"),
474
+ ]
475
+
476
+ available = []
477
+ missing = []
478
+
479
+ for package, description in optional_packages:
480
+ try:
481
+ __import__(package)
482
+ available.append(package)
483
+ except ImportError:
484
+ missing.append(f"{package} ({description})")
485
+
486
+ if missing:
487
+ return CheckResult(
488
+ id="optional_deps",
489
+ title="Optional Dependencies",
490
+ category=CheckCategory.ENVIRONMENT,
491
+ status=CheckStatus.PASS,
492
+ message=f"{len(available)} optional packages available, {len(missing)} not installed",
493
+ details=f"Missing: {', '.join(missing[:5])}{'...' if len(missing) > 5 else ''}",
494
+ metadata={"available": available, "missing": missing},
495
+ )
496
+ else:
497
+ return CheckResult(
498
+ id="optional_deps",
499
+ title="Optional Dependencies",
500
+ category=CheckCategory.ENVIRONMENT,
501
+ status=CheckStatus.PASS,
502
+ message=f"All {len(available)} optional packages available",
503
+ metadata={"available": available},
504
+ )
505
+
506
+
507
+ @register_check(
508
+ id="model_env_vars",
509
+ title="Model Configuration",
510
+ description="Check model-related environment variables",
511
+ category=CheckCategory.ENVIRONMENT,
512
+ severity=CheckSeverity.INFO,
513
+ )
514
+ def check_model_env_vars(config: DoctorConfig) -> CheckResult:
515
+ """Check model-related environment variables."""
516
+ model_vars = {
517
+ "MODEL_NAME": os.environ.get("MODEL_NAME"),
518
+ "OPENAI_MODEL_NAME": os.environ.get("OPENAI_MODEL_NAME"),
519
+ "OPENAI_BASE_URL": os.environ.get("OPENAI_BASE_URL"),
520
+ "OPENAI_API_BASE": os.environ.get("OPENAI_API_BASE"),
521
+ }
522
+
523
+ configured = {k: v for k, v in model_vars.items() if v}
524
+
525
+ if configured:
526
+ details = ", ".join(f"{k}={v}" for k, v in configured.items())
527
+ return CheckResult(
528
+ id="model_env_vars",
529
+ title="Model Configuration",
530
+ category=CheckCategory.ENVIRONMENT,
531
+ status=CheckStatus.PASS,
532
+ message=f"Model config: {details}",
533
+ metadata=configured,
534
+ )
535
+ else:
536
+ return CheckResult(
537
+ id="model_env_vars",
538
+ title="Model Configuration",
539
+ category=CheckCategory.ENVIRONMENT,
540
+ status=CheckStatus.PASS,
541
+ message="Using default model configuration (gpt-4o-mini)",
542
+ metadata={"default_model": "gpt-4o-mini"},
543
+ )