wandb 0.21.2__py3-none-macosx_12_0_arm64.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 (904) hide show
  1. package_readme.md +97 -0
  2. wandb/__init__.py +248 -0
  3. wandb/__init__.pyi +1230 -0
  4. wandb/__main__.py +3 -0
  5. wandb/_iterutils.py +65 -0
  6. wandb/_pydantic/__init__.py +30 -0
  7. wandb/_pydantic/base.py +128 -0
  8. wandb/_pydantic/utils.py +80 -0
  9. wandb/_pydantic/v1_compat.py +284 -0
  10. wandb/agents/__init__.py +0 -0
  11. wandb/agents/pyagent.py +386 -0
  12. wandb/analytics/__init__.py +3 -0
  13. wandb/analytics/sentry.py +267 -0
  14. wandb/apis/__init__.py +48 -0
  15. wandb/apis/attrs.py +50 -0
  16. wandb/apis/importers/__init__.py +1 -0
  17. wandb/apis/importers/internals/internal.py +382 -0
  18. wandb/apis/importers/internals/protocols.py +103 -0
  19. wandb/apis/importers/internals/util.py +78 -0
  20. wandb/apis/importers/mlflow.py +254 -0
  21. wandb/apis/importers/validation.py +108 -0
  22. wandb/apis/importers/wandb.py +1608 -0
  23. wandb/apis/internal.py +239 -0
  24. wandb/apis/normalize.py +81 -0
  25. wandb/apis/paginator.py +138 -0
  26. wandb/apis/public/__init__.py +35 -0
  27. wandb/apis/public/api.py +2449 -0
  28. wandb/apis/public/artifacts.py +1046 -0
  29. wandb/apis/public/automations.py +85 -0
  30. wandb/apis/public/const.py +4 -0
  31. wandb/apis/public/files.py +402 -0
  32. wandb/apis/public/history.py +201 -0
  33. wandb/apis/public/integrations.py +203 -0
  34. wandb/apis/public/jobs.py +742 -0
  35. wandb/apis/public/projects.py +276 -0
  36. wandb/apis/public/query_generator.py +176 -0
  37. wandb/apis/public/registries/__init__.py +0 -0
  38. wandb/apis/public/registries/_freezable_list.py +179 -0
  39. wandb/apis/public/registries/_utils.py +138 -0
  40. wandb/apis/public/registries/registries_search.py +347 -0
  41. wandb/apis/public/registries/registry.py +358 -0
  42. wandb/apis/public/reports.py +595 -0
  43. wandb/apis/public/runs.py +1216 -0
  44. wandb/apis/public/sweeps.py +440 -0
  45. wandb/apis/public/teams.py +235 -0
  46. wandb/apis/public/users.py +177 -0
  47. wandb/apis/public/utils.py +210 -0
  48. wandb/apis/reports/__init__.py +1 -0
  49. wandb/apis/reports/v1/__init__.py +8 -0
  50. wandb/apis/reports/v2/__init__.py +8 -0
  51. wandb/apis/workspaces/__init__.py +8 -0
  52. wandb/automations/__init__.py +73 -0
  53. wandb/automations/_filters/__init__.py +40 -0
  54. wandb/automations/_filters/expressions.py +181 -0
  55. wandb/automations/_filters/operators.py +258 -0
  56. wandb/automations/_filters/run_metrics.py +330 -0
  57. wandb/automations/_generated/__init__.py +177 -0
  58. wandb/automations/_generated/create_automation.py +17 -0
  59. wandb/automations/_generated/create_generic_webhook_integration.py +43 -0
  60. wandb/automations/_generated/delete_automation.py +15 -0
  61. wandb/automations/_generated/enums.py +35 -0
  62. wandb/automations/_generated/fragments.py +358 -0
  63. wandb/automations/_generated/generic_webhook_integrations_by_entity.py +22 -0
  64. wandb/automations/_generated/get_automations.py +24 -0
  65. wandb/automations/_generated/get_automations_by_entity.py +26 -0
  66. wandb/automations/_generated/input_types.py +104 -0
  67. wandb/automations/_generated/integrations_by_entity.py +22 -0
  68. wandb/automations/_generated/operations.py +647 -0
  69. wandb/automations/_generated/slack_integrations_by_entity.py +22 -0
  70. wandb/automations/_generated/update_automation.py +17 -0
  71. wandb/automations/_utils.py +235 -0
  72. wandb/automations/_validators.py +165 -0
  73. wandb/automations/actions.py +218 -0
  74. wandb/automations/automations.py +85 -0
  75. wandb/automations/events.py +285 -0
  76. wandb/automations/integrations.py +45 -0
  77. wandb/automations/scopes.py +78 -0
  78. wandb/beta/workflows.py +324 -0
  79. wandb/bin/gpu_stats +0 -0
  80. wandb/bin/wandb-core +0 -0
  81. wandb/cli/__init__.py +0 -0
  82. wandb/cli/beta.py +175 -0
  83. wandb/cli/cli.py +2883 -0
  84. wandb/data_types.py +66 -0
  85. wandb/docker/__init__.py +290 -0
  86. wandb/docker/names.py +40 -0
  87. wandb/docker/wandb-entrypoint.sh +33 -0
  88. wandb/env.py +535 -0
  89. wandb/errors/__init__.py +17 -0
  90. wandb/errors/errors.py +40 -0
  91. wandb/errors/links.py +73 -0
  92. wandb/errors/term.py +415 -0
  93. wandb/errors/util.py +57 -0
  94. wandb/errors/warnings.py +2 -0
  95. wandb/filesync/__init__.py +0 -0
  96. wandb/filesync/dir_watcher.py +404 -0
  97. wandb/filesync/stats.py +100 -0
  98. wandb/filesync/step_checksum.py +142 -0
  99. wandb/filesync/step_prepare.py +179 -0
  100. wandb/filesync/step_upload.py +287 -0
  101. wandb/filesync/upload_job.py +142 -0
  102. wandb/integration/__init__.py +0 -0
  103. wandb/integration/catboost/__init__.py +5 -0
  104. wandb/integration/catboost/catboost.py +182 -0
  105. wandb/integration/cohere/__init__.py +3 -0
  106. wandb/integration/cohere/cohere.py +21 -0
  107. wandb/integration/cohere/resolver.py +347 -0
  108. wandb/integration/diffusers/__init__.py +3 -0
  109. wandb/integration/diffusers/autologger.py +76 -0
  110. wandb/integration/diffusers/pipeline_resolver.py +50 -0
  111. wandb/integration/diffusers/resolvers/__init__.py +9 -0
  112. wandb/integration/diffusers/resolvers/multimodal.py +881 -0
  113. wandb/integration/diffusers/resolvers/utils.py +102 -0
  114. wandb/integration/fastai/__init__.py +243 -0
  115. wandb/integration/gym/__init__.py +98 -0
  116. wandb/integration/huggingface/__init__.py +3 -0
  117. wandb/integration/huggingface/huggingface.py +18 -0
  118. wandb/integration/huggingface/resolver.py +213 -0
  119. wandb/integration/keras/__init__.py +11 -0
  120. wandb/integration/keras/callbacks/__init__.py +5 -0
  121. wandb/integration/keras/callbacks/metrics_logger.py +129 -0
  122. wandb/integration/keras/callbacks/model_checkpoint.py +188 -0
  123. wandb/integration/keras/callbacks/tables_builder.py +228 -0
  124. wandb/integration/keras/keras.py +1086 -0
  125. wandb/integration/kfp/__init__.py +6 -0
  126. wandb/integration/kfp/helpers.py +28 -0
  127. wandb/integration/kfp/kfp_patch.py +335 -0
  128. wandb/integration/kfp/wandb_logging.py +182 -0
  129. wandb/integration/langchain/__init__.py +3 -0
  130. wandb/integration/langchain/wandb_tracer.py +49 -0
  131. wandb/integration/lightgbm/__init__.py +239 -0
  132. wandb/integration/lightning/__init__.py +0 -0
  133. wandb/integration/lightning/fabric/__init__.py +3 -0
  134. wandb/integration/lightning/fabric/logger.py +763 -0
  135. wandb/integration/metaflow/__init__.py +9 -0
  136. wandb/integration/metaflow/data_pandas.py +74 -0
  137. wandb/integration/metaflow/data_pytorch.py +75 -0
  138. wandb/integration/metaflow/data_sklearn.py +76 -0
  139. wandb/integration/metaflow/errors.py +13 -0
  140. wandb/integration/metaflow/metaflow.py +327 -0
  141. wandb/integration/openai/__init__.py +3 -0
  142. wandb/integration/openai/fine_tuning.py +480 -0
  143. wandb/integration/openai/openai.py +22 -0
  144. wandb/integration/openai/resolver.py +240 -0
  145. wandb/integration/prodigy/__init__.py +3 -0
  146. wandb/integration/prodigy/prodigy.py +291 -0
  147. wandb/integration/sacred/__init__.py +117 -0
  148. wandb/integration/sagemaker/__init__.py +14 -0
  149. wandb/integration/sagemaker/auth.py +29 -0
  150. wandb/integration/sagemaker/config.py +58 -0
  151. wandb/integration/sagemaker/files.py +2 -0
  152. wandb/integration/sagemaker/resources.py +63 -0
  153. wandb/integration/sb3/__init__.py +3 -0
  154. wandb/integration/sb3/sb3.py +147 -0
  155. wandb/integration/sklearn/__init__.py +37 -0
  156. wandb/integration/sklearn/calculate/__init__.py +32 -0
  157. wandb/integration/sklearn/calculate/calibration_curves.py +125 -0
  158. wandb/integration/sklearn/calculate/class_proportions.py +68 -0
  159. wandb/integration/sklearn/calculate/confusion_matrix.py +93 -0
  160. wandb/integration/sklearn/calculate/decision_boundaries.py +40 -0
  161. wandb/integration/sklearn/calculate/elbow_curve.py +55 -0
  162. wandb/integration/sklearn/calculate/feature_importances.py +67 -0
  163. wandb/integration/sklearn/calculate/learning_curve.py +64 -0
  164. wandb/integration/sklearn/calculate/outlier_candidates.py +69 -0
  165. wandb/integration/sklearn/calculate/residuals.py +86 -0
  166. wandb/integration/sklearn/calculate/silhouette.py +118 -0
  167. wandb/integration/sklearn/calculate/summary_metrics.py +62 -0
  168. wandb/integration/sklearn/plot/__init__.py +35 -0
  169. wandb/integration/sklearn/plot/classifier.py +329 -0
  170. wandb/integration/sklearn/plot/clusterer.py +146 -0
  171. wandb/integration/sklearn/plot/regressor.py +121 -0
  172. wandb/integration/sklearn/plot/shared.py +91 -0
  173. wandb/integration/sklearn/utils.py +184 -0
  174. wandb/integration/tensorboard/__init__.py +10 -0
  175. wandb/integration/tensorboard/log.py +351 -0
  176. wandb/integration/tensorboard/monkeypatch.py +186 -0
  177. wandb/integration/tensorflow/__init__.py +5 -0
  178. wandb/integration/tensorflow/estimator_hook.py +54 -0
  179. wandb/integration/torch/__init__.py +0 -0
  180. wandb/integration/torch/wandb_torch.py +554 -0
  181. wandb/integration/ultralytics/__init__.py +11 -0
  182. wandb/integration/ultralytics/bbox_utils.py +215 -0
  183. wandb/integration/ultralytics/callback.py +528 -0
  184. wandb/integration/ultralytics/classification_utils.py +83 -0
  185. wandb/integration/ultralytics/mask_utils.py +202 -0
  186. wandb/integration/ultralytics/pose_utils.py +103 -0
  187. wandb/integration/weave/__init__.py +6 -0
  188. wandb/integration/weave/interface.py +49 -0
  189. wandb/integration/weave/weave.py +63 -0
  190. wandb/integration/xgboost/__init__.py +11 -0
  191. wandb/integration/xgboost/xgboost.py +189 -0
  192. wandb/integration/yolov8/__init__.py +0 -0
  193. wandb/integration/yolov8/yolov8.py +284 -0
  194. wandb/jupyter.py +538 -0
  195. wandb/mpmain/__init__.py +0 -0
  196. wandb/mpmain/__main__.py +1 -0
  197. wandb/old/__init__.py +0 -0
  198. wandb/old/core.py +53 -0
  199. wandb/old/settings.py +176 -0
  200. wandb/old/summary.py +438 -0
  201. wandb/plot/__init__.py +30 -0
  202. wandb/plot/bar.py +71 -0
  203. wandb/plot/confusion_matrix.py +185 -0
  204. wandb/plot/custom_chart.py +147 -0
  205. wandb/plot/histogram.py +66 -0
  206. wandb/plot/line.py +75 -0
  207. wandb/plot/line_series.py +173 -0
  208. wandb/plot/pr_curve.py +186 -0
  209. wandb/plot/roc_curve.py +163 -0
  210. wandb/plot/scatter.py +66 -0
  211. wandb/plot/utils.py +184 -0
  212. wandb/plot/viz.py +41 -0
  213. wandb/proto/__init__.py +0 -0
  214. wandb/proto/v3/__init__.py +0 -0
  215. wandb/proto/v3/wandb_base_pb2.py +55 -0
  216. wandb/proto/v3/wandb_internal_pb2.py +1728 -0
  217. wandb/proto/v3/wandb_server_pb2.py +228 -0
  218. wandb/proto/v3/wandb_settings_pb2.py +122 -0
  219. wandb/proto/v3/wandb_telemetry_pb2.py +106 -0
  220. wandb/proto/v4/__init__.py +0 -0
  221. wandb/proto/v4/wandb_base_pb2.py +30 -0
  222. wandb/proto/v4/wandb_internal_pb2.py +382 -0
  223. wandb/proto/v4/wandb_server_pb2.py +67 -0
  224. wandb/proto/v4/wandb_settings_pb2.py +47 -0
  225. wandb/proto/v4/wandb_telemetry_pb2.py +41 -0
  226. wandb/proto/v5/wandb_base_pb2.py +31 -0
  227. wandb/proto/v5/wandb_internal_pb2.py +383 -0
  228. wandb/proto/v5/wandb_server_pb2.py +68 -0
  229. wandb/proto/v5/wandb_settings_pb2.py +48 -0
  230. wandb/proto/v5/wandb_telemetry_pb2.py +42 -0
  231. wandb/proto/v6/wandb_base_pb2.py +41 -0
  232. wandb/proto/v6/wandb_internal_pb2.py +393 -0
  233. wandb/proto/v6/wandb_server_pb2.py +78 -0
  234. wandb/proto/v6/wandb_settings_pb2.py +58 -0
  235. wandb/proto/v6/wandb_telemetry_pb2.py +52 -0
  236. wandb/proto/wandb_base_pb2.py +12 -0
  237. wandb/proto/wandb_deprecated.py +59 -0
  238. wandb/proto/wandb_generate_deprecated.py +30 -0
  239. wandb/proto/wandb_generate_proto.py +49 -0
  240. wandb/proto/wandb_internal_pb2.py +18 -0
  241. wandb/proto/wandb_server_pb2.py +12 -0
  242. wandb/proto/wandb_settings_pb2.py +12 -0
  243. wandb/proto/wandb_telemetry_pb2.py +12 -0
  244. wandb/py.typed +0 -0
  245. wandb/sdk/__init__.py +37 -0
  246. wandb/sdk/artifacts/__init__.py +0 -0
  247. wandb/sdk/artifacts/_factories.py +17 -0
  248. wandb/sdk/artifacts/_generated/__init__.py +508 -0
  249. wandb/sdk/artifacts/_generated/add_aliases.py +21 -0
  250. wandb/sdk/artifacts/_generated/artifact_by_id.py +17 -0
  251. wandb/sdk/artifacts/_generated/artifact_by_name.py +22 -0
  252. wandb/sdk/artifacts/_generated/artifact_collection_membership_file_urls.py +43 -0
  253. wandb/sdk/artifacts/_generated/artifact_collection_membership_files.py +43 -0
  254. wandb/sdk/artifacts/_generated/artifact_created_by.py +47 -0
  255. wandb/sdk/artifacts/_generated/artifact_file_urls.py +22 -0
  256. wandb/sdk/artifacts/_generated/artifact_type.py +31 -0
  257. wandb/sdk/artifacts/_generated/artifact_used_by.py +43 -0
  258. wandb/sdk/artifacts/_generated/artifact_version_files.py +36 -0
  259. wandb/sdk/artifacts/_generated/artifact_via_membership_by_name.py +26 -0
  260. wandb/sdk/artifacts/_generated/create_artifact_collection_tag_assignments.py +36 -0
  261. wandb/sdk/artifacts/_generated/delete_aliases.py +21 -0
  262. wandb/sdk/artifacts/_generated/delete_artifact.py +28 -0
  263. wandb/sdk/artifacts/_generated/delete_artifact_collection_tag_assignments.py +25 -0
  264. wandb/sdk/artifacts/_generated/delete_artifact_portfolio.py +35 -0
  265. wandb/sdk/artifacts/_generated/delete_artifact_sequence.py +35 -0
  266. wandb/sdk/artifacts/_generated/enums.py +22 -0
  267. wandb/sdk/artifacts/_generated/fetch_artifact_manifest.py +38 -0
  268. wandb/sdk/artifacts/_generated/fetch_linked_artifacts.py +67 -0
  269. wandb/sdk/artifacts/_generated/fetch_registries.py +32 -0
  270. wandb/sdk/artifacts/_generated/fragments.py +459 -0
  271. wandb/sdk/artifacts/_generated/input_types.py +46 -0
  272. wandb/sdk/artifacts/_generated/link_artifact.py +27 -0
  273. wandb/sdk/artifacts/_generated/move_artifact_collection.py +35 -0
  274. wandb/sdk/artifacts/_generated/operations.py +1223 -0
  275. wandb/sdk/artifacts/_generated/project_artifact_collection.py +101 -0
  276. wandb/sdk/artifacts/_generated/project_artifact_collections.py +33 -0
  277. wandb/sdk/artifacts/_generated/project_artifact_type.py +24 -0
  278. wandb/sdk/artifacts/_generated/project_artifact_types.py +24 -0
  279. wandb/sdk/artifacts/_generated/project_artifacts.py +42 -0
  280. wandb/sdk/artifacts/_generated/registry_collections.py +34 -0
  281. wandb/sdk/artifacts/_generated/registry_versions.py +34 -0
  282. wandb/sdk/artifacts/_generated/run_input_artifacts.py +51 -0
  283. wandb/sdk/artifacts/_generated/run_output_artifacts.py +51 -0
  284. wandb/sdk/artifacts/_generated/unlink_artifact.py +25 -0
  285. wandb/sdk/artifacts/_generated/update_artifact.py +26 -0
  286. wandb/sdk/artifacts/_generated/update_artifact_portfolio.py +35 -0
  287. wandb/sdk/artifacts/_generated/update_artifact_sequence.py +35 -0
  288. wandb/sdk/artifacts/_graphql_fragments.py +19 -0
  289. wandb/sdk/artifacts/_internal_artifact.py +54 -0
  290. wandb/sdk/artifacts/_validators.py +309 -0
  291. wandb/sdk/artifacts/artifact.py +2702 -0
  292. wandb/sdk/artifacts/artifact_download_logger.py +45 -0
  293. wandb/sdk/artifacts/artifact_file_cache.py +251 -0
  294. wandb/sdk/artifacts/artifact_instance_cache.py +17 -0
  295. wandb/sdk/artifacts/artifact_manifest.py +76 -0
  296. wandb/sdk/artifacts/artifact_manifest_entry.py +258 -0
  297. wandb/sdk/artifacts/artifact_manifests/__init__.py +0 -0
  298. wandb/sdk/artifacts/artifact_manifests/artifact_manifest_v1.py +94 -0
  299. wandb/sdk/artifacts/artifact_saver.py +277 -0
  300. wandb/sdk/artifacts/artifact_state.py +13 -0
  301. wandb/sdk/artifacts/artifact_ttl.py +9 -0
  302. wandb/sdk/artifacts/exceptions.py +71 -0
  303. wandb/sdk/artifacts/staging.py +27 -0
  304. wandb/sdk/artifacts/storage_handler.py +62 -0
  305. wandb/sdk/artifacts/storage_handlers/__init__.py +0 -0
  306. wandb/sdk/artifacts/storage_handlers/azure_handler.py +214 -0
  307. wandb/sdk/artifacts/storage_handlers/gcs_handler.py +224 -0
  308. wandb/sdk/artifacts/storage_handlers/http_handler.py +114 -0
  309. wandb/sdk/artifacts/storage_handlers/local_file_handler.py +142 -0
  310. wandb/sdk/artifacts/storage_handlers/multi_handler.py +56 -0
  311. wandb/sdk/artifacts/storage_handlers/s3_handler.py +339 -0
  312. wandb/sdk/artifacts/storage_handlers/tracking_handler.py +68 -0
  313. wandb/sdk/artifacts/storage_handlers/wb_artifact_handler.py +131 -0
  314. wandb/sdk/artifacts/storage_handlers/wb_local_artifact_handler.py +74 -0
  315. wandb/sdk/artifacts/storage_layout.py +8 -0
  316. wandb/sdk/artifacts/storage_policies/__init__.py +4 -0
  317. wandb/sdk/artifacts/storage_policies/register.py +1 -0
  318. wandb/sdk/artifacts/storage_policies/wandb_storage_policy.py +580 -0
  319. wandb/sdk/artifacts/storage_policy.py +75 -0
  320. wandb/sdk/backend/__init__.py +0 -0
  321. wandb/sdk/backend/backend.py +57 -0
  322. wandb/sdk/data_types/__init__.py +0 -0
  323. wandb/sdk/data_types/_dtypes.py +914 -0
  324. wandb/sdk/data_types/_private.py +10 -0
  325. wandb/sdk/data_types/audio.py +208 -0
  326. wandb/sdk/data_types/base_types/__init__.py +0 -0
  327. wandb/sdk/data_types/base_types/json_metadata.py +55 -0
  328. wandb/sdk/data_types/base_types/media.py +339 -0
  329. wandb/sdk/data_types/base_types/wb_value.py +295 -0
  330. wandb/sdk/data_types/bokeh.py +87 -0
  331. wandb/sdk/data_types/graph.py +439 -0
  332. wandb/sdk/data_types/helper_types/__init__.py +0 -0
  333. wandb/sdk/data_types/helper_types/bounding_boxes_2d.py +327 -0
  334. wandb/sdk/data_types/helper_types/classes.py +159 -0
  335. wandb/sdk/data_types/helper_types/image_mask.py +251 -0
  336. wandb/sdk/data_types/histogram.py +107 -0
  337. wandb/sdk/data_types/html.py +165 -0
  338. wandb/sdk/data_types/image.py +974 -0
  339. wandb/sdk/data_types/molecule.py +250 -0
  340. wandb/sdk/data_types/object_3d.py +495 -0
  341. wandb/sdk/data_types/plotly.py +95 -0
  342. wandb/sdk/data_types/saved_model.py +435 -0
  343. wandb/sdk/data_types/table.py +1468 -0
  344. wandb/sdk/data_types/table_decorators.py +108 -0
  345. wandb/sdk/data_types/trace_tree.py +440 -0
  346. wandb/sdk/data_types/utils.py +260 -0
  347. wandb/sdk/data_types/video.py +303 -0
  348. wandb/sdk/integration_utils/__init__.py +0 -0
  349. wandb/sdk/integration_utils/auto_logging.py +232 -0
  350. wandb/sdk/integration_utils/data_logging.py +475 -0
  351. wandb/sdk/interface/__init__.py +0 -0
  352. wandb/sdk/interface/constants.py +4 -0
  353. wandb/sdk/interface/interface.py +1056 -0
  354. wandb/sdk/interface/interface_queue.py +40 -0
  355. wandb/sdk/interface/interface_shared.py +471 -0
  356. wandb/sdk/interface/interface_sock.py +49 -0
  357. wandb/sdk/interface/summary_record.py +67 -0
  358. wandb/sdk/internal/__init__.py +0 -0
  359. wandb/sdk/internal/_generated/__init__.py +15 -0
  360. wandb/sdk/internal/_generated/enums.py +4 -0
  361. wandb/sdk/internal/_generated/input_types.py +4 -0
  362. wandb/sdk/internal/_generated/operations.py +15 -0
  363. wandb/sdk/internal/_generated/server_features_query.py +27 -0
  364. wandb/sdk/internal/context.py +89 -0
  365. wandb/sdk/internal/datastore.py +293 -0
  366. wandb/sdk/internal/file_pusher.py +177 -0
  367. wandb/sdk/internal/file_stream.py +686 -0
  368. wandb/sdk/internal/handler.py +854 -0
  369. wandb/sdk/internal/incremental_table_util.py +53 -0
  370. wandb/sdk/internal/internal_api.py +4723 -0
  371. wandb/sdk/internal/job_builder.py +639 -0
  372. wandb/sdk/internal/profiler.py +79 -0
  373. wandb/sdk/internal/progress.py +77 -0
  374. wandb/sdk/internal/run.py +27 -0
  375. wandb/sdk/internal/sample.py +70 -0
  376. wandb/sdk/internal/sender.py +1692 -0
  377. wandb/sdk/internal/sender_config.py +203 -0
  378. wandb/sdk/internal/settings_static.py +120 -0
  379. wandb/sdk/internal/tb_watcher.py +519 -0
  380. wandb/sdk/internal/thread_local_settings.py +18 -0
  381. wandb/sdk/launch/__init__.py +15 -0
  382. wandb/sdk/launch/_launch.py +331 -0
  383. wandb/sdk/launch/_launch_add.py +255 -0
  384. wandb/sdk/launch/_project_spec.py +565 -0
  385. wandb/sdk/launch/agent/__init__.py +5 -0
  386. wandb/sdk/launch/agent/agent.py +931 -0
  387. wandb/sdk/launch/agent/config.py +296 -0
  388. wandb/sdk/launch/agent/job_status_tracker.py +55 -0
  389. wandb/sdk/launch/agent/run_queue_item_file_saver.py +39 -0
  390. wandb/sdk/launch/builder/__init__.py +0 -0
  391. wandb/sdk/launch/builder/abstract.py +156 -0
  392. wandb/sdk/launch/builder/build.py +296 -0
  393. wandb/sdk/launch/builder/context_manager.py +235 -0
  394. wandb/sdk/launch/builder/docker_builder.py +177 -0
  395. wandb/sdk/launch/builder/kaniko_builder.py +595 -0
  396. wandb/sdk/launch/builder/noop.py +58 -0
  397. wandb/sdk/launch/builder/templates/_wandb_bootstrap.py +188 -0
  398. wandb/sdk/launch/builder/templates/dockerfile.py +92 -0
  399. wandb/sdk/launch/create_job.py +541 -0
  400. wandb/sdk/launch/environment/abstract.py +29 -0
  401. wandb/sdk/launch/environment/aws_environment.py +322 -0
  402. wandb/sdk/launch/environment/azure_environment.py +105 -0
  403. wandb/sdk/launch/environment/gcp_environment.py +334 -0
  404. wandb/sdk/launch/environment/local_environment.py +65 -0
  405. wandb/sdk/launch/errors.py +13 -0
  406. wandb/sdk/launch/git_reference.py +109 -0
  407. wandb/sdk/launch/inputs/files.py +148 -0
  408. wandb/sdk/launch/inputs/internal.py +314 -0
  409. wandb/sdk/launch/inputs/manage.py +113 -0
  410. wandb/sdk/launch/inputs/schema.py +40 -0
  411. wandb/sdk/launch/loader.py +249 -0
  412. wandb/sdk/launch/registry/abstract.py +48 -0
  413. wandb/sdk/launch/registry/anon.py +29 -0
  414. wandb/sdk/launch/registry/azure_container_registry.py +124 -0
  415. wandb/sdk/launch/registry/elastic_container_registry.py +192 -0
  416. wandb/sdk/launch/registry/google_artifact_registry.py +219 -0
  417. wandb/sdk/launch/registry/local_registry.py +65 -0
  418. wandb/sdk/launch/runner/__init__.py +0 -0
  419. wandb/sdk/launch/runner/abstract.py +185 -0
  420. wandb/sdk/launch/runner/kubernetes_monitor.py +473 -0
  421. wandb/sdk/launch/runner/kubernetes_runner.py +1285 -0
  422. wandb/sdk/launch/runner/local_container.py +301 -0
  423. wandb/sdk/launch/runner/local_process.py +78 -0
  424. wandb/sdk/launch/runner/sagemaker_runner.py +424 -0
  425. wandb/sdk/launch/runner/vertex_runner.py +225 -0
  426. wandb/sdk/launch/sweeps/__init__.py +37 -0
  427. wandb/sdk/launch/sweeps/scheduler.py +739 -0
  428. wandb/sdk/launch/sweeps/scheduler_sweep.py +90 -0
  429. wandb/sdk/launch/sweeps/utils.py +324 -0
  430. wandb/sdk/launch/utils.py +746 -0
  431. wandb/sdk/launch/wandb_reference.py +138 -0
  432. wandb/sdk/lib/__init__.py +5 -0
  433. wandb/sdk/lib/apikey.py +334 -0
  434. wandb/sdk/lib/asyncio_compat.py +213 -0
  435. wandb/sdk/lib/asyncio_manager.py +252 -0
  436. wandb/sdk/lib/capped_dict.py +26 -0
  437. wandb/sdk/lib/config_util.py +101 -0
  438. wandb/sdk/lib/console_capture.py +219 -0
  439. wandb/sdk/lib/credentials.py +141 -0
  440. wandb/sdk/lib/deprecate.py +27 -0
  441. wandb/sdk/lib/disabled.py +30 -0
  442. wandb/sdk/lib/exit_hooks.py +54 -0
  443. wandb/sdk/lib/file_stream_utils.py +118 -0
  444. wandb/sdk/lib/filenames.py +64 -0
  445. wandb/sdk/lib/filesystem.py +372 -0
  446. wandb/sdk/lib/fsm.py +165 -0
  447. wandb/sdk/lib/gitlib.py +240 -0
  448. wandb/sdk/lib/gql_request.py +65 -0
  449. wandb/sdk/lib/handler_util.py +21 -0
  450. wandb/sdk/lib/hashutil.py +106 -0
  451. wandb/sdk/lib/import_hooks.py +275 -0
  452. wandb/sdk/lib/interrupt.py +37 -0
  453. wandb/sdk/lib/ipython.py +126 -0
  454. wandb/sdk/lib/json_util.py +75 -0
  455. wandb/sdk/lib/lazyloader.py +63 -0
  456. wandb/sdk/lib/module.py +72 -0
  457. wandb/sdk/lib/paths.py +106 -0
  458. wandb/sdk/lib/preinit.py +42 -0
  459. wandb/sdk/lib/printer.py +571 -0
  460. wandb/sdk/lib/printer_asyncio.py +48 -0
  461. wandb/sdk/lib/progress.py +320 -0
  462. wandb/sdk/lib/proto_util.py +90 -0
  463. wandb/sdk/lib/redirect.py +876 -0
  464. wandb/sdk/lib/retry.py +395 -0
  465. wandb/sdk/lib/run_moment.py +82 -0
  466. wandb/sdk/lib/runid.py +12 -0
  467. wandb/sdk/lib/server.py +58 -0
  468. wandb/sdk/lib/service/ipc_support.py +13 -0
  469. wandb/sdk/lib/service/service_client.py +106 -0
  470. wandb/sdk/lib/service/service_connection.py +192 -0
  471. wandb/sdk/lib/service/service_port_file.py +105 -0
  472. wandb/sdk/lib/service/service_process.py +111 -0
  473. wandb/sdk/lib/service/service_token.py +181 -0
  474. wandb/sdk/lib/sparkline.py +44 -0
  475. wandb/sdk/lib/telemetry.py +100 -0
  476. wandb/sdk/lib/timed_input.py +133 -0
  477. wandb/sdk/lib/timer.py +19 -0
  478. wandb/sdk/lib/wb_logging.py +161 -0
  479. wandb/sdk/mailbox/__init__.py +23 -0
  480. wandb/sdk/mailbox/mailbox.py +143 -0
  481. wandb/sdk/mailbox/mailbox_handle.py +132 -0
  482. wandb/sdk/mailbox/response_handle.py +99 -0
  483. wandb/sdk/mailbox/wait_with_progress.py +100 -0
  484. wandb/sdk/projects/_generated/__init__.py +47 -0
  485. wandb/sdk/projects/_generated/delete_project.py +22 -0
  486. wandb/sdk/projects/_generated/enums.py +4 -0
  487. wandb/sdk/projects/_generated/fetch_registry.py +22 -0
  488. wandb/sdk/projects/_generated/fragments.py +41 -0
  489. wandb/sdk/projects/_generated/input_types.py +13 -0
  490. wandb/sdk/projects/_generated/operations.py +88 -0
  491. wandb/sdk/projects/_generated/rename_project.py +27 -0
  492. wandb/sdk/projects/_generated/upsert_registry_project.py +27 -0
  493. wandb/sdk/verify/__init__.py +0 -0
  494. wandb/sdk/verify/verify.py +555 -0
  495. wandb/sdk/wandb_alerts.py +12 -0
  496. wandb/sdk/wandb_config.py +323 -0
  497. wandb/sdk/wandb_helper.py +54 -0
  498. wandb/sdk/wandb_init.py +1581 -0
  499. wandb/sdk/wandb_login.py +332 -0
  500. wandb/sdk/wandb_metric.py +112 -0
  501. wandb/sdk/wandb_require.py +88 -0
  502. wandb/sdk/wandb_require_helpers.py +44 -0
  503. wandb/sdk/wandb_run.py +4088 -0
  504. wandb/sdk/wandb_settings.py +2105 -0
  505. wandb/sdk/wandb_setup.py +560 -0
  506. wandb/sdk/wandb_summary.py +150 -0
  507. wandb/sdk/wandb_sweep.py +120 -0
  508. wandb/sdk/wandb_sync.py +71 -0
  509. wandb/sdk/wandb_watch.py +146 -0
  510. wandb/sklearn.py +35 -0
  511. wandb/sync/__init__.py +3 -0
  512. wandb/sync/sync.py +452 -0
  513. wandb/trigger.py +29 -0
  514. wandb/util.py +2040 -0
  515. wandb/vendor/__init__.py +0 -0
  516. wandb/vendor/gql-0.2.0/setup.py +40 -0
  517. wandb/vendor/gql-0.2.0/tests/__init__.py +0 -0
  518. wandb/vendor/gql-0.2.0/tests/starwars/__init__.py +0 -0
  519. wandb/vendor/gql-0.2.0/tests/starwars/fixtures.py +96 -0
  520. wandb/vendor/gql-0.2.0/tests/starwars/schema.py +146 -0
  521. wandb/vendor/gql-0.2.0/tests/starwars/test_dsl.py +293 -0
  522. wandb/vendor/gql-0.2.0/tests/starwars/test_query.py +355 -0
  523. wandb/vendor/gql-0.2.0/tests/starwars/test_validation.py +171 -0
  524. wandb/vendor/gql-0.2.0/tests/test_client.py +31 -0
  525. wandb/vendor/gql-0.2.0/tests/test_transport.py +89 -0
  526. wandb/vendor/gql-0.2.0/wandb_gql/__init__.py +4 -0
  527. wandb/vendor/gql-0.2.0/wandb_gql/client.py +75 -0
  528. wandb/vendor/gql-0.2.0/wandb_gql/dsl.py +152 -0
  529. wandb/vendor/gql-0.2.0/wandb_gql/gql.py +10 -0
  530. wandb/vendor/gql-0.2.0/wandb_gql/transport/__init__.py +0 -0
  531. wandb/vendor/gql-0.2.0/wandb_gql/transport/http.py +6 -0
  532. wandb/vendor/gql-0.2.0/wandb_gql/transport/local_schema.py +15 -0
  533. wandb/vendor/gql-0.2.0/wandb_gql/transport/requests.py +46 -0
  534. wandb/vendor/gql-0.2.0/wandb_gql/utils.py +21 -0
  535. wandb/vendor/graphql-core-1.1/setup.py +86 -0
  536. wandb/vendor/graphql-core-1.1/wandb_graphql/__init__.py +287 -0
  537. wandb/vendor/graphql-core-1.1/wandb_graphql/error/__init__.py +6 -0
  538. wandb/vendor/graphql-core-1.1/wandb_graphql/error/base.py +42 -0
  539. wandb/vendor/graphql-core-1.1/wandb_graphql/error/format_error.py +11 -0
  540. wandb/vendor/graphql-core-1.1/wandb_graphql/error/located_error.py +29 -0
  541. wandb/vendor/graphql-core-1.1/wandb_graphql/error/syntax_error.py +36 -0
  542. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/__init__.py +26 -0
  543. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/base.py +311 -0
  544. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executor.py +398 -0
  545. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/__init__.py +0 -0
  546. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/asyncio.py +53 -0
  547. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/gevent.py +22 -0
  548. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/process.py +32 -0
  549. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/sync.py +7 -0
  550. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/thread.py +35 -0
  551. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/executors/utils.py +6 -0
  552. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/experimental/__init__.py +0 -0
  553. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/experimental/executor.py +66 -0
  554. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/experimental/fragment.py +252 -0
  555. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/experimental/resolver.py +151 -0
  556. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/experimental/utils.py +7 -0
  557. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/middleware.py +57 -0
  558. wandb/vendor/graphql-core-1.1/wandb_graphql/execution/values.py +145 -0
  559. wandb/vendor/graphql-core-1.1/wandb_graphql/graphql.py +60 -0
  560. wandb/vendor/graphql-core-1.1/wandb_graphql/language/__init__.py +0 -0
  561. wandb/vendor/graphql-core-1.1/wandb_graphql/language/ast.py +1349 -0
  562. wandb/vendor/graphql-core-1.1/wandb_graphql/language/base.py +19 -0
  563. wandb/vendor/graphql-core-1.1/wandb_graphql/language/lexer.py +435 -0
  564. wandb/vendor/graphql-core-1.1/wandb_graphql/language/location.py +30 -0
  565. wandb/vendor/graphql-core-1.1/wandb_graphql/language/parser.py +779 -0
  566. wandb/vendor/graphql-core-1.1/wandb_graphql/language/printer.py +193 -0
  567. wandb/vendor/graphql-core-1.1/wandb_graphql/language/source.py +18 -0
  568. wandb/vendor/graphql-core-1.1/wandb_graphql/language/visitor.py +222 -0
  569. wandb/vendor/graphql-core-1.1/wandb_graphql/language/visitor_meta.py +82 -0
  570. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/__init__.py +0 -0
  571. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/cached_property.py +17 -0
  572. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/contain_subset.py +28 -0
  573. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/default_ordered_dict.py +40 -0
  574. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/ordereddict.py +8 -0
  575. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/pair_set.py +43 -0
  576. wandb/vendor/graphql-core-1.1/wandb_graphql/pyutils/version.py +78 -0
  577. wandb/vendor/graphql-core-1.1/wandb_graphql/type/__init__.py +67 -0
  578. wandb/vendor/graphql-core-1.1/wandb_graphql/type/definition.py +619 -0
  579. wandb/vendor/graphql-core-1.1/wandb_graphql/type/directives.py +132 -0
  580. wandb/vendor/graphql-core-1.1/wandb_graphql/type/introspection.py +440 -0
  581. wandb/vendor/graphql-core-1.1/wandb_graphql/type/scalars.py +131 -0
  582. wandb/vendor/graphql-core-1.1/wandb_graphql/type/schema.py +100 -0
  583. wandb/vendor/graphql-core-1.1/wandb_graphql/type/typemap.py +145 -0
  584. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/__init__.py +0 -0
  585. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/assert_valid_name.py +9 -0
  586. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/ast_from_value.py +65 -0
  587. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/ast_to_code.py +49 -0
  588. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/ast_to_dict.py +24 -0
  589. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/base.py +75 -0
  590. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/build_ast_schema.py +291 -0
  591. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/build_client_schema.py +250 -0
  592. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/concat_ast.py +9 -0
  593. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/extend_schema.py +357 -0
  594. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/get_field_def.py +27 -0
  595. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/get_operation_ast.py +21 -0
  596. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/introspection_query.py +90 -0
  597. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/is_valid_literal_value.py +67 -0
  598. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/is_valid_value.py +66 -0
  599. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/quoted_or_list.py +21 -0
  600. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/schema_printer.py +168 -0
  601. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/suggestion_list.py +56 -0
  602. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/type_comparators.py +69 -0
  603. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/type_from_ast.py +21 -0
  604. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/type_info.py +149 -0
  605. wandb/vendor/graphql-core-1.1/wandb_graphql/utils/value_from_ast.py +69 -0
  606. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/__init__.py +4 -0
  607. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/__init__.py +79 -0
  608. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/arguments_of_correct_type.py +24 -0
  609. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/base.py +8 -0
  610. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/default_values_of_correct_type.py +44 -0
  611. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/fields_on_correct_type.py +113 -0
  612. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/fragments_on_composite_types.py +33 -0
  613. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/known_argument_names.py +70 -0
  614. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/known_directives.py +97 -0
  615. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/known_fragment_names.py +19 -0
  616. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/known_type_names.py +43 -0
  617. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/lone_anonymous_operation.py +23 -0
  618. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/no_fragment_cycles.py +59 -0
  619. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/no_undefined_variables.py +36 -0
  620. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/no_unused_fragments.py +38 -0
  621. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/no_unused_variables.py +37 -0
  622. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/overlapping_fields_can_be_merged.py +529 -0
  623. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/possible_fragment_spreads.py +44 -0
  624. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/provided_non_null_arguments.py +46 -0
  625. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/scalar_leafs.py +33 -0
  626. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/unique_argument_names.py +32 -0
  627. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/unique_fragment_names.py +28 -0
  628. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/unique_input_field_names.py +33 -0
  629. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/unique_operation_names.py +31 -0
  630. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/unique_variable_names.py +27 -0
  631. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/variables_are_input_types.py +21 -0
  632. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/rules/variables_in_allowed_position.py +53 -0
  633. wandb/vendor/graphql-core-1.1/wandb_graphql/validation/validation.py +158 -0
  634. wandb/vendor/promise-2.3.0/conftest.py +30 -0
  635. wandb/vendor/promise-2.3.0/setup.py +64 -0
  636. wandb/vendor/promise-2.3.0/tests/__init__.py +0 -0
  637. wandb/vendor/promise-2.3.0/tests/conftest.py +8 -0
  638. wandb/vendor/promise-2.3.0/tests/test_awaitable.py +32 -0
  639. wandb/vendor/promise-2.3.0/tests/test_awaitable_35.py +47 -0
  640. wandb/vendor/promise-2.3.0/tests/test_benchmark.py +116 -0
  641. wandb/vendor/promise-2.3.0/tests/test_complex_threads.py +23 -0
  642. wandb/vendor/promise-2.3.0/tests/test_dataloader.py +452 -0
  643. wandb/vendor/promise-2.3.0/tests/test_dataloader_awaitable_35.py +99 -0
  644. wandb/vendor/promise-2.3.0/tests/test_dataloader_extra.py +65 -0
  645. wandb/vendor/promise-2.3.0/tests/test_extra.py +670 -0
  646. wandb/vendor/promise-2.3.0/tests/test_issues.py +132 -0
  647. wandb/vendor/promise-2.3.0/tests/test_promise_list.py +70 -0
  648. wandb/vendor/promise-2.3.0/tests/test_spec.py +584 -0
  649. wandb/vendor/promise-2.3.0/tests/test_thread_safety.py +115 -0
  650. wandb/vendor/promise-2.3.0/tests/utils.py +3 -0
  651. wandb/vendor/promise-2.3.0/wandb_promise/__init__.py +38 -0
  652. wandb/vendor/promise-2.3.0/wandb_promise/async_.py +135 -0
  653. wandb/vendor/promise-2.3.0/wandb_promise/compat.py +32 -0
  654. wandb/vendor/promise-2.3.0/wandb_promise/dataloader.py +326 -0
  655. wandb/vendor/promise-2.3.0/wandb_promise/iterate_promise.py +12 -0
  656. wandb/vendor/promise-2.3.0/wandb_promise/promise.py +848 -0
  657. wandb/vendor/promise-2.3.0/wandb_promise/promise_list.py +151 -0
  658. wandb/vendor/promise-2.3.0/wandb_promise/pyutils/__init__.py +0 -0
  659. wandb/vendor/promise-2.3.0/wandb_promise/pyutils/version.py +83 -0
  660. wandb/vendor/promise-2.3.0/wandb_promise/schedulers/__init__.py +0 -0
  661. wandb/vendor/promise-2.3.0/wandb_promise/schedulers/asyncio.py +22 -0
  662. wandb/vendor/promise-2.3.0/wandb_promise/schedulers/gevent.py +21 -0
  663. wandb/vendor/promise-2.3.0/wandb_promise/schedulers/immediate.py +27 -0
  664. wandb/vendor/promise-2.3.0/wandb_promise/schedulers/thread.py +18 -0
  665. wandb/vendor/promise-2.3.0/wandb_promise/utils.py +56 -0
  666. wandb/vendor/pygments/__init__.py +90 -0
  667. wandb/vendor/pygments/cmdline.py +568 -0
  668. wandb/vendor/pygments/console.py +74 -0
  669. wandb/vendor/pygments/filter.py +74 -0
  670. wandb/vendor/pygments/filters/__init__.py +350 -0
  671. wandb/vendor/pygments/formatter.py +95 -0
  672. wandb/vendor/pygments/formatters/__init__.py +153 -0
  673. wandb/vendor/pygments/formatters/_mapping.py +85 -0
  674. wandb/vendor/pygments/formatters/bbcode.py +109 -0
  675. wandb/vendor/pygments/formatters/html.py +851 -0
  676. wandb/vendor/pygments/formatters/img.py +600 -0
  677. wandb/vendor/pygments/formatters/irc.py +182 -0
  678. wandb/vendor/pygments/formatters/latex.py +482 -0
  679. wandb/vendor/pygments/formatters/other.py +160 -0
  680. wandb/vendor/pygments/formatters/rtf.py +147 -0
  681. wandb/vendor/pygments/formatters/svg.py +153 -0
  682. wandb/vendor/pygments/formatters/terminal.py +136 -0
  683. wandb/vendor/pygments/formatters/terminal256.py +309 -0
  684. wandb/vendor/pygments/lexer.py +871 -0
  685. wandb/vendor/pygments/lexers/__init__.py +329 -0
  686. wandb/vendor/pygments/lexers/_asy_builtins.py +1645 -0
  687. wandb/vendor/pygments/lexers/_cl_builtins.py +232 -0
  688. wandb/vendor/pygments/lexers/_cocoa_builtins.py +72 -0
  689. wandb/vendor/pygments/lexers/_csound_builtins.py +1346 -0
  690. wandb/vendor/pygments/lexers/_lasso_builtins.py +5327 -0
  691. wandb/vendor/pygments/lexers/_lua_builtins.py +295 -0
  692. wandb/vendor/pygments/lexers/_mapping.py +500 -0
  693. wandb/vendor/pygments/lexers/_mql_builtins.py +1172 -0
  694. wandb/vendor/pygments/lexers/_openedge_builtins.py +2547 -0
  695. wandb/vendor/pygments/lexers/_php_builtins.py +4756 -0
  696. wandb/vendor/pygments/lexers/_postgres_builtins.py +621 -0
  697. wandb/vendor/pygments/lexers/_scilab_builtins.py +3094 -0
  698. wandb/vendor/pygments/lexers/_sourcemod_builtins.py +1163 -0
  699. wandb/vendor/pygments/lexers/_stan_builtins.py +532 -0
  700. wandb/vendor/pygments/lexers/_stata_builtins.py +419 -0
  701. wandb/vendor/pygments/lexers/_tsql_builtins.py +1004 -0
  702. wandb/vendor/pygments/lexers/_vim_builtins.py +1939 -0
  703. wandb/vendor/pygments/lexers/actionscript.py +240 -0
  704. wandb/vendor/pygments/lexers/agile.py +24 -0
  705. wandb/vendor/pygments/lexers/algebra.py +221 -0
  706. wandb/vendor/pygments/lexers/ambient.py +76 -0
  707. wandb/vendor/pygments/lexers/ampl.py +87 -0
  708. wandb/vendor/pygments/lexers/apl.py +101 -0
  709. wandb/vendor/pygments/lexers/archetype.py +318 -0
  710. wandb/vendor/pygments/lexers/asm.py +641 -0
  711. wandb/vendor/pygments/lexers/automation.py +374 -0
  712. wandb/vendor/pygments/lexers/basic.py +500 -0
  713. wandb/vendor/pygments/lexers/bibtex.py +160 -0
  714. wandb/vendor/pygments/lexers/business.py +612 -0
  715. wandb/vendor/pygments/lexers/c_cpp.py +252 -0
  716. wandb/vendor/pygments/lexers/c_like.py +541 -0
  717. wandb/vendor/pygments/lexers/capnproto.py +78 -0
  718. wandb/vendor/pygments/lexers/chapel.py +102 -0
  719. wandb/vendor/pygments/lexers/clean.py +288 -0
  720. wandb/vendor/pygments/lexers/compiled.py +34 -0
  721. wandb/vendor/pygments/lexers/configs.py +833 -0
  722. wandb/vendor/pygments/lexers/console.py +114 -0
  723. wandb/vendor/pygments/lexers/crystal.py +393 -0
  724. wandb/vendor/pygments/lexers/csound.py +366 -0
  725. wandb/vendor/pygments/lexers/css.py +689 -0
  726. wandb/vendor/pygments/lexers/d.py +251 -0
  727. wandb/vendor/pygments/lexers/dalvik.py +125 -0
  728. wandb/vendor/pygments/lexers/data.py +555 -0
  729. wandb/vendor/pygments/lexers/diff.py +165 -0
  730. wandb/vendor/pygments/lexers/dotnet.py +691 -0
  731. wandb/vendor/pygments/lexers/dsls.py +878 -0
  732. wandb/vendor/pygments/lexers/dylan.py +289 -0
  733. wandb/vendor/pygments/lexers/ecl.py +125 -0
  734. wandb/vendor/pygments/lexers/eiffel.py +65 -0
  735. wandb/vendor/pygments/lexers/elm.py +121 -0
  736. wandb/vendor/pygments/lexers/erlang.py +533 -0
  737. wandb/vendor/pygments/lexers/esoteric.py +277 -0
  738. wandb/vendor/pygments/lexers/ezhil.py +69 -0
  739. wandb/vendor/pygments/lexers/factor.py +344 -0
  740. wandb/vendor/pygments/lexers/fantom.py +250 -0
  741. wandb/vendor/pygments/lexers/felix.py +273 -0
  742. wandb/vendor/pygments/lexers/forth.py +177 -0
  743. wandb/vendor/pygments/lexers/fortran.py +205 -0
  744. wandb/vendor/pygments/lexers/foxpro.py +428 -0
  745. wandb/vendor/pygments/lexers/functional.py +21 -0
  746. wandb/vendor/pygments/lexers/go.py +101 -0
  747. wandb/vendor/pygments/lexers/grammar_notation.py +213 -0
  748. wandb/vendor/pygments/lexers/graph.py +80 -0
  749. wandb/vendor/pygments/lexers/graphics.py +553 -0
  750. wandb/vendor/pygments/lexers/haskell.py +843 -0
  751. wandb/vendor/pygments/lexers/haxe.py +936 -0
  752. wandb/vendor/pygments/lexers/hdl.py +382 -0
  753. wandb/vendor/pygments/lexers/hexdump.py +103 -0
  754. wandb/vendor/pygments/lexers/html.py +602 -0
  755. wandb/vendor/pygments/lexers/idl.py +270 -0
  756. wandb/vendor/pygments/lexers/igor.py +288 -0
  757. wandb/vendor/pygments/lexers/inferno.py +96 -0
  758. wandb/vendor/pygments/lexers/installers.py +322 -0
  759. wandb/vendor/pygments/lexers/int_fiction.py +1343 -0
  760. wandb/vendor/pygments/lexers/iolang.py +63 -0
  761. wandb/vendor/pygments/lexers/j.py +146 -0
  762. wandb/vendor/pygments/lexers/javascript.py +1525 -0
  763. wandb/vendor/pygments/lexers/julia.py +333 -0
  764. wandb/vendor/pygments/lexers/jvm.py +1573 -0
  765. wandb/vendor/pygments/lexers/lisp.py +2621 -0
  766. wandb/vendor/pygments/lexers/make.py +202 -0
  767. wandb/vendor/pygments/lexers/markup.py +595 -0
  768. wandb/vendor/pygments/lexers/math.py +21 -0
  769. wandb/vendor/pygments/lexers/matlab.py +663 -0
  770. wandb/vendor/pygments/lexers/ml.py +769 -0
  771. wandb/vendor/pygments/lexers/modeling.py +358 -0
  772. wandb/vendor/pygments/lexers/modula2.py +1561 -0
  773. wandb/vendor/pygments/lexers/monte.py +204 -0
  774. wandb/vendor/pygments/lexers/ncl.py +894 -0
  775. wandb/vendor/pygments/lexers/nimrod.py +159 -0
  776. wandb/vendor/pygments/lexers/nit.py +64 -0
  777. wandb/vendor/pygments/lexers/nix.py +136 -0
  778. wandb/vendor/pygments/lexers/oberon.py +105 -0
  779. wandb/vendor/pygments/lexers/objective.py +504 -0
  780. wandb/vendor/pygments/lexers/ooc.py +85 -0
  781. wandb/vendor/pygments/lexers/other.py +41 -0
  782. wandb/vendor/pygments/lexers/parasail.py +79 -0
  783. wandb/vendor/pygments/lexers/parsers.py +835 -0
  784. wandb/vendor/pygments/lexers/pascal.py +644 -0
  785. wandb/vendor/pygments/lexers/pawn.py +199 -0
  786. wandb/vendor/pygments/lexers/perl.py +620 -0
  787. wandb/vendor/pygments/lexers/php.py +267 -0
  788. wandb/vendor/pygments/lexers/praat.py +294 -0
  789. wandb/vendor/pygments/lexers/prolog.py +306 -0
  790. wandb/vendor/pygments/lexers/python.py +939 -0
  791. wandb/vendor/pygments/lexers/qvt.py +152 -0
  792. wandb/vendor/pygments/lexers/r.py +453 -0
  793. wandb/vendor/pygments/lexers/rdf.py +270 -0
  794. wandb/vendor/pygments/lexers/rebol.py +431 -0
  795. wandb/vendor/pygments/lexers/resource.py +85 -0
  796. wandb/vendor/pygments/lexers/rnc.py +67 -0
  797. wandb/vendor/pygments/lexers/roboconf.py +82 -0
  798. wandb/vendor/pygments/lexers/robotframework.py +560 -0
  799. wandb/vendor/pygments/lexers/ruby.py +519 -0
  800. wandb/vendor/pygments/lexers/rust.py +220 -0
  801. wandb/vendor/pygments/lexers/sas.py +228 -0
  802. wandb/vendor/pygments/lexers/scripting.py +1222 -0
  803. wandb/vendor/pygments/lexers/shell.py +794 -0
  804. wandb/vendor/pygments/lexers/smalltalk.py +195 -0
  805. wandb/vendor/pygments/lexers/smv.py +79 -0
  806. wandb/vendor/pygments/lexers/snobol.py +83 -0
  807. wandb/vendor/pygments/lexers/special.py +103 -0
  808. wandb/vendor/pygments/lexers/sql.py +681 -0
  809. wandb/vendor/pygments/lexers/stata.py +108 -0
  810. wandb/vendor/pygments/lexers/supercollider.py +90 -0
  811. wandb/vendor/pygments/lexers/tcl.py +145 -0
  812. wandb/vendor/pygments/lexers/templates.py +2283 -0
  813. wandb/vendor/pygments/lexers/testing.py +207 -0
  814. wandb/vendor/pygments/lexers/text.py +25 -0
  815. wandb/vendor/pygments/lexers/textedit.py +169 -0
  816. wandb/vendor/pygments/lexers/textfmts.py +297 -0
  817. wandb/vendor/pygments/lexers/theorem.py +458 -0
  818. wandb/vendor/pygments/lexers/trafficscript.py +54 -0
  819. wandb/vendor/pygments/lexers/typoscript.py +226 -0
  820. wandb/vendor/pygments/lexers/urbi.py +133 -0
  821. wandb/vendor/pygments/lexers/varnish.py +190 -0
  822. wandb/vendor/pygments/lexers/verification.py +111 -0
  823. wandb/vendor/pygments/lexers/web.py +24 -0
  824. wandb/vendor/pygments/lexers/webmisc.py +988 -0
  825. wandb/vendor/pygments/lexers/whiley.py +116 -0
  826. wandb/vendor/pygments/lexers/x10.py +69 -0
  827. wandb/vendor/pygments/modeline.py +44 -0
  828. wandb/vendor/pygments/plugin.py +68 -0
  829. wandb/vendor/pygments/regexopt.py +92 -0
  830. wandb/vendor/pygments/scanner.py +105 -0
  831. wandb/vendor/pygments/sphinxext.py +158 -0
  832. wandb/vendor/pygments/style.py +155 -0
  833. wandb/vendor/pygments/styles/__init__.py +80 -0
  834. wandb/vendor/pygments/styles/abap.py +29 -0
  835. wandb/vendor/pygments/styles/algol.py +63 -0
  836. wandb/vendor/pygments/styles/algol_nu.py +63 -0
  837. wandb/vendor/pygments/styles/arduino.py +98 -0
  838. wandb/vendor/pygments/styles/autumn.py +65 -0
  839. wandb/vendor/pygments/styles/borland.py +51 -0
  840. wandb/vendor/pygments/styles/bw.py +49 -0
  841. wandb/vendor/pygments/styles/colorful.py +81 -0
  842. wandb/vendor/pygments/styles/default.py +73 -0
  843. wandb/vendor/pygments/styles/emacs.py +72 -0
  844. wandb/vendor/pygments/styles/friendly.py +72 -0
  845. wandb/vendor/pygments/styles/fruity.py +42 -0
  846. wandb/vendor/pygments/styles/igor.py +29 -0
  847. wandb/vendor/pygments/styles/lovelace.py +97 -0
  848. wandb/vendor/pygments/styles/manni.py +75 -0
  849. wandb/vendor/pygments/styles/monokai.py +106 -0
  850. wandb/vendor/pygments/styles/murphy.py +80 -0
  851. wandb/vendor/pygments/styles/native.py +65 -0
  852. wandb/vendor/pygments/styles/paraiso_dark.py +125 -0
  853. wandb/vendor/pygments/styles/paraiso_light.py +125 -0
  854. wandb/vendor/pygments/styles/pastie.py +75 -0
  855. wandb/vendor/pygments/styles/perldoc.py +69 -0
  856. wandb/vendor/pygments/styles/rainbow_dash.py +89 -0
  857. wandb/vendor/pygments/styles/rrt.py +33 -0
  858. wandb/vendor/pygments/styles/sas.py +44 -0
  859. wandb/vendor/pygments/styles/stata.py +40 -0
  860. wandb/vendor/pygments/styles/tango.py +141 -0
  861. wandb/vendor/pygments/styles/trac.py +63 -0
  862. wandb/vendor/pygments/styles/vim.py +63 -0
  863. wandb/vendor/pygments/styles/vs.py +38 -0
  864. wandb/vendor/pygments/styles/xcode.py +51 -0
  865. wandb/vendor/pygments/token.py +213 -0
  866. wandb/vendor/pygments/unistring.py +217 -0
  867. wandb/vendor/pygments/util.py +388 -0
  868. wandb/vendor/watchdog_0_9_0/wandb_watchdog/__init__.py +17 -0
  869. wandb/vendor/watchdog_0_9_0/wandb_watchdog/events.py +615 -0
  870. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/__init__.py +98 -0
  871. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/api.py +369 -0
  872. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/fsevents.py +172 -0
  873. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/fsevents2.py +239 -0
  874. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/inotify.py +218 -0
  875. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/inotify_buffer.py +81 -0
  876. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/inotify_c.py +575 -0
  877. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/kqueue.py +730 -0
  878. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/polling.py +145 -0
  879. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/read_directory_changes.py +133 -0
  880. wandb/vendor/watchdog_0_9_0/wandb_watchdog/observers/winapi.py +348 -0
  881. wandb/vendor/watchdog_0_9_0/wandb_watchdog/patterns.py +265 -0
  882. wandb/vendor/watchdog_0_9_0/wandb_watchdog/tricks/__init__.py +174 -0
  883. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/__init__.py +151 -0
  884. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/bricks.py +249 -0
  885. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/compat.py +29 -0
  886. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/decorators.py +198 -0
  887. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/delayed_queue.py +88 -0
  888. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/dirsnapshot.py +293 -0
  889. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/echo.py +157 -0
  890. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/event_backport.py +41 -0
  891. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/importlib2.py +40 -0
  892. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/platform.py +57 -0
  893. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/unicode_paths.py +64 -0
  894. wandb/vendor/watchdog_0_9_0/wandb_watchdog/utils/win32stat.py +123 -0
  895. wandb/vendor/watchdog_0_9_0/wandb_watchdog/version.py +28 -0
  896. wandb/vendor/watchdog_0_9_0/wandb_watchdog/watchmedo.py +577 -0
  897. wandb/wandb_agent.py +580 -0
  898. wandb/wandb_controller.py +719 -0
  899. wandb/wandb_run.py +8 -0
  900. wandb-0.21.2.dist-info/METADATA +223 -0
  901. wandb-0.21.2.dist-info/RECORD +904 -0
  902. wandb-0.21.2.dist-info/WHEEL +4 -0
  903. wandb-0.21.2.dist-info/entry_points.txt +3 -0
  904. wandb-0.21.2.dist-info/licenses/LICENSE +21 -0
@@ -0,0 +1,931 @@
1
+ """Implementation of launch agent."""
2
+
3
+ import asyncio
4
+ import copy
5
+ import logging
6
+ import os
7
+ import pprint
8
+ import threading
9
+ import time
10
+ import traceback
11
+ from dataclasses import dataclass
12
+ from multiprocessing import Event
13
+ from typing import Any, Dict, List, Optional, Tuple, Union
14
+
15
+ import yaml
16
+
17
+ import wandb
18
+ from wandb.apis.internal import Api
19
+ from wandb.errors import CommError
20
+ from wandb.sdk.launch._launch_add import launch_add
21
+ from wandb.sdk.launch.runner.local_container import LocalSubmittedRun
22
+ from wandb.sdk.launch.runner.local_process import LocalProcessRunner
23
+ from wandb.sdk.launch.sweeps.scheduler import Scheduler
24
+ from wandb.sdk.launch.utils import LAUNCH_CONFIG_FILE, resolve_build_and_registry_config
25
+ from wandb.sdk.lib import runid
26
+
27
+ from .. import loader
28
+ from .._project_spec import LaunchProject
29
+ from ..errors import LaunchDockerError, LaunchError
30
+ from ..utils import (
31
+ LAUNCH_DEFAULT_PROJECT,
32
+ LOG_PREFIX,
33
+ PROJECT_SYNCHRONOUS,
34
+ event_loop_thread_exec,
35
+ )
36
+ from .job_status_tracker import JobAndRunStatusTracker
37
+ from .run_queue_item_file_saver import RunQueueItemFileSaver
38
+
39
+ AGENT_POLLING_INTERVAL = 10
40
+ RECEIVED_JOB_POLLING_INTERVAL = 0.0 # more frequent when we know we have jobs
41
+
42
+ AGENT_POLLING = "POLLING"
43
+ AGENT_RUNNING = "RUNNING"
44
+ AGENT_KILLED = "KILLED"
45
+
46
+ HIDDEN_AGENT_RUN_TYPE = "sweep-controller"
47
+
48
+ MAX_RESUME_COUNT = 5
49
+
50
+ RUN_INFO_GRACE_PERIOD = 60
51
+
52
+ DEFAULT_STOPPED_RUN_TIMEOUT = 60
53
+
54
+ DEFAULT_PRINT_INTERVAL = 5 * 60
55
+ VERBOSE_PRINT_INTERVAL = 20
56
+
57
+ _env_timeout = os.environ.get("WANDB_LAUNCH_START_TIMEOUT")
58
+ if _env_timeout:
59
+ try:
60
+ RUN_START_TIMEOUT = float(_env_timeout)
61
+ except ValueError:
62
+ raise LaunchError(
63
+ f"Invalid value for WANDB_LAUNCH_START_TIMEOUT: {_env_timeout}"
64
+ )
65
+ else:
66
+ RUN_START_TIMEOUT = 60 * 30 # default 30 minutes
67
+
68
+ _logger = logging.getLogger(__name__)
69
+
70
+
71
+ @dataclass
72
+ class JobSpecAndQueue:
73
+ job: Dict[str, Any]
74
+ queue: str
75
+
76
+
77
+ def _convert_access(access: str) -> str:
78
+ """Convert access string to a value accepted by wandb."""
79
+ access = access.upper()
80
+ assert access == "PROJECT" or access == "USER", (
81
+ "Queue access must be either project or user"
82
+ )
83
+ return access
84
+
85
+
86
+ def _max_from_config(
87
+ config: Dict[str, Any], key: str, default: int = 1
88
+ ) -> Union[int, float]:
89
+ """Get an integer from the config, or float.inf if -1.
90
+
91
+ Utility for parsing integers from the agent config with a default, infinity
92
+ handling, and integer parsing. Raises more informative error if parse error.
93
+ """
94
+ try:
95
+ val = config.get(key)
96
+ if val is None:
97
+ val = default
98
+ max_from_config = int(val)
99
+ except ValueError as e:
100
+ raise LaunchError(
101
+ f"Error when parsing LaunchAgent config key: ['{key}': "
102
+ f"{config.get(key)}]. Error: {str(e)}"
103
+ )
104
+ if max_from_config == -1:
105
+ return float("inf")
106
+
107
+ if max_from_config < 0:
108
+ raise LaunchError(
109
+ f"Error when parsing LaunchAgent config key: ['{key}': "
110
+ f"{config.get(key)}]. Error: negative value."
111
+ )
112
+ return max_from_config
113
+
114
+
115
+ class InternalAgentLogger:
116
+ def __init__(self, verbosity=0):
117
+ self._print_to_terminal = verbosity >= 2
118
+
119
+ def error(self, message: str):
120
+ if self._print_to_terminal:
121
+ wandb.termerror(f"{LOG_PREFIX}{message}")
122
+ _logger.error(f"{LOG_PREFIX}{message}")
123
+
124
+ def warn(self, message: str):
125
+ if self._print_to_terminal:
126
+ wandb.termwarn(f"{LOG_PREFIX}{message}")
127
+ _logger.warning(f"{LOG_PREFIX}{message}")
128
+
129
+ def info(self, message: str):
130
+ if self._print_to_terminal:
131
+ wandb.termlog(f"{LOG_PREFIX}{message}")
132
+ _logger.info(f"{LOG_PREFIX}{message}")
133
+
134
+ def debug(self, message: str):
135
+ if self._print_to_terminal:
136
+ wandb.termlog(f"{LOG_PREFIX}{message}")
137
+ _logger.debug(f"{LOG_PREFIX}{message}")
138
+
139
+
140
+ def construct_agent_configs(
141
+ launch_config: Optional[Dict] = None,
142
+ build_config: Optional[Dict] = None,
143
+ ) -> Tuple[Optional[Dict[str, Any]], Dict[str, Any], Dict[str, Any]]:
144
+ registry_config = None
145
+ environment_config = None
146
+ if launch_config is not None:
147
+ build_config = launch_config.get("builder")
148
+ registry_config = launch_config.get("registry")
149
+
150
+ default_launch_config = None
151
+ if os.path.exists(os.path.expanduser(LAUNCH_CONFIG_FILE)):
152
+ with open(os.path.expanduser(LAUNCH_CONFIG_FILE)) as f:
153
+ default_launch_config = (
154
+ yaml.safe_load(f) or {}
155
+ ) # In case the config is empty, we want it to be {} instead of None.
156
+ environment_config = default_launch_config.get("environment")
157
+
158
+ build_config, registry_config = resolve_build_and_registry_config(
159
+ default_launch_config, build_config, registry_config
160
+ )
161
+
162
+ return environment_config, build_config, registry_config
163
+
164
+
165
+ class LaunchAgent:
166
+ """Launch agent class which polls run given run queues and launches runs for wandb launch."""
167
+
168
+ _instance = None
169
+
170
+ def __new__(cls, *args: Any, **kwargs: Any) -> "LaunchAgent":
171
+ """Create a new instance of the LaunchAgent.
172
+
173
+ This method ensures that only one instance of the LaunchAgent is created.
174
+ This is done so that information about the agent can be accessed from
175
+ elsewhere in the library.
176
+ """
177
+ if cls._instance is None:
178
+ cls._instance = super().__new__(cls)
179
+ return cls._instance
180
+
181
+ @classmethod
182
+ def name(cls) -> str:
183
+ """Return the name of the agent."""
184
+ if cls._instance is None:
185
+ raise LaunchError("LaunchAgent has not been initialized")
186
+ name = cls._instance._name
187
+ if isinstance(name, str):
188
+ return name
189
+ raise LaunchError(f"Found invalid name for agent {name}")
190
+
191
+ @classmethod
192
+ def initialized(cls) -> bool:
193
+ """Return whether the agent is initialized."""
194
+ return cls._instance is not None
195
+
196
+ def __init__(self, api: Api, config: Dict[str, Any]):
197
+ """Initialize a launch agent.
198
+
199
+ Arguments:
200
+ api: Api object to use for making requests to the backend.
201
+ config: Config dictionary for the agent.
202
+ """
203
+ self._entity = config["entity"]
204
+ self._project = LAUNCH_DEFAULT_PROJECT
205
+ self._api = api
206
+ self._base_url = self._api.settings().get("base_url")
207
+ self._ticks = 0
208
+ self._jobs: Dict[int, JobAndRunStatusTracker] = {}
209
+ self._jobs_lock = threading.Lock()
210
+ self._jobs_event = Event()
211
+ self._jobs_event.set()
212
+ self._cwd = os.getcwd()
213
+ self._namespace = runid.generate_id()
214
+ self._access = _convert_access("project")
215
+ self._max_jobs = _max_from_config(config, "max_jobs")
216
+ self._max_schedulers = _max_from_config(config, "max_schedulers")
217
+ self._secure_mode = config.get("secure_mode", False)
218
+ self._verbosity = config.get("verbosity", 0)
219
+ self._internal_logger = InternalAgentLogger(verbosity=self._verbosity)
220
+ self._last_status_print_time = 0.0
221
+ self.default_config: Dict[str, Any] = config
222
+ self._stopped_run_timeout = config.get(
223
+ "stopped_run_timeout", DEFAULT_STOPPED_RUN_TIMEOUT
224
+ )
225
+ self._known_warnings: List[str] = []
226
+
227
+ # Get agent version from env var if present, otherwise wandb version
228
+ self.version: str = "wandb@" + wandb.__version__
229
+ env_agent_version = os.environ.get("WANDB_AGENT_VERSION")
230
+ if env_agent_version and env_agent_version != "wandb-launch-agent":
231
+ self.version = env_agent_version
232
+
233
+ # serverside creation
234
+ self.gorilla_supports_agents = (
235
+ self._api.launch_agent_introspection() is not None
236
+ )
237
+ self._gorilla_supports_fail_run_queue_items = (
238
+ self._api.fail_run_queue_item_introspection()
239
+ )
240
+
241
+ self._queues: List[str] = config.get("queues", ["default"])
242
+
243
+ # remove project field from agent config before sending to back end
244
+ # because otherwise it shows up in the config in the UI and confuses users
245
+ sent_config = config.copy()
246
+ if "project" in sent_config:
247
+ del sent_config["project"]
248
+
249
+ create_response = self._api.create_launch_agent(
250
+ self._entity,
251
+ self._project,
252
+ self._queues,
253
+ sent_config,
254
+ self.version,
255
+ self.gorilla_supports_agents,
256
+ )
257
+ self._id = create_response["launchAgentId"]
258
+ if self._api.entity_is_team(self._entity):
259
+ wandb.termwarn(
260
+ f"{LOG_PREFIX}Agent is running on team entity ({self._entity}). Members of this team will be able to run code on this device."
261
+ )
262
+
263
+ agent_response = self._api.get_launch_agent(
264
+ self._id, self.gorilla_supports_agents
265
+ )
266
+ self._name = agent_response["name"]
267
+ self._init_agent_run()
268
+
269
+ def _is_scheduler_job(self, run_spec: Dict[str, Any]) -> bool:
270
+ """Determine whether a job/runSpec is a sweep scheduler."""
271
+ if not run_spec:
272
+ self._internal_logger.debug(
273
+ "Received runSpec in _is_scheduler_job that was empty"
274
+ )
275
+
276
+ if run_spec.get("uri") != Scheduler.PLACEHOLDER_URI:
277
+ return False
278
+
279
+ if run_spec.get("resource") == "local-process":
280
+ # Any job pushed to a run queue that has a scheduler uri is
281
+ # allowed to use local-process
282
+ if run_spec.get("job"):
283
+ return True
284
+
285
+ # If a scheduler is local-process and run through CLI, also
286
+ # confirm command is in format: [wandb scheduler <sweep>]
287
+ cmd = run_spec.get("overrides", {}).get("entry_point", [])
288
+ if len(cmd) < 3:
289
+ return False
290
+
291
+ if cmd[:2] != ["wandb", "scheduler"]:
292
+ return False
293
+
294
+ return True
295
+
296
+ async def fail_run_queue_item(
297
+ self,
298
+ run_queue_item_id: str,
299
+ message: str,
300
+ phase: str,
301
+ files: Optional[List[str]] = None,
302
+ ) -> None:
303
+ if self._gorilla_supports_fail_run_queue_items:
304
+ fail_rqi = event_loop_thread_exec(self._api.fail_run_queue_item)
305
+ await fail_rqi(run_queue_item_id, message, phase, files)
306
+
307
+ def _init_agent_run(self) -> None:
308
+ # TODO: has it been long enough that all backends support agents?
309
+ self._wandb_run = None
310
+
311
+ if self.gorilla_supports_agents:
312
+ settings = wandb.Settings(
313
+ silent=True, disable_git=True, disable_job_creation=True
314
+ )
315
+ self._wandb_run = wandb.init(
316
+ project=self._project,
317
+ entity=self._entity,
318
+ settings=settings,
319
+ id=self._name,
320
+ job_type=HIDDEN_AGENT_RUN_TYPE,
321
+ )
322
+
323
+ @property
324
+ def thread_ids(self) -> List[int]:
325
+ """Returns a list of keys running thread ids for the agent."""
326
+ with self._jobs_lock:
327
+ return list(self._jobs.keys())
328
+
329
+ @property
330
+ def num_running_schedulers(self) -> int:
331
+ """Return just the number of schedulers."""
332
+ with self._jobs_lock:
333
+ return len([x for x in self._jobs if self._jobs[x].is_scheduler])
334
+
335
+ @property
336
+ def num_running_jobs(self) -> int:
337
+ """Return the number of jobs not including schedulers."""
338
+ with self._jobs_lock:
339
+ return len([x for x in self._jobs if not self._jobs[x].is_scheduler])
340
+
341
+ async def pop_from_queue(self, queue: str) -> Any:
342
+ """Pops an item off the runqueue to run as a job.
343
+
344
+ Arguments:
345
+ queue: Queue to pop from.
346
+
347
+ Returns:
348
+ Item popped off the queue.
349
+
350
+ Raises:
351
+ Exception: if there is an error popping from the queue.
352
+ """
353
+ try:
354
+ pop = event_loop_thread_exec(self._api.pop_from_run_queue)
355
+ ups = await pop(
356
+ queue,
357
+ entity=self._entity,
358
+ project=self._project,
359
+ agent_id=self._id,
360
+ )
361
+ return ups
362
+ except Exception as e:
363
+ print("Exception:", e)
364
+ return None
365
+
366
+ def print_status(self) -> None:
367
+ """Prints the current status of the agent."""
368
+ self._last_status_print_time = time.time()
369
+ output_str = "agent "
370
+ if self._name:
371
+ output_str += f"{self._name} "
372
+ if self.num_running_jobs < self._max_jobs:
373
+ output_str += f"polling on queues {','.join(self._queues)}, "
374
+ output_str += (
375
+ f"running {self.num_running_jobs} out of a maximum of {self._max_jobs} jobs"
376
+ )
377
+
378
+ wandb.termlog(f"{LOG_PREFIX}{output_str}")
379
+ if self.num_running_jobs > 0:
380
+ output_str += f": {','.join(str(job_id) for job_id in self.thread_ids)}"
381
+
382
+ _logger.info(output_str)
383
+
384
+ async def update_status(self, status: str) -> None:
385
+ """Update the status of the agent.
386
+
387
+ Arguments:
388
+ status: Status to update the agent to.
389
+ """
390
+ _update_status = event_loop_thread_exec(self._api.update_launch_agent_status)
391
+ update_ret = await _update_status(
392
+ self._id, status, self.gorilla_supports_agents
393
+ )
394
+ if not update_ret["success"]:
395
+ wandb.termerror(f"{LOG_PREFIX}Failed to update agent status to {status}")
396
+
397
+ def _check_run_exists_and_inited(
398
+ self, entity: str, project: str, run_id: str, rqi_id: str
399
+ ) -> bool:
400
+ """Checks the stateof the run to ensure it has been inited. Note this will not behave well with resuming."""
401
+ # Checks the _wandb key in the run config for the run queue item id. If it exists, the
402
+ # submitted run definitely called init. Falls back to checking state of run.
403
+ # TODO: handle resuming runs
404
+
405
+ # Sweep runs exist but are in pending state, normal launch runs won't exist
406
+ # so will raise a CommError.
407
+ try:
408
+ run_state = self._api.get_run_state(entity, project, run_id)
409
+ if run_state.lower() != "pending":
410
+ return True
411
+ except CommError:
412
+ self._internal_logger.info(
413
+ f"Run {entity}/{project}/{run_id} with rqi id: {rqi_id} did not have associated run",
414
+ )
415
+ return False
416
+
417
+ async def finish_thread_id(
418
+ self,
419
+ thread_id: int,
420
+ exception: Optional[Union[Exception, LaunchDockerError]] = None,
421
+ ) -> None:
422
+ """Removes the job from our list for now."""
423
+ with self._jobs_lock:
424
+ job_and_run_status = self._jobs[thread_id]
425
+
426
+ if (
427
+ job_and_run_status.entity is not None
428
+ and job_and_run_status.entity != self._entity
429
+ ):
430
+ self._internal_logger.info(
431
+ "Skipping check for completed run status because run is on a different entity than agent",
432
+ )
433
+ elif exception is not None:
434
+ tb_str = traceback.format_exception(
435
+ type(exception), value=exception, tb=exception.__traceback__
436
+ )
437
+ fnames = job_and_run_status.saver.save_contents(
438
+ "".join(tb_str), "error.log", "error"
439
+ )
440
+ await self.fail_run_queue_item(
441
+ job_and_run_status.run_queue_item_id,
442
+ str(exception),
443
+ job_and_run_status.err_stage,
444
+ fnames,
445
+ )
446
+ elif job_and_run_status.project is None or job_and_run_status.run_id is None:
447
+ self._internal_logger.info(
448
+ f"called finish_thread_id on thread whose tracker has no project or run id. RunQueueItemID: {job_and_run_status.run_queue_item_id}",
449
+ )
450
+ wandb.termerror(
451
+ "Missing project or run id on thread called finish thread id"
452
+ )
453
+ await self.fail_run_queue_item(
454
+ job_and_run_status.run_queue_item_id,
455
+ "submitted job was finished without assigned project or run id",
456
+ "agent",
457
+ )
458
+ elif job_and_run_status.run is not None:
459
+ called_init = False
460
+ # We do some weird stuff here getting run info to check for a
461
+ # created in run in W&B.
462
+ #
463
+ # We retry for 60 seconds with an exponential backoff in case
464
+ # upsert run is taking a while.
465
+ logs = None
466
+ interval = 1
467
+ while True:
468
+ called_init = self._check_run_exists_and_inited(
469
+ self._entity,
470
+ job_and_run_status.project,
471
+ job_and_run_status.run_id,
472
+ job_and_run_status.run_queue_item_id,
473
+ )
474
+ if called_init or interval > RUN_INFO_GRACE_PERIOD:
475
+ break
476
+ if not called_init:
477
+ # Fetch the logs now if we don't get run info on the
478
+ # first try, in case the logs are cleaned from the runner
479
+ # environment (e.g. k8s) during the run info grace period.
480
+ if interval == 1:
481
+ logs = await job_and_run_status.run.get_logs()
482
+ await asyncio.sleep(interval)
483
+ interval *= 2
484
+ if not called_init:
485
+ fnames = None
486
+ if job_and_run_status.completed_status == "finished":
487
+ _msg = "The submitted job exited successfully but failed to call wandb.init"
488
+ else:
489
+ _msg = "The submitted run was not successfully started"
490
+ if logs:
491
+ fnames = job_and_run_status.saver.save_contents(
492
+ logs, "error.log", "error"
493
+ )
494
+ await self.fail_run_queue_item(
495
+ job_and_run_status.run_queue_item_id, _msg, "run", fnames
496
+ )
497
+ else:
498
+ self._internal_logger.info(
499
+ f"Finish thread id {thread_id} had no exception and no run"
500
+ )
501
+ wandb._sentry.exception(
502
+ "launch agent called finish thread id on thread without run or exception"
503
+ )
504
+
505
+ # TODO: keep logs or something for the finished jobs
506
+ with self._jobs_lock:
507
+ del self._jobs[thread_id]
508
+
509
+ # update status back to polling if no jobs are running
510
+ if len(self.thread_ids) == 0:
511
+ await self.update_status(AGENT_POLLING)
512
+
513
+ async def run_job(
514
+ self, job: Dict[str, Any], queue: str, file_saver: RunQueueItemFileSaver
515
+ ) -> None:
516
+ """Set up project and run the job.
517
+
518
+ Arguments:
519
+ job: Job to run.
520
+ """
521
+ job_copy = copy.deepcopy(job)
522
+ if "runSpec" in job_copy and "_wandb_api_key" in job_copy["runSpec"]:
523
+ job_copy["runSpec"]["_wandb_api_key"] = "<redacted>"
524
+
525
+ _msg = f"{LOG_PREFIX}Launch agent received job:\n{pprint.pformat(job_copy)}\n"
526
+ wandb.termlog(_msg)
527
+ _logger.info(_msg)
528
+ # update agent status
529
+ await self.update_status(AGENT_RUNNING)
530
+
531
+ # parse job
532
+ self._internal_logger.info("Parsing launch spec")
533
+ launch_spec = job["runSpec"]
534
+
535
+ # Abort if this job attempts to override secure mode
536
+ self._assert_secure(launch_spec)
537
+ job_tracker = JobAndRunStatusTracker(job["runQueueItemId"], queue, file_saver)
538
+
539
+ asyncio.create_task(
540
+ self.task_run_job(
541
+ launch_spec,
542
+ job,
543
+ self.default_config,
544
+ self._api,
545
+ job_tracker,
546
+ )
547
+ )
548
+
549
+ def _assert_secure(self, launch_spec: Dict[str, Any]) -> None:
550
+ """If secure mode is set, make sure no vulnerable keys are overridden."""
551
+ if not self._secure_mode:
552
+ return
553
+ k8s_config = launch_spec.get("resource_args", {}).get("kubernetes", {})
554
+
555
+ pod_secure_keys = ["hostPID", "hostIPC", "hostNetwork", "initContainers"]
556
+ pod_spec = k8s_config.get("spec", {}).get("template", {}).get("spec", {})
557
+ for key in pod_secure_keys:
558
+ if key in pod_spec:
559
+ raise ValueError(
560
+ f'This agent is configured to lock "{key}" in pod spec '
561
+ "but the job specification attempts to override it."
562
+ )
563
+
564
+ container_specs = pod_spec.get("containers", [])
565
+ for container_spec in container_specs:
566
+ if "command" in container_spec:
567
+ raise ValueError(
568
+ 'This agent is configured to lock "command" in container spec '
569
+ "but the job specification attempts to override it."
570
+ )
571
+
572
+ if launch_spec.get("overrides", {}).get("entry_point"):
573
+ raise ValueError(
574
+ 'This agent is configured to lock the "entrypoint" override '
575
+ "but the job specification attempts to override it."
576
+ )
577
+
578
+ async def loop(self) -> None:
579
+ """Loop infinitely to poll for jobs and run them.
580
+
581
+ Raises:
582
+ KeyboardInterrupt: if the agent is requested to stop.
583
+ """
584
+ self.print_status()
585
+ if self._verbosity == 0:
586
+ print_interval = DEFAULT_PRINT_INTERVAL
587
+ else:
588
+ print_interval = VERBOSE_PRINT_INTERVAL
589
+ try:
590
+ while True:
591
+ job = None
592
+ self._ticks += 1
593
+ agent_response = self._api.get_launch_agent(
594
+ self._id, self.gorilla_supports_agents
595
+ )
596
+ if agent_response["stopPolling"]:
597
+ # shutdown process and all jobs if requested from ui
598
+ raise KeyboardInterrupt # noqa: TRY301
599
+ if self.num_running_jobs < self._max_jobs:
600
+ # only check for new jobs if we're not at max
601
+ job_and_queue = await self.get_job_and_queue()
602
+ # these will either both be None, or neither will be None
603
+ if job_and_queue is not None:
604
+ job = job_and_queue.job
605
+ queue = job_and_queue.queue
606
+ try:
607
+ file_saver = RunQueueItemFileSaver(
608
+ self._wandb_run, job["runQueueItemId"]
609
+ )
610
+ if self._is_scheduler_job(job.get("runSpec", {})):
611
+ # If job is a scheduler, and we are already at the cap, ignore,
612
+ # don't ack, and it will be pushed back onto the queue in 1 min
613
+ if self.num_running_schedulers >= self._max_schedulers:
614
+ wandb.termwarn(
615
+ f"{LOG_PREFIX}Agent already running the maximum number "
616
+ f"of sweep schedulers: {self._max_schedulers}. To set "
617
+ "this value use `max_schedulers` key in the agent config"
618
+ )
619
+ continue
620
+ await self.run_job(job, queue, file_saver)
621
+ except Exception as e:
622
+ wandb.termerror(
623
+ f"{LOG_PREFIX}Error running job: {traceback.format_exc()}"
624
+ )
625
+ wandb._sentry.exception(e)
626
+
627
+ # always the first phase, because we only enter phase 2 within the thread
628
+ files = file_saver.save_contents(
629
+ contents=traceback.format_exc(),
630
+ fname="error.log",
631
+ file_sub_type="error",
632
+ )
633
+ await self.fail_run_queue_item(
634
+ run_queue_item_id=job["runQueueItemId"],
635
+ message=str(e),
636
+ phase="agent",
637
+ files=files,
638
+ )
639
+
640
+ if self._ticks % 2 == 0:
641
+ if len(self.thread_ids) == 0:
642
+ await self.update_status(AGENT_POLLING)
643
+ else:
644
+ await self.update_status(AGENT_RUNNING)
645
+ if time.time() - self._last_status_print_time > print_interval:
646
+ self.print_status()
647
+
648
+ if self.num_running_jobs == self._max_jobs or job is None:
649
+ # all threads busy or did not receive job
650
+ await asyncio.sleep(AGENT_POLLING_INTERVAL)
651
+ else:
652
+ await asyncio.sleep(RECEIVED_JOB_POLLING_INTERVAL)
653
+
654
+ except KeyboardInterrupt:
655
+ await self.update_status(AGENT_KILLED)
656
+ wandb.termlog(f"{LOG_PREFIX}Shutting down, active jobs:")
657
+ self.print_status()
658
+ finally:
659
+ self._jobs_event.clear()
660
+
661
+ # Threaded functions
662
+ async def task_run_job(
663
+ self,
664
+ launch_spec: Dict[str, Any],
665
+ job: Dict[str, Any],
666
+ default_config: Dict[str, Any],
667
+ api: Api,
668
+ job_tracker: JobAndRunStatusTracker,
669
+ ) -> None:
670
+ rqi_id = job["runQueueItemId"]
671
+ assert rqi_id
672
+ exception: Optional[Union[LaunchDockerError, Exception]] = None
673
+ try:
674
+ with self._jobs_lock:
675
+ self._jobs[rqi_id] = job_tracker
676
+ await self._task_run_job(
677
+ launch_spec, job, default_config, api, rqi_id, job_tracker
678
+ )
679
+ except LaunchDockerError as e:
680
+ wandb.termerror(
681
+ f"{LOG_PREFIX}agent {self._name} encountered an issue while starting Docker, see above output for details."
682
+ )
683
+ exception = e
684
+ wandb._sentry.exception(e)
685
+ except LaunchError as e:
686
+ wandb.termerror(f"{LOG_PREFIX}Error running job: {e}")
687
+ exception = e
688
+ wandb._sentry.exception(e)
689
+ except Exception as e:
690
+ wandb.termerror(f"{LOG_PREFIX}Error running job: {traceback.format_exc()}")
691
+ exception = e
692
+ wandb._sentry.exception(e)
693
+ finally:
694
+ await self.finish_thread_id(rqi_id, exception)
695
+
696
+ async def _task_run_job(
697
+ self,
698
+ launch_spec: Dict[str, Any],
699
+ job: Dict[str, Any],
700
+ default_config: Dict[str, Any],
701
+ api: Api,
702
+ thread_id: int,
703
+ job_tracker: JobAndRunStatusTracker,
704
+ ) -> None:
705
+ project = LaunchProject.from_spec(launch_spec, api)
706
+ self._set_queue_and_rqi_in_project(project, job, job_tracker.queue)
707
+ ack = event_loop_thread_exec(api.ack_run_queue_item)
708
+ await ack(job["runQueueItemId"], project.run_id)
709
+ # don't launch sweep runs if the sweep isn't healthy
710
+ await self.check_sweep_state(launch_spec, api)
711
+
712
+ job_tracker.update_run_info(project)
713
+ self._internal_logger.info("Fetching and validating project...")
714
+ project.fetch_and_validate_project()
715
+ self._internal_logger.info("Fetching resource...")
716
+ resource = launch_spec.get("resource") or "local-container"
717
+ backend_config: Dict[str, Any] = {
718
+ PROJECT_SYNCHRONOUS: False, # agent always runs async
719
+ }
720
+ self._internal_logger.info("Loading backend")
721
+ override_build_config = launch_spec.get("builder")
722
+
723
+ _, build_config, registry_config = construct_agent_configs(
724
+ default_config, override_build_config
725
+ )
726
+ image_uri = project.docker_image or project.job_base_image
727
+ entrypoint = project.get_job_entry_point()
728
+ environment = loader.environment_from_config(
729
+ default_config.get("environment", {})
730
+ )
731
+ registry = loader.registry_from_config(registry_config, environment)
732
+ builder = loader.builder_from_config(build_config, environment, registry)
733
+ backend = loader.runner_from_config(
734
+ resource, api, backend_config, environment, registry
735
+ )
736
+
737
+ if not (
738
+ project.docker_image
739
+ or project.job_base_image
740
+ or isinstance(backend, LocalProcessRunner)
741
+ ):
742
+ assert entrypoint is not None
743
+ image_uri = await builder.build_image(project, entrypoint, job_tracker)
744
+
745
+ self._internal_logger.info("Backend loaded...")
746
+ if isinstance(backend, LocalProcessRunner):
747
+ run = await backend.run(project, image_uri)
748
+ else:
749
+ assert image_uri
750
+ run = await backend.run(project, image_uri)
751
+ if self._is_scheduler_job(launch_spec):
752
+ with self._jobs_lock:
753
+ self._jobs[thread_id].is_scheduler = True
754
+ wandb.termlog(
755
+ f"{LOG_PREFIX}Preparing to run sweep scheduler "
756
+ f"({self.num_running_schedulers}/{self._max_schedulers})"
757
+ )
758
+
759
+ if not run:
760
+ with self._jobs_lock:
761
+ job_tracker.failed_to_start = True
762
+ return
763
+ with self._jobs_lock:
764
+ job_tracker.run = run
765
+ start_time = time.time()
766
+ stopped_time: Optional[float] = None
767
+ while self._jobs_event.is_set():
768
+ # If run has failed to start before timeout, kill it
769
+ state = (await run.get_status()).state
770
+ if state == "starting" and RUN_START_TIMEOUT > 0:
771
+ if time.time() - start_time > RUN_START_TIMEOUT:
772
+ await run.cancel()
773
+ raise LaunchError(
774
+ f"Run failed to start within {RUN_START_TIMEOUT} seconds. "
775
+ "If you want to increase this timeout, set WANDB_LAUNCH_START_TIMEOUT "
776
+ "to a larger value."
777
+ )
778
+ if await self._check_run_finished(job_tracker, launch_spec):
779
+ return
780
+ if await job_tracker.check_wandb_run_stopped(self._api):
781
+ if stopped_time is None:
782
+ stopped_time = time.time()
783
+ else:
784
+ if time.time() - stopped_time > self._stopped_run_timeout:
785
+ await run.cancel()
786
+ await asyncio.sleep(AGENT_POLLING_INTERVAL)
787
+
788
+ # temp: for local, kill all jobs. we don't yet have good handling for different
789
+ # types of runners in general
790
+ if isinstance(run, LocalSubmittedRun) and run._command_proc is not None:
791
+ run._command_proc.kill()
792
+
793
+ async def check_sweep_state(self, launch_spec: Dict[str, Any], api: Api) -> None:
794
+ """Check the state of a sweep before launching a run for the sweep."""
795
+ if launch_spec.get("sweep_id"):
796
+ try:
797
+ get_sweep_state = event_loop_thread_exec(api.get_sweep_state)
798
+ state = await get_sweep_state(
799
+ sweep=launch_spec["sweep_id"],
800
+ entity=launch_spec["entity"],
801
+ project=launch_spec["project"],
802
+ )
803
+ except Exception as e:
804
+ self._internal_logger.debug(f"Fetch sweep state error: {e}")
805
+ state = None
806
+
807
+ if state != "RUNNING" and state != "PAUSED":
808
+ raise LaunchError(
809
+ f"Launch agent picked up sweep job, but sweep ({launch_spec['sweep_id']}) was in a terminal state ({state})"
810
+ )
811
+
812
+ async def _check_run_finished(
813
+ self, job_tracker: JobAndRunStatusTracker, launch_spec: Dict[str, Any]
814
+ ) -> bool:
815
+ if job_tracker.completed_status:
816
+ return True
817
+
818
+ # the run can be done before the run has started
819
+ # but can also be none if the run failed to start
820
+ # so if there is no run, either the run hasn't started yet
821
+ # or it has failed
822
+ if job_tracker.run is None:
823
+ if job_tracker.failed_to_start:
824
+ return True
825
+ return False
826
+
827
+ known_error = False
828
+ try:
829
+ run = job_tracker.run
830
+ status = await run.get_status()
831
+ state = status.state
832
+
833
+ for warning in status.messages:
834
+ if warning not in self._known_warnings:
835
+ self._known_warnings.append(warning)
836
+ success = self._api.update_run_queue_item_warning(
837
+ job_tracker.run_queue_item_id,
838
+ warning,
839
+ "Kubernetes",
840
+ [],
841
+ )
842
+ if not success:
843
+ _logger.warning(
844
+ f"Error adding warning {warning} to run queue item {job_tracker.run_queue_item_id}"
845
+ )
846
+ self._known_warnings.remove(warning)
847
+
848
+ if state == "preempted" and job_tracker.entity == self._entity:
849
+ config = launch_spec.copy()
850
+ config["run_id"] = job_tracker.run_id
851
+ config["_resume_count"] = config.get("_resume_count", 0) + 1
852
+ with self._jobs_lock:
853
+ job_tracker.completed_status = state
854
+ if config["_resume_count"] > MAX_RESUME_COUNT:
855
+ wandb.termlog(
856
+ f"{LOG_PREFIX}Run {job_tracker.run_id} has already resumed {MAX_RESUME_COUNT} times."
857
+ )
858
+ return True
859
+ wandb.termlog(
860
+ f"{LOG_PREFIX}Run {job_tracker.run_id} was preempted, requeuing..."
861
+ )
862
+
863
+ if "sweep_id" in config:
864
+ # allow resumed runs from sweeps that have already completed by removing
865
+ # the sweep id before pushing to queue
866
+ del config["sweep_id"]
867
+
868
+ launch_add(
869
+ config=config,
870
+ project_queue=self._project,
871
+ queue_name=job_tracker.queue,
872
+ )
873
+ return True
874
+ # TODO change these statuses to an enum
875
+ if state in ["stopped", "failed", "finished", "preempted"]:
876
+ if job_tracker.is_scheduler:
877
+ wandb.termlog(f"{LOG_PREFIX}Scheduler finished with ID: {run.id}")
878
+ if state == "failed":
879
+ # on fail, update sweep state. scheduler run_id should == sweep_id
880
+ try:
881
+ self._api.set_sweep_state(
882
+ sweep=job_tracker.run_id,
883
+ entity=job_tracker.entity,
884
+ project=job_tracker.project,
885
+ state="CANCELED",
886
+ )
887
+ except Exception as e:
888
+ raise LaunchError(f"Failed to update sweep state: {e}")
889
+ else:
890
+ wandb.termlog(f"{LOG_PREFIX}Job finished with ID: {run.id}")
891
+ with self._jobs_lock:
892
+ job_tracker.completed_status = state
893
+ return True
894
+
895
+ return False
896
+ except LaunchError as e:
897
+ wandb.termerror(
898
+ f"{LOG_PREFIX}Terminating job {run.id} because it failed to start: {str(e)}"
899
+ )
900
+ known_error = True
901
+ with self._jobs_lock:
902
+ job_tracker.failed_to_start = True
903
+ # TODO: make get_status robust to errors for each runner, and handle them
904
+ except Exception as e:
905
+ wandb.termerror(f"{LOG_PREFIX}Error getting status for job {run.id}")
906
+ wandb.termerror(traceback.format_exc())
907
+ _logger.info("---")
908
+ _logger.info("Caught exception while getting status.")
909
+ _logger.info(f"Job ID: {run.id}")
910
+ _logger.info(traceback.format_exc())
911
+ _logger.info("---")
912
+ wandb._sentry.exception(e)
913
+ return known_error
914
+
915
+ async def get_job_and_queue(self) -> Optional[JobSpecAndQueue]:
916
+ for queue in self._queues:
917
+ job = await self.pop_from_queue(queue)
918
+ if job is not None:
919
+ self._queues.remove(queue)
920
+ self._queues.append(queue)
921
+ return JobSpecAndQueue(job, queue)
922
+ return None
923
+
924
+ def _set_queue_and_rqi_in_project(
925
+ self, project: LaunchProject, job: Dict[str, Any], queue: str
926
+ ) -> None:
927
+ project.queue_name = queue
928
+
929
+ # queue entity currently always matches the agent
930
+ project.queue_entity = self._entity
931
+ project.run_queue_item_id = job["runQueueItemId"]