wandb 0.18.3__py3-none-macosx_11_0_arm64.whl → 0.18.5__py3-none-macosx_11_0_arm64.whl

Sign up to get free protection for your applications and to get access to all the features.
Files changed (46) hide show
  1. wandb/__init__.py +16 -7
  2. wandb/__init__.pyi +96 -63
  3. wandb/analytics/sentry.py +91 -88
  4. wandb/apis/public/api.py +18 -4
  5. wandb/apis/public/runs.py +53 -2
  6. wandb/bin/gpu_stats +0 -0
  7. wandb/bin/wandb-core +0 -0
  8. wandb/cli/beta.py +178 -0
  9. wandb/cli/cli.py +5 -171
  10. wandb/data_types.py +3 -0
  11. wandb/env.py +74 -73
  12. wandb/errors/term.py +300 -43
  13. wandb/proto/v3/wandb_internal_pb2.py +263 -223
  14. wandb/proto/v3/wandb_server_pb2.py +57 -37
  15. wandb/proto/v3/wandb_settings_pb2.py +2 -2
  16. wandb/proto/v4/wandb_internal_pb2.py +226 -218
  17. wandb/proto/v4/wandb_server_pb2.py +41 -37
  18. wandb/proto/v4/wandb_settings_pb2.py +2 -2
  19. wandb/proto/v5/wandb_internal_pb2.py +226 -218
  20. wandb/proto/v5/wandb_server_pb2.py +41 -37
  21. wandb/proto/v5/wandb_settings_pb2.py +2 -2
  22. wandb/sdk/__init__.py +3 -3
  23. wandb/sdk/artifacts/_validators.py +41 -8
  24. wandb/sdk/artifacts/artifact.py +32 -1
  25. wandb/sdk/artifacts/artifact_file_cache.py +1 -2
  26. wandb/sdk/data_types/_dtypes.py +7 -3
  27. wandb/sdk/data_types/video.py +15 -6
  28. wandb/sdk/interface/interface.py +2 -0
  29. wandb/sdk/internal/internal_api.py +126 -5
  30. wandb/sdk/internal/sender.py +16 -3
  31. wandb/sdk/launch/inputs/internal.py +1 -1
  32. wandb/sdk/lib/module.py +12 -0
  33. wandb/sdk/lib/printer.py +291 -105
  34. wandb/sdk/lib/progress.py +274 -0
  35. wandb/sdk/service/streams.py +21 -11
  36. wandb/sdk/wandb_init.py +58 -54
  37. wandb/sdk/wandb_run.py +380 -454
  38. wandb/sdk/wandb_settings.py +2 -0
  39. wandb/sdk/wandb_watch.py +17 -11
  40. wandb/util.py +6 -2
  41. {wandb-0.18.3.dist-info → wandb-0.18.5.dist-info}/METADATA +4 -3
  42. {wandb-0.18.3.dist-info → wandb-0.18.5.dist-info}/RECORD +45 -43
  43. wandb/bin/apple_gpu_stats +0 -0
  44. {wandb-0.18.3.dist-info → wandb-0.18.5.dist-info}/WHEEL +0 -0
  45. {wandb-0.18.3.dist-info → wandb-0.18.5.dist-info}/entry_points.txt +0 -0
  46. {wandb-0.18.3.dist-info → wandb-0.18.5.dist-info}/licenses/LICENSE +0 -0
wandb/sdk/wandb_run.py CHANGED
@@ -1,3 +1,5 @@
1
+ from __future__ import annotations
2
+
1
3
  import _thread as thread
2
4
  import atexit
3
5
  import functools
@@ -18,20 +20,12 @@ from dataclasses import dataclass, field
18
20
  from datetime import datetime, timedelta, timezone
19
21
  from enum import IntEnum
20
22
  from types import TracebackType
21
- from typing import (
22
- TYPE_CHECKING,
23
- Any,
24
- Callable,
25
- Dict,
26
- List,
27
- NamedTuple,
28
- Optional,
29
- Sequence,
30
- TextIO,
31
- Tuple,
32
- Type,
33
- Union,
34
- )
23
+ from typing import TYPE_CHECKING, Any, Callable, NamedTuple, Sequence, TextIO
24
+
25
+ if sys.version_info < (3, 8):
26
+ from typing_extensions import Literal
27
+ else:
28
+ from typing import Literal
35
29
 
36
30
  import requests
37
31
 
@@ -69,7 +63,11 @@ from wandb.util import (
69
63
  )
70
64
 
71
65
  from . import wandb_config, wandb_metric, wandb_summary
72
- from .artifacts._validators import validate_aliases, validate_tags
66
+ from .artifacts._validators import (
67
+ is_artifact_registry_project,
68
+ validate_aliases,
69
+ validate_tags,
70
+ )
73
71
  from .data_types._dtypes import TypeRegistry
74
72
  from .interface.interface import FilesDict, GlobStr, InterfaceBase, PolicyName
75
73
  from .interface.summary_record import SummaryRecord
@@ -80,6 +78,8 @@ from .lib import (
80
78
  filesystem,
81
79
  ipython,
82
80
  module,
81
+ printer,
82
+ progress,
83
83
  proto_util,
84
84
  redirect,
85
85
  telemetry,
@@ -87,7 +87,6 @@ from .lib import (
87
87
  from .lib.exit_hooks import ExitHooks
88
88
  from .lib.gitlib import GitRepo
89
89
  from .lib.mailbox import MailboxError, MailboxHandle, MailboxProbe, MailboxProgress
90
- from .lib.printer import get_printer
91
90
  from .lib.proto_util import message_to_dict
92
91
  from .lib.reporting import Reporter
93
92
  from .lib.wburls import wburls
@@ -101,26 +100,27 @@ if TYPE_CHECKING:
101
100
  else:
102
101
  from typing_extensions import TypedDict
103
102
 
103
+ import torch # type: ignore [import-not-found]
104
+
104
105
  import wandb.apis.public
105
106
  import wandb.sdk.backend.backend
106
107
  import wandb.sdk.interface.interface_queue
108
+ from wandb.data_types import Table
107
109
  from wandb.proto.wandb_internal_pb2 import (
108
110
  GetSummaryResponse,
109
111
  InternalMessagesResponse,
110
112
  SampledHistoryResponse,
111
113
  )
112
114
 
113
- from .lib.printer import PrinterJupyter, PrinterTerm
114
-
115
115
  class GitSourceDict(TypedDict):
116
116
  remote: str
117
117
  commit: str
118
- entrypoint: List[str]
118
+ entrypoint: list[str]
119
119
  args: Sequence[str]
120
120
 
121
121
  class ArtifactSourceDict(TypedDict):
122
122
  artifact: str
123
- entrypoint: List[str]
123
+ entrypoint: list[str]
124
124
  args: Sequence[str]
125
125
 
126
126
  class ImageSourceDict(TypedDict):
@@ -130,10 +130,10 @@ if TYPE_CHECKING:
130
130
  class JobSourceDict(TypedDict, total=False):
131
131
  _version: str
132
132
  source_type: str
133
- source: Union[GitSourceDict, ArtifactSourceDict, ImageSourceDict]
134
- input_types: Dict[str, Any]
135
- output_types: Dict[str, Any]
136
- runtime: Optional[str]
133
+ source: GitSourceDict | ArtifactSourceDict | ImageSourceDict
134
+ input_types: dict[str, Any]
135
+ output_types: dict[str, Any]
136
+ runtime: str | None
137
137
 
138
138
 
139
139
  logger = logging.getLogger("wandb")
@@ -160,11 +160,11 @@ class RunStatusChecker:
160
160
  """
161
161
 
162
162
  _stop_status_lock: threading.Lock
163
- _stop_status_handle: Optional[MailboxHandle]
163
+ _stop_status_handle: MailboxHandle | None
164
164
  _network_status_lock: threading.Lock
165
- _network_status_handle: Optional[MailboxHandle]
165
+ _network_status_handle: MailboxHandle | None
166
166
  _internal_messages_lock: threading.Lock
167
- _internal_messages_handle: Optional[MailboxHandle]
167
+ _internal_messages_handle: MailboxHandle | None
168
168
 
169
169
  def __init__(
170
170
  self,
@@ -212,7 +212,7 @@ class RunStatusChecker:
212
212
  @staticmethod
213
213
  def _abandon_status_check(
214
214
  lock: threading.Lock,
215
- handle: Optional[MailboxHandle],
215
+ handle: MailboxHandle | None,
216
216
  ):
217
217
  with lock:
218
218
  if handle:
@@ -227,7 +227,7 @@ class RunStatusChecker:
227
227
  request: Any,
228
228
  process: Any,
229
229
  ) -> None:
230
- local_handle: Optional[MailboxHandle] = None
230
+ local_handle: MailboxHandle | None = None
231
231
  join_requested = False
232
232
  while not join_requested:
233
233
  time_probe = time.monotonic()
@@ -365,7 +365,7 @@ class _run_decorator: # noqa: N801
365
365
  @classmethod
366
366
  def _attach(cls, func: Callable) -> Callable:
367
367
  @functools.wraps(func)
368
- def wrapper(self: Type["Run"], *args: Any, **kwargs: Any) -> Any:
368
+ def wrapper(self: type[Run], *args: Any, **kwargs: Any) -> Any:
369
369
  # * `_attach_id` is only assigned in service hence for all non-service cases
370
370
  # it will be a passthrough.
371
371
  # * `_attach_pid` is only assigned in _init (using _attach_pid guarantees single attach):
@@ -398,7 +398,7 @@ class _run_decorator: # noqa: N801
398
398
  def _noop_on_finish(cls, message: str = "", only_warn: bool = False) -> Callable:
399
399
  def decorator_fn(func: Callable) -> Callable:
400
400
  @functools.wraps(func)
401
- def wrapper_fn(self: Type["Run"], *args: Any, **kwargs: Any) -> Any:
401
+ def wrapper_fn(self: type[Run], *args: Any, **kwargs: Any) -> Any:
402
402
  if not getattr(self, "_is_finished", False):
403
403
  return func(self, *args, **kwargs)
404
404
 
@@ -419,7 +419,7 @@ class _run_decorator: # noqa: N801
419
419
  @classmethod
420
420
  def _noop(cls, func: Callable) -> Callable:
421
421
  @functools.wraps(func)
422
- def wrapper(self: Type["Run"], *args: Any, **kwargs: Any) -> Any:
422
+ def wrapper(self: type[Run], *args: Any, **kwargs: Any) -> Any:
423
423
  # `_attach_id` is only assigned in service hence for all service cases
424
424
  # it will be a passthrough. We don't pickle non-service so again a way
425
425
  # to see that we are in non-service case
@@ -458,7 +458,7 @@ class _run_decorator: # noqa: N801
458
458
  class RunStatus:
459
459
  sync_items_total: int = field(default=0)
460
460
  sync_items_pending: int = field(default=0)
461
- sync_time: Optional[datetime] = field(default=None)
461
+ sync_time: datetime | None = field(default=None)
462
462
 
463
463
 
464
464
  class Run:
@@ -528,67 +528,65 @@ class Run:
528
528
  _telemetry_obj_dirty: bool
529
529
  _telemetry_obj_flushed: bytes
530
530
 
531
- _teardown_hooks: List[TeardownHook]
532
- _tags: Optional[Tuple[Any, ...]]
531
+ _teardown_hooks: list[TeardownHook]
532
+ _tags: tuple[Any, ...] | None
533
533
 
534
- _entity: Optional[str]
535
- _project: Optional[str]
536
- _group: Optional[str]
537
- _job_type: Optional[str]
538
- _name: Optional[str]
539
- _notes: Optional[str]
540
- _sweep_id: Optional[str]
534
+ _entity: str | None
535
+ _project: str | None
536
+ _group: str | None
537
+ _job_type: str | None
538
+ _name: str | None
539
+ _notes: str | None
540
+ _sweep_id: str | None
541
541
 
542
- _run_obj: Optional[RunRecord]
542
+ _run_obj: RunRecord | None
543
543
  # Use string literal annotation because of type reference loop
544
- _backend: Optional["wandb.sdk.backend.backend.Backend"]
545
- _internal_run_interface: Optional[
546
- "wandb.sdk.interface.interface_queue.InterfaceQueue"
547
- ]
548
- _wl: Optional[_WandbSetup]
549
-
550
- _out_redir: Optional[redirect.RedirectBase]
551
- _err_redir: Optional[redirect.RedirectBase]
552
- _redirect_cb: Optional[Callable[[str, str], None]]
553
- _redirect_raw_cb: Optional[Callable[[str, str], None]]
554
- _output_writer: Optional["filesystem.CRDedupedFile"]
555
- _quiet: Optional[bool]
544
+ _backend: wandb.sdk.backend.backend.Backend | None
545
+ _internal_run_interface: wandb.sdk.interface.interface_queue.InterfaceQueue | None
546
+ _wl: _WandbSetup | None
547
+
548
+ _out_redir: redirect.RedirectBase | None
549
+ _err_redir: redirect.RedirectBase | None
550
+ _redirect_cb: Callable[[str, str], None] | None
551
+ _redirect_raw_cb: Callable[[str, str], None] | None
552
+ _output_writer: filesystem.CRDedupedFile | None
553
+ _quiet: bool | None
556
554
 
557
555
  _atexit_cleanup_called: bool
558
- _hooks: Optional[ExitHooks]
559
- _exit_code: Optional[int]
556
+ _hooks: ExitHooks | None
557
+ _exit_code: int | None
560
558
 
561
- _run_status_checker: Optional[RunStatusChecker]
559
+ _run_status_checker: RunStatusChecker | None
562
560
 
563
- _sampled_history: Optional["SampledHistoryResponse"]
564
- _final_summary: Optional["GetSummaryResponse"]
565
- _poll_exit_handle: Optional[MailboxHandle]
566
- _poll_exit_response: Optional[PollExitResponse]
567
- _internal_messages_response: Optional["InternalMessagesResponse"]
561
+ _sampled_history: SampledHistoryResponse | None
562
+ _final_summary: GetSummaryResponse | None
563
+ _poll_exit_handle: MailboxHandle | None
564
+ _poll_exit_response: PollExitResponse | None
565
+ _internal_messages_response: InternalMessagesResponse | None
568
566
 
569
- _stdout_slave_fd: Optional[int]
570
- _stderr_slave_fd: Optional[int]
571
- _artifact_slots: List[str]
567
+ _stdout_slave_fd: int | None
568
+ _stderr_slave_fd: int | None
569
+ _artifact_slots: list[str]
572
570
 
573
571
  _init_pid: int
574
572
  _attach_pid: int
575
- _iface_pid: Optional[int]
576
- _iface_port: Optional[int]
573
+ _iface_pid: int | None
574
+ _iface_port: int | None
577
575
 
578
- _attach_id: Optional[str]
576
+ _attach_id: str | None
579
577
  _is_attached: bool
580
578
  _is_finished: bool
581
579
  _settings: Settings
582
580
 
583
- _launch_artifacts: Optional[Dict[str, Any]]
584
- _printer: Union["PrinterTerm", "PrinterJupyter"]
581
+ _launch_artifacts: dict[str, Any] | None
582
+ _printer: printer.Printer
585
583
 
586
584
  def __init__(
587
585
  self,
588
586
  settings: Settings,
589
- config: Optional[Dict[str, Any]] = None,
590
- sweep_config: Optional[Dict[str, Any]] = None,
591
- launch_config: Optional[Dict[str, Any]] = None,
587
+ config: dict[str, Any] | None = None,
588
+ sweep_config: dict[str, Any] | None = None,
589
+ launch_config: dict[str, Any] | None = None,
592
590
  ) -> None:
593
591
  # pid is set, so we know if this run object was initialized by this process
594
592
  self._init_pid = os.getpid()
@@ -608,9 +606,9 @@ class Run:
608
606
  def _init(
609
607
  self,
610
608
  settings: Settings,
611
- config: Optional[Dict[str, Any]] = None,
612
- sweep_config: Optional[Dict[str, Any]] = None,
613
- launch_config: Optional[Dict[str, Any]] = None,
609
+ config: dict[str, Any] | None = None,
610
+ sweep_config: dict[str, Any] | None = None,
611
+ launch_config: dict[str, Any] | None = None,
614
612
  ) -> None:
615
613
  self._settings = settings
616
614
 
@@ -626,7 +624,7 @@ class Run:
626
624
  )
627
625
  self.summary._set_update_callback(self._summary_update_callback)
628
626
  self._step = 0
629
- self._torch_history: Optional[wandb_torch.TorchHistory] = None # type: ignore
627
+ self._torch_history: wandb_torch.TorchHistory | None = None # type: ignore
630
628
 
631
629
  # todo: eventually would be nice to make this configurable using self._settings._start_time
632
630
  # need to test (jhr): if you set start time to 2 days ago and run a test for 15 minutes,
@@ -635,9 +633,9 @@ class Run:
635
633
 
636
634
  _datatypes_set_callback(self._datatypes_callback)
637
635
 
638
- self._printer = get_printer(self._settings._jupyter)
636
+ self._printer = printer.get_printer(self._settings._jupyter)
639
637
  self._wl = None
640
- self._reporter: Optional[Reporter] = None
638
+ self._reporter: Reporter | None = None
641
639
 
642
640
  self._entity = None
643
641
  self._project = None
@@ -663,7 +661,7 @@ class Run:
663
661
  self._quiet = self._settings.quiet
664
662
 
665
663
  self._output_writer = None
666
- self._used_artifact_slots: Dict[str, str] = {}
664
+ self._used_artifact_slots: dict[str, str] = {}
667
665
 
668
666
  # Returned from backend request_run(), set from wandb_init?
669
667
  self._run_obj = None
@@ -699,8 +697,8 @@ class Run:
699
697
  config = config or dict()
700
698
  wandb_key = "_wandb"
701
699
  config.setdefault(wandb_key, dict())
702
- self._launch_artifact_mapping: Dict[str, Any] = {}
703
- self._unique_launch_artifact_sequence_names: Dict[str, Any] = {}
700
+ self._launch_artifact_mapping: dict[str, Any] = {}
701
+ self._unique_launch_artifact_sequence_names: dict[str, Any] = {}
704
702
  if self._settings.save_code and self._settings.program_relpath:
705
703
  config[wandb_key]["code_path"] = LogicalPath(
706
704
  os.path.join("code", self._settings.program_relpath)
@@ -766,7 +764,7 @@ class Run:
766
764
  def _handle_launch_artifact_overrides(self) -> None:
767
765
  if self._settings.launch and (os.environ.get("WANDB_ARTIFACTS") is not None):
768
766
  try:
769
- artifacts: Dict[str, Any] = json.loads(
767
+ artifacts: dict[str, Any] = json.loads(
770
768
  os.environ.get("WANDB_ARTIFACTS", "{}")
771
769
  )
772
770
  except (ValueError, SyntaxError):
@@ -786,7 +784,7 @@ class Run:
786
784
  artifacts = launch_config.get("overrides").get("artifacts")
787
785
  self._initialize_launch_artifact_maps(artifacts)
788
786
 
789
- def _initialize_launch_artifact_maps(self, artifacts: Dict[str, Any]) -> None:
787
+ def _initialize_launch_artifact_maps(self, artifacts: dict[str, Any]) -> None:
790
788
  for key, item in artifacts.items():
791
789
  self._launch_artifact_mapping[key] = item
792
790
  artifact_sequence_tuple_or_slot = key.split(":")
@@ -893,7 +891,7 @@ class Run:
893
891
  except Exception:
894
892
  wandb.termwarn("Cannot find valid git repo associated with this directory.")
895
893
 
896
- def __deepcopy__(self, memo: Dict[int, Any]) -> "Run":
894
+ def __deepcopy__(self, memo: dict[int, Any]) -> Run:
897
895
  return self
898
896
 
899
897
  def __getstate__(self) -> Any:
@@ -927,7 +925,7 @@ class Run:
927
925
  self.__dict__.update(state)
928
926
 
929
927
  @property
930
- def _torch(self) -> "wandb_torch.TorchHistory": # type: ignore
928
+ def _torch(self) -> wandb_torch.TorchHistory: # type: ignore
931
929
  if self._torch_history is None:
932
930
  self._torch_history = wandb_torch.TorchHistory() # type: ignore
933
931
  return self._torch_history
@@ -959,7 +957,7 @@ class Run:
959
957
 
960
958
  @property
961
959
  @_run_decorator._attach
962
- def name(self) -> Optional[str]:
960
+ def name(self) -> str | None:
963
961
  """Display name of the run.
964
962
 
965
963
  Display names are not guaranteed to be unique and may be descriptive.
@@ -982,7 +980,7 @@ class Run:
982
980
 
983
981
  @property
984
982
  @_run_decorator._attach
985
- def notes(self) -> Optional[str]:
983
+ def notes(self) -> str | None:
986
984
  """Notes associated with the run, if there are any.
987
985
 
988
986
  Notes can be a multiline string and can also use markdown and latex equations
@@ -1003,7 +1001,7 @@ class Run:
1003
1001
 
1004
1002
  @property
1005
1003
  @_run_decorator._attach
1006
- def tags(self) -> Optional[Tuple]:
1004
+ def tags(self) -> tuple | None:
1007
1005
  """Tags associated with the run, if there are any."""
1008
1006
  if self._tags:
1009
1007
  return self._tags
@@ -1030,7 +1028,7 @@ class Run:
1030
1028
 
1031
1029
  @property
1032
1030
  @_run_decorator._attach
1033
- def sweep_id(self) -> Optional[str]:
1031
+ def sweep_id(self) -> str | None:
1034
1032
  """ID of the sweep associated with the run, if there is one."""
1035
1033
  if not self._run_obj:
1036
1034
  return None
@@ -1149,15 +1147,13 @@ class Run:
1149
1147
  @_run_decorator._attach
1150
1148
  def log_code(
1151
1149
  self,
1152
- root: Optional[str] = ".",
1153
- name: Optional[str] = None,
1154
- include_fn: Union[
1155
- Callable[[str, str], bool], Callable[[str], bool]
1156
- ] = _is_py_requirements_or_dockerfile,
1157
- exclude_fn: Union[
1158
- Callable[[str, str], bool], Callable[[str], bool]
1159
- ] = filenames.exclude_wandb_fn,
1160
- ) -> Optional[Artifact]:
1150
+ root: str | None = ".",
1151
+ name: str | None = None,
1152
+ include_fn: Callable[[str, str], bool]
1153
+ | Callable[[str], bool] = _is_py_requirements_or_dockerfile,
1154
+ exclude_fn: Callable[[str, str], bool]
1155
+ | Callable[[str], bool] = filenames.exclude_wandb_fn,
1156
+ ) -> Artifact | None:
1161
1157
  """Save the current state of your code to a W&B Artifact.
1162
1158
 
1163
1159
  By default, it walks the current directory and logs all files that end with `.py`.
@@ -1230,7 +1226,7 @@ class Run:
1230
1226
 
1231
1227
  return self._log_artifact(art)
1232
1228
 
1233
- def get_url(self) -> Optional[str]:
1229
+ def get_url(self) -> str | None:
1234
1230
  """Return the url for the W&B run, if there is one.
1235
1231
 
1236
1232
  Offline runs will not have a url.
@@ -1240,7 +1236,7 @@ class Run:
1240
1236
  return None
1241
1237
  return self._settings.run_url
1242
1238
 
1243
- def get_project_url(self) -> Optional[str]:
1239
+ def get_project_url(self) -> str | None:
1244
1240
  """Return the url for the W&B project associated with the run, if there is one.
1245
1241
 
1246
1242
  Offline runs will not have a project url.
@@ -1250,7 +1246,7 @@ class Run:
1250
1246
  return None
1251
1247
  return self._settings.project_url
1252
1248
 
1253
- def get_sweep_url(self) -> Optional[str]:
1249
+ def get_sweep_url(self) -> str | None:
1254
1250
  """Return the url for the sweep associated with the run, if there is one."""
1255
1251
  if self._settings._offline:
1256
1252
  wandb.termwarn("URL not available in offline run")
@@ -1259,7 +1255,7 @@ class Run:
1259
1255
 
1260
1256
  @property
1261
1257
  @_run_decorator._attach
1262
- def url(self) -> Optional[str]:
1258
+ def url(self) -> str | None:
1263
1259
  """The W&B url associated with the run."""
1264
1260
  return self.get_url()
1265
1261
 
@@ -1274,9 +1270,9 @@ class Run:
1274
1270
 
1275
1271
  def _label_internal(
1276
1272
  self,
1277
- code: Optional[str] = None,
1278
- repo: Optional[str] = None,
1279
- code_version: Optional[str] = None,
1273
+ code: str | None = None,
1274
+ repo: str | None = None,
1275
+ code_version: str | None = None,
1280
1276
  ) -> None:
1281
1277
  with telemetry.context(run=self) as tel:
1282
1278
  if code and RE_LABEL.match(code):
@@ -1288,9 +1284,9 @@ class Run:
1288
1284
 
1289
1285
  def _label(
1290
1286
  self,
1291
- code: Optional[str] = None,
1292
- repo: Optional[str] = None,
1293
- code_version: Optional[str] = None,
1287
+ code: str | None = None,
1288
+ repo: str | None = None,
1289
+ code_version: str | None = None,
1294
1290
  **kwargs: str,
1295
1291
  ) -> None:
1296
1292
  if self._settings.label_disable:
@@ -1312,7 +1308,7 @@ class Run:
1312
1308
  # update telemetry in the backend immediately for _label() callers
1313
1309
  self._telemetry_flush()
1314
1310
 
1315
- def _label_probe_lines(self, lines: List[str]) -> None:
1311
+ def _label_probe_lines(self, lines: list[str]) -> None:
1316
1312
  if not lines:
1317
1313
  return
1318
1314
  parsed = telemetry._parse_label_lines(lines)
@@ -1380,23 +1376,23 @@ class Run:
1380
1376
  return prefix + f"<iframe src={url!r} style={style!r}></iframe>"
1381
1377
 
1382
1378
  def _repr_mimebundle_(
1383
- self, include: Optional[Any] = None, exclude: Optional[Any] = None
1384
- ) -> Dict[str, str]:
1379
+ self, include: Any | None = None, exclude: Any | None = None
1380
+ ) -> dict[str, str]:
1385
1381
  return {"text/html": self.to_html(hidden=True)}
1386
1382
 
1387
1383
  @_run_decorator._noop_on_finish()
1388
1384
  def _config_callback(
1389
1385
  self,
1390
- key: Optional[Union[Tuple[str, ...], str]] = None,
1391
- val: Optional[Any] = None,
1392
- data: Optional[Dict[str, object]] = None,
1386
+ key: tuple[str, ...] | str | None = None,
1387
+ val: Any | None = None,
1388
+ data: dict[str, object] | None = None,
1393
1389
  ) -> None:
1394
1390
  logger.info(f"config_cb {key} {val} {data}")
1395
1391
  if self._backend and self._backend.interface:
1396
1392
  self._backend.interface.publish_config(key=key, val=val, data=data)
1397
1393
 
1398
1394
  def _config_artifact_callback(
1399
- self, key: str, val: Union[str, Artifact, dict]
1395
+ self, key: str, val: str | Artifact | dict
1400
1396
  ) -> Artifact:
1401
1397
  # artifacts can look like dicts as they are passed into the run config
1402
1398
  # since the run config stores them on the backend as a dict with fields shown
@@ -1447,7 +1443,7 @@ class Run:
1447
1443
  # line = "Waiting for run.summary data..."
1448
1444
  # self._printer.display(line)
1449
1445
 
1450
- def _summary_get_current_summary_callback(self) -> Dict[str, Any]:
1446
+ def _summary_get_current_summary_callback(self) -> dict[str, Any]:
1451
1447
  if self._is_finished:
1452
1448
  # TODO: WB-18420: fetch summary from backend and stage it before run is finished
1453
1449
  wandb.termwarn("Summary data not available in finished run")
@@ -1474,7 +1470,7 @@ class Run:
1474
1470
  files: FilesDict = dict(files=[(GlobStr(glob.escape(fname)), "now")])
1475
1471
  self._backend.interface.publish_files(files)
1476
1472
 
1477
- def _visualization_hack(self, row: Dict[str, Any]) -> Dict[str, Any]:
1473
+ def _visualization_hack(self, row: dict[str, Any]) -> dict[str, Any]:
1478
1474
  # TODO(jhr): move visualize hack somewhere else
1479
1475
  chart_keys = set()
1480
1476
  split_table_set = set()
@@ -1511,9 +1507,9 @@ class Run:
1511
1507
 
1512
1508
  def _partial_history_callback(
1513
1509
  self,
1514
- row: Dict[str, Any],
1515
- step: Optional[int] = None,
1516
- commit: Optional[bool] = None,
1510
+ row: dict[str, Any],
1511
+ step: int | None = None,
1512
+ commit: bool | None = None,
1517
1513
  ) -> None:
1518
1514
  row = row.copy()
1519
1515
  if row:
@@ -1560,19 +1556,19 @@ class Run:
1560
1556
  def _set_library(self, library: _WandbSetup) -> None:
1561
1557
  self._wl = library
1562
1558
 
1563
- def _set_backend(self, backend: "wandb.sdk.backend.backend.Backend") -> None:
1559
+ def _set_backend(self, backend: wandb.sdk.backend.backend.Backend) -> None:
1564
1560
  self._backend = backend
1565
1561
 
1566
1562
  def _set_internal_run_interface(
1567
1563
  self,
1568
- interface: "wandb.sdk.interface.interface_queue.InterfaceQueue",
1564
+ interface: wandb.sdk.interface.interface_queue.InterfaceQueue,
1569
1565
  ) -> None:
1570
1566
  self._internal_run_interface = interface
1571
1567
 
1572
1568
  def _set_reporter(self, reporter: Reporter) -> None:
1573
1569
  self._reporter = reporter
1574
1570
 
1575
- def _set_teardown_hooks(self, hooks: List[TeardownHook]) -> None:
1571
+ def _set_teardown_hooks(self, hooks: list[TeardownHook]) -> None:
1576
1572
  self._teardown_hooks = hooks
1577
1573
 
1578
1574
  def _set_run_obj(self, run_obj: RunRecord) -> None:
@@ -1610,7 +1606,7 @@ class Run:
1610
1606
  )
1611
1607
 
1612
1608
  def _add_singleton(
1613
- self, data_type: str, key: str, value: Dict[Union[int, str], str]
1609
+ self, data_type: str, key: str, value: dict[int | str, str]
1614
1610
  ) -> None:
1615
1611
  """Store a singleton item to wandb config.
1616
1612
 
@@ -1640,9 +1636,9 @@ class Run:
1640
1636
 
1641
1637
  def _log(
1642
1638
  self,
1643
- data: Dict[str, Any],
1644
- step: Optional[int] = None,
1645
- commit: Optional[bool] = None,
1639
+ data: dict[str, Any],
1640
+ step: int | None = None,
1641
+ commit: bool | None = None,
1646
1642
  ) -> None:
1647
1643
  if not isinstance(data, Mapping):
1648
1644
  raise ValueError("wandb.log must be passed a dictionary")
@@ -1681,10 +1677,10 @@ class Run:
1681
1677
  @_run_decorator._attach
1682
1678
  def log(
1683
1679
  self,
1684
- data: Dict[str, Any],
1685
- step: Optional[int] = None,
1686
- commit: Optional[bool] = None,
1687
- sync: Optional[bool] = None,
1680
+ data: dict[str, Any],
1681
+ step: int | None = None,
1682
+ commit: bool | None = None,
1683
+ sync: bool | None = None,
1688
1684
  ) -> None:
1689
1685
  """Upload run data.
1690
1686
 
@@ -1937,10 +1933,10 @@ class Run:
1937
1933
  @_run_decorator._attach
1938
1934
  def save(
1939
1935
  self,
1940
- glob_str: Optional[Union[str, os.PathLike]] = None,
1941
- base_path: Optional[Union[str, os.PathLike]] = None,
1936
+ glob_str: str | os.PathLike | None = None,
1937
+ base_path: str | os.PathLike | None = None,
1942
1938
  policy: PolicyName = "live",
1943
- ) -> Union[bool, List[str]]:
1939
+ ) -> bool | list[str]:
1944
1940
  """Sync one or more files to W&B.
1945
1941
 
1946
1942
  Relative paths are relative to the current working directory.
@@ -2045,8 +2041,8 @@ class Run:
2045
2041
  self,
2046
2042
  glob_path: pathlib.PurePath,
2047
2043
  base_path: pathlib.PurePath,
2048
- policy: "PolicyName",
2049
- ) -> List[str]:
2044
+ policy: PolicyName,
2045
+ ) -> list[str]:
2050
2046
  # Can't use is_relative_to() because that's added in Python 3.9,
2051
2047
  # but we support down to Python 3.7.
2052
2048
  if not str(glob_path).startswith(str(base_path)):
@@ -2129,10 +2125,10 @@ class Run:
2129
2125
  def restore(
2130
2126
  self,
2131
2127
  name: str,
2132
- run_path: Optional[str] = None,
2128
+ run_path: str | None = None,
2133
2129
  replace: bool = False,
2134
- root: Optional[str] = None,
2135
- ) -> Union[None, TextIO]:
2130
+ root: str | None = None,
2131
+ ) -> None | TextIO:
2136
2132
  return restore(
2137
2133
  name,
2138
2134
  run_path or self._get_path(),
@@ -2142,9 +2138,7 @@ class Run:
2142
2138
 
2143
2139
  @_run_decorator._noop
2144
2140
  @_run_decorator._attach
2145
- def finish(
2146
- self, exit_code: Optional[int] = None, quiet: Optional[bool] = None
2147
- ) -> None:
2141
+ def finish(self, exit_code: int | None = None, quiet: bool | None = None) -> None:
2148
2142
  """Mark a run as finished, and finish uploading all data.
2149
2143
 
2150
2144
  This is used when creating multiple runs in the same process. We automatically
@@ -2158,8 +2152,8 @@ class Run:
2158
2152
 
2159
2153
  def _finish(
2160
2154
  self,
2161
- exit_code: Optional[int] = None,
2162
- quiet: Optional[bool] = None,
2155
+ exit_code: int | None = None,
2156
+ quiet: bool | None = None,
2163
2157
  ) -> None:
2164
2158
  logger.info(f"finishing run {self._get_path()}")
2165
2159
  with telemetry.context(run=self) as tel:
@@ -2210,7 +2204,7 @@ class Run:
2210
2204
 
2211
2205
  @_run_decorator._noop
2212
2206
  @_run_decorator._attach
2213
- def join(self, exit_code: Optional[int] = None) -> None:
2207
+ def join(self, exit_code: int | None = None) -> None:
2214
2208
  """Deprecated alias for `finish()` - use finish instead."""
2215
2209
  if hasattr(self, "_telemetry_obj"):
2216
2210
  deprecate.deprecate(
@@ -2249,10 +2243,10 @@ class Run:
2249
2243
  @staticmethod
2250
2244
  def plot_table(
2251
2245
  vega_spec_name: str,
2252
- data_table: "wandb.Table",
2253
- fields: Dict[str, Any],
2254
- string_fields: Optional[Dict[str, Any]] = None,
2255
- split_table: Optional[bool] = False,
2246
+ data_table: Table,
2247
+ fields: dict[str, Any],
2248
+ string_fields: dict[str, Any] | None = None,
2249
+ split_table: bool | None = False,
2256
2250
  ) -> CustomChart:
2257
2251
  """Create a custom plot on a table.
2258
2252
 
@@ -2264,6 +2258,8 @@ class Run:
2264
2258
  visualization needs
2265
2259
  string_fields: a dict that provides values for any string constants
2266
2260
  the custom visualization needs
2261
+ split_table: a boolean that indicates whether the table should be in
2262
+ a separate section in the UI
2267
2263
  """
2268
2264
  return custom_chart(
2269
2265
  vega_spec_name, data_table, fields, string_fields or {}, split_table
@@ -2290,6 +2286,8 @@ class Run:
2290
2286
  define_metric=self.define_metric,
2291
2287
  plot_table=self.plot_table,
2292
2288
  alert=self.alert,
2289
+ watch=self.watch,
2290
+ unwatch=self.unwatch,
2293
2291
  mark_preempting=self.mark_preempting,
2294
2292
  log_model=self.log_model,
2295
2293
  use_model=self.use_model,
@@ -2298,9 +2296,9 @@ class Run:
2298
2296
 
2299
2297
  def _redirect(
2300
2298
  self,
2301
- stdout_slave_fd: Optional[int],
2302
- stderr_slave_fd: Optional[int],
2303
- console: Optional[str] = None,
2299
+ stdout_slave_fd: int | None,
2300
+ stderr_slave_fd: int | None,
2301
+ console: str | None = None,
2304
2302
  ) -> None:
2305
2303
  if console is None:
2306
2304
  console = self._settings.console
@@ -2415,7 +2413,7 @@ class Run:
2415
2413
  self._err_redir.uninstall()
2416
2414
  logger.info("restore done")
2417
2415
 
2418
- def _atexit_cleanup(self, exit_code: Optional[int] = None) -> None:
2416
+ def _atexit_cleanup(self, exit_code: int | None = None) -> None:
2419
2417
  if self._backend is None:
2420
2418
  logger.warning("process exited without backend configured")
2421
2419
  return
@@ -2526,7 +2524,7 @@ class Run:
2526
2524
  self,
2527
2525
  ) -> None:
2528
2526
  def _telemetry_import_hook(
2529
- run: "Run",
2527
+ run: Run,
2530
2528
  module: Any,
2531
2529
  ) -> None:
2532
2530
  with telemetry.context(run=run) as tel:
@@ -2569,7 +2567,7 @@ class Run:
2569
2567
 
2570
2568
  StagedLaunchInputs().apply(self)
2571
2569
 
2572
- def _make_job_source_reqs(self) -> Tuple[List[str], Dict[str, Any], Dict[str, Any]]:
2570
+ def _make_job_source_reqs(self) -> tuple[list[str], dict[str, Any], dict[str, Any]]:
2573
2571
  from wandb.util import working_set
2574
2572
 
2575
2573
  installed_packages_list = sorted(f"{d.key}=={d.version}" for d in working_set())
@@ -2581,10 +2579,10 @@ class Run:
2581
2579
  def _construct_job_artifact(
2582
2580
  self,
2583
2581
  name: str,
2584
- source_dict: "JobSourceDict",
2585
- installed_packages_list: List[str],
2586
- patch_path: Optional[os.PathLike] = None,
2587
- ) -> "Artifact":
2582
+ source_dict: JobSourceDict,
2583
+ installed_packages_list: list[str],
2584
+ patch_path: os.PathLike | None = None,
2585
+ ) -> Artifact:
2588
2586
  job_artifact = job_builder.JobArtifact(name)
2589
2587
  if patch_path and os.path.exists(patch_path):
2590
2588
  job_artifact.add_file(FilePathStr(str(patch_path)), "diff.patch")
@@ -2597,12 +2595,12 @@ class Run:
2597
2595
 
2598
2596
  def _create_image_job(
2599
2597
  self,
2600
- input_types: Dict[str, Any],
2601
- output_types: Dict[str, Any],
2602
- installed_packages_list: List[str],
2603
- docker_image_name: Optional[str] = None,
2604
- args: Optional[List[str]] = None,
2605
- ) -> Optional["Artifact"]:
2598
+ input_types: dict[str, Any],
2599
+ output_types: dict[str, Any],
2600
+ installed_packages_list: list[str],
2601
+ docker_image_name: str | None = None,
2602
+ args: list[str] | None = None,
2603
+ ) -> Artifact | None:
2606
2604
  docker_image_name = docker_image_name or os.getenv("WANDB_DOCKER")
2607
2605
 
2608
2606
  if not docker_image_name:
@@ -2625,7 +2623,7 @@ class Run:
2625
2623
  return job_artifact
2626
2624
 
2627
2625
  def _log_job_artifact_with_image(
2628
- self, docker_image_name: str, args: Optional[List[str]] = None
2626
+ self, docker_image_name: str, args: list[str] | None = None
2629
2627
  ) -> Artifact:
2630
2628
  packages, in_types, out_types = self._make_job_source_reqs()
2631
2629
  job_artifact = self._create_image_job(
@@ -2654,16 +2652,20 @@ class Run:
2654
2652
  handle = self._backend.interface.deliver_poll_exit()
2655
2653
  probe_handle.set_mailbox_handle(handle)
2656
2654
 
2657
- def _on_progress_exit(self, progress_handle: MailboxProgress) -> None:
2655
+ def _on_progress_exit(
2656
+ self,
2657
+ progress_printer: progress.ProgressPrinter,
2658
+ progress_handle: MailboxProgress,
2659
+ ) -> None:
2658
2660
  probe_handles = progress_handle.get_probe_handles()
2659
- assert probe_handles and len(probe_handles) == 1
2661
+ if not probe_handles or len(probe_handles) != 1:
2662
+ return
2660
2663
 
2661
2664
  result = probe_handles[0].get_probe_result()
2662
2665
  if not result:
2663
2666
  return
2664
- self._footer_file_pusher_status_info(
2665
- result.response.poll_exit_response, printer=self._printer
2666
- )
2667
+
2668
+ progress_printer.update([result.response.poll_exit_response])
2667
2669
 
2668
2670
  def _on_finish(self) -> None:
2669
2671
  trigger.call("on_finished")
@@ -2678,16 +2680,28 @@ class Run:
2678
2680
  exit_handle = self._backend.interface.deliver_exit(self._exit_code)
2679
2681
  exit_handle.add_probe(on_probe=self._on_probe_exit)
2680
2682
 
2681
- # wait for the exit to complete
2682
- _ = exit_handle.wait(timeout=-1, on_progress=self._on_progress_exit)
2683
+ with progress.progress_printer(
2684
+ self._printer,
2685
+ self._settings,
2686
+ ) as progress_printer:
2687
+ # Wait for the run to complete.
2688
+ _ = exit_handle.wait(
2689
+ timeout=-1,
2690
+ on_progress=functools.partial(
2691
+ self._on_progress_exit,
2692
+ progress_printer,
2693
+ ),
2694
+ )
2683
2695
 
2696
+ # Print some final statistics.
2684
2697
  poll_exit_handle = self._backend.interface.deliver_poll_exit()
2685
- # wait for them, it's ok to do this serially but this can be improved
2686
2698
  result = poll_exit_handle.wait(timeout=-1)
2687
2699
  assert result
2688
- self._footer_file_pusher_status_info(
2689
- result.response.poll_exit_response, printer=self._printer
2700
+ progress.print_sync_dedupe_stats(
2701
+ self._printer,
2702
+ result.response.poll_exit_response,
2690
2703
  )
2704
+
2691
2705
  self._poll_exit_response = result.response.poll_exit_response
2692
2706
  internal_messages_handle = self._backend.interface.deliver_internal_messages()
2693
2707
  result = internal_messages_handle.wait(timeout=-1)
@@ -2728,12 +2742,12 @@ class Run:
2728
2742
  def define_metric(
2729
2743
  self,
2730
2744
  name: str,
2731
- step_metric: Union[str, wandb_metric.Metric, None] = None,
2732
- step_sync: Optional[bool] = None,
2733
- hidden: Optional[bool] = None,
2734
- summary: Optional[str] = None,
2735
- goal: Optional[str] = None,
2736
- overwrite: Optional[bool] = None,
2745
+ step_metric: str | wandb_metric.Metric | None = None,
2746
+ step_sync: bool | None = None,
2747
+ hidden: bool | None = None,
2748
+ summary: str | None = None,
2749
+ goal: str | None = None,
2750
+ overwrite: bool | None = None,
2737
2751
  ) -> wandb_metric.Metric:
2738
2752
  """Customize metrics logged with `wandb.log()`.
2739
2753
 
@@ -2789,12 +2803,12 @@ class Run:
2789
2803
  def _define_metric(
2790
2804
  self,
2791
2805
  name: str,
2792
- step_metric: Union[str, wandb_metric.Metric, None] = None,
2793
- step_sync: Optional[bool] = None,
2794
- hidden: Optional[bool] = None,
2795
- summary: Optional[str] = None,
2796
- goal: Optional[str] = None,
2797
- overwrite: Optional[bool] = None,
2806
+ step_metric: str | wandb_metric.Metric | None = None,
2807
+ step_sync: bool | None = None,
2808
+ hidden: bool | None = None,
2809
+ summary: str | None = None,
2810
+ goal: str | None = None,
2811
+ overwrite: bool | None = None,
2798
2812
  ) -> wandb_metric.Metric:
2799
2813
  if not name:
2800
2814
  raise wandb.Error("define_metric() requires non-empty name argument")
@@ -2820,7 +2834,7 @@ class Run:
2820
2834
  raise wandb.Error(
2821
2835
  f"Unhandled define_metric() arg: name (glob suffixes only): {name}"
2822
2836
  )
2823
- summary_ops: Optional[Sequence[str]] = None
2837
+ summary_ops: Sequence[str] | None = None
2824
2838
  if summary:
2825
2839
  summary_items = [s.lower() for s in summary.split(",")]
2826
2840
  summary_ops = []
@@ -2833,7 +2847,7 @@ class Run:
2833
2847
  with telemetry.context(run=self) as tel:
2834
2848
  tel.feature.metric_summary = True
2835
2849
  # TODO: deprecate goal
2836
- goal_cleaned: Optional[str] = None
2850
+ goal_cleaned: str | None = None
2837
2851
  if goal is not None:
2838
2852
  goal_cleaned = goal[:3].lower()
2839
2853
  valid_goal = {"min", "max"}
@@ -2864,26 +2878,57 @@ class Run:
2864
2878
  m._commit()
2865
2879
  return m
2866
2880
 
2867
- # TODO(jhr): annotate this
2868
2881
  @_run_decorator._attach
2869
- def watch( # type: ignore
2882
+ def watch(
2870
2883
  self,
2871
- models,
2872
- criterion=None,
2873
- log="gradients",
2874
- log_freq=100,
2875
- idx=None,
2876
- log_graph=False,
2884
+ models: torch.nn.Module | Sequence[torch.nn.Module],
2885
+ criterion: torch.F | None = None,
2886
+ log: Literal["gradients", "parameters", "all"] | None = "gradients",
2887
+ log_freq: int = 1000,
2888
+ idx: int | None = None,
2889
+ log_graph: bool = False,
2877
2890
  ) -> None:
2878
- wandb.watch(models, criterion, log, log_freq, idx, log_graph) # type: ignore
2891
+ """Hooks into the given PyTorch model(s) to monitor gradients and the model's computational graph.
2892
+
2893
+ This function can track parameters, gradients, or both during training. It should be
2894
+ extended to support arbitrary machine learning models in the future.
2895
+
2896
+ Args:
2897
+ models (Union[torch.nn.Module, Sequence[torch.nn.Module]]):
2898
+ A single model or a sequence of models to be monitored.
2899
+ criterion (Optional[torch.F]):
2900
+ The loss function being optimized (optional).
2901
+ log (Optional[Literal["gradients", "parameters", "all"]]):
2902
+ Specifies whether to log "gradients", "parameters", or "all".
2903
+ Set to None to disable logging. (default="gradients")
2904
+ log_freq (int):
2905
+ Frequency (in batches) to log gradients and parameters. (default=1000)
2906
+ idx (Optional[int]):
2907
+ Index used when tracking multiple models with `wandb.watch`. (default=None)
2908
+ log_graph (bool):
2909
+ Whether to log the model's computational graph. (default=False)
2910
+
2911
+ Raises:
2912
+ ValueError:
2913
+ If `wandb.init` has not been called or if any of the models are not instances
2914
+ of `torch.nn.Module`.
2915
+ """
2916
+ wandb.sdk._watch(self, models, criterion, log, log_freq, idx, log_graph)
2879
2917
 
2880
- # TODO(jhr): annotate this
2881
2918
  @_run_decorator._attach
2882
- def unwatch(self, models=None) -> None: # type: ignore
2883
- wandb.unwatch(models=models) # type: ignore
2919
+ def unwatch(
2920
+ self, models: torch.nn.Module | Sequence[torch.nn.Module] | None = None
2921
+ ) -> None:
2922
+ """Remove pytorch model topology, gradient and parameter hooks.
2923
+
2924
+ Args:
2925
+ models (torch.nn.Module | Sequence[torch.nn.Module]):
2926
+ Optional list of pytorch models that have had watch called on them
2927
+ """
2928
+ wandb.sdk._unwatch(self, models=models)
2884
2929
 
2885
2930
  # TODO(kdg): remove all artifact swapping logic
2886
- def _swap_artifact_name(self, artifact_name: str, use_as: Optional[str]) -> str:
2931
+ def _swap_artifact_name(self, artifact_name: str, use_as: str | None) -> str:
2887
2932
  artifact_key_string = use_as or artifact_name
2888
2933
  replacement_artifact_info = self._launch_artifact_mapping.get(
2889
2934
  artifact_key_string
@@ -2926,7 +2971,7 @@ class Run:
2926
2971
  self,
2927
2972
  artifact: Artifact,
2928
2973
  target_path: str,
2929
- aliases: Optional[List[str]] = None,
2974
+ aliases: list[str] | None = None,
2930
2975
  ) -> None:
2931
2976
  """Link the given artifact to a portfolio (a promoted collection of artifacts).
2932
2977
 
@@ -2961,6 +3006,12 @@ class Run:
2961
3006
  # Wait until the artifact is committed before trying to link it.
2962
3007
  artifact.wait()
2963
3008
 
3009
+ organization = ""
3010
+ if is_artifact_registry_project(project):
3011
+ organization = entity
3012
+ # In a Registry linking, the entity is used to fetch the organization of the artifact
3013
+ # therefore the source artifact's entity is passed to the backend
3014
+ entity = artifact._source_entity
2964
3015
  handle = self._backend.interface.deliver_link_artifact(
2965
3016
  self,
2966
3017
  artifact,
@@ -2968,6 +3019,7 @@ class Run:
2968
3019
  aliases,
2969
3020
  entity,
2970
3021
  project,
3022
+ organization,
2971
3023
  )
2972
3024
  if artifact._ttl_duration_seconds is not None:
2973
3025
  wandb.termwarn(
@@ -2985,10 +3037,10 @@ class Run:
2985
3037
  @_run_decorator._attach
2986
3038
  def use_artifact(
2987
3039
  self,
2988
- artifact_or_name: Union[str, Artifact],
2989
- type: Optional[str] = None,
2990
- aliases: Optional[List[str]] = None,
2991
- use_as: Optional[str] = None,
3040
+ artifact_or_name: str | Artifact,
3041
+ type: str | None = None,
3042
+ aliases: list[str] | None = None,
3043
+ use_as: str | None = None,
2992
3044
  ) -> Artifact:
2993
3045
  """Declare an artifact as an input to a run.
2994
3046
 
@@ -2996,8 +3048,9 @@ class Run:
2996
3048
 
2997
3049
  Arguments:
2998
3050
  artifact_or_name: (str or Artifact) An artifact name.
2999
- May be prefixed with entity/project/. Valid names
3000
- can be in the following forms:
3051
+ May be prefixed with project/ or entity/project/.
3052
+ If no entity is specified in the name, the Run or API setting's entity is used.
3053
+ Valid names can be in the following forms:
3001
3054
  - name:version
3002
3055
  - name:alias
3003
3056
  You can also pass an Artifact object created by calling `wandb.Artifact`
@@ -3094,11 +3147,11 @@ class Run:
3094
3147
  @_run_decorator._attach
3095
3148
  def log_artifact(
3096
3149
  self,
3097
- artifact_or_path: Union[Artifact, StrPath],
3098
- name: Optional[str] = None,
3099
- type: Optional[str] = None,
3100
- aliases: Optional[List[str]] = None,
3101
- tags: Optional[List[str]] = None,
3150
+ artifact_or_path: Artifact | StrPath,
3151
+ name: str | None = None,
3152
+ type: str | None = None,
3153
+ aliases: list[str] | None = None,
3154
+ tags: list[str] | None = None,
3102
3155
  ) -> Artifact:
3103
3156
  """Declare an artifact as an output of a run.
3104
3157
 
@@ -3136,11 +3189,11 @@ class Run:
3136
3189
  @_run_decorator._attach
3137
3190
  def upsert_artifact(
3138
3191
  self,
3139
- artifact_or_path: Union[Artifact, str],
3140
- name: Optional[str] = None,
3141
- type: Optional[str] = None,
3142
- aliases: Optional[List[str]] = None,
3143
- distributed_id: Optional[str] = None,
3192
+ artifact_or_path: Artifact | str,
3193
+ name: str | None = None,
3194
+ type: str | None = None,
3195
+ aliases: list[str] | None = None,
3196
+ distributed_id: str | None = None,
3144
3197
  ) -> Artifact:
3145
3198
  """Declare (or append to) a non-finalized artifact as output of a run.
3146
3199
 
@@ -3190,11 +3243,11 @@ class Run:
3190
3243
  @_run_decorator._attach
3191
3244
  def finish_artifact(
3192
3245
  self,
3193
- artifact_or_path: Union[Artifact, str],
3194
- name: Optional[str] = None,
3195
- type: Optional[str] = None,
3196
- aliases: Optional[List[str]] = None,
3197
- distributed_id: Optional[str] = None,
3246
+ artifact_or_path: Artifact | str,
3247
+ name: str | None = None,
3248
+ type: str | None = None,
3249
+ aliases: list[str] | None = None,
3250
+ distributed_id: str | None = None,
3198
3251
  ) -> Artifact:
3199
3252
  """Finishes a non-finalized artifact as output of a run.
3200
3253
 
@@ -3242,12 +3295,12 @@ class Run:
3242
3295
 
3243
3296
  def _log_artifact(
3244
3297
  self,
3245
- artifact_or_path: Union[Artifact, StrPath],
3246
- name: Optional[str] = None,
3247
- type: Optional[str] = None,
3248
- aliases: Optional[List[str]] = None,
3249
- tags: Optional[List[str]] = None,
3250
- distributed_id: Optional[str] = None,
3298
+ artifact_or_path: Artifact | StrPath,
3299
+ name: str | None = None,
3300
+ type: str | None = None,
3301
+ aliases: list[str] | None = None,
3302
+ tags: list[str] | None = None,
3303
+ distributed_id: str | None = None,
3251
3304
  finalize: bool = True,
3252
3305
  is_user_created: bool = False,
3253
3306
  use_after_commit: bool = False,
@@ -3308,7 +3361,7 @@ class Run:
3308
3361
  )
3309
3362
  return artifact
3310
3363
 
3311
- def _public_api(self, overrides: Optional[Dict[str, str]] = None) -> PublicApi:
3364
+ def _public_api(self, overrides: dict[str, str] | None = None) -> PublicApi:
3312
3365
  overrides = {"run": self._run_id}
3313
3366
  if not (self._settings._offline or self._run_obj is None):
3314
3367
  overrides["entity"] = self._run_obj.entity
@@ -3349,11 +3402,11 @@ class Run:
3349
3402
 
3350
3403
  def _prepare_artifact(
3351
3404
  self,
3352
- artifact_or_path: Union[Artifact, StrPath],
3353
- name: Optional[str] = None,
3354
- type: Optional[str] = None,
3355
- aliases: Optional[List[str]] = None,
3356
- ) -> Tuple[Artifact, List[str]]:
3405
+ artifact_or_path: Artifact | StrPath,
3406
+ name: str | None = None,
3407
+ type: str | None = None,
3408
+ aliases: list[str] | None = None,
3409
+ ) -> tuple[Artifact, list[str]]:
3357
3410
  if isinstance(artifact_or_path, (str, os.PathLike)):
3358
3411
  name = name or f"run-{self._run_id}-{os.path.basename(artifact_or_path)}"
3359
3412
  artifact = wandb.Artifact(name, type or "unspecified")
@@ -3384,8 +3437,8 @@ class Run:
3384
3437
  def log_model(
3385
3438
  self,
3386
3439
  path: StrPath,
3387
- name: Optional[str] = None,
3388
- aliases: Optional[List[str]] = None,
3440
+ name: str | None = None,
3441
+ aliases: list[str] | None = None,
3389
3442
  ) -> None:
3390
3443
  """Logs a model artifact containing the contents inside the 'path' to a run and marks it as an output to this run.
3391
3444
 
@@ -3489,8 +3542,8 @@ class Run:
3489
3542
  self,
3490
3543
  path: StrPath,
3491
3544
  registered_model_name: str,
3492
- name: Optional[str] = None,
3493
- aliases: Optional[List[str]] = None,
3545
+ name: str | None = None,
3546
+ aliases: list[str] | None = None,
3494
3547
  ) -> None:
3495
3548
  """Log a model artifact version and link it to a registered model in the model registry.
3496
3549
 
@@ -3583,8 +3636,8 @@ class Run:
3583
3636
  self,
3584
3637
  title: str,
3585
3638
  text: str,
3586
- level: Optional[Union[str, "AlertLevel"]] = None,
3587
- wait_duration: Union[int, float, timedelta, None] = None,
3639
+ level: str | AlertLevel | None = None,
3640
+ wait_duration: int | float | timedelta | None = None,
3588
3641
  ) -> None:
3589
3642
  """Launch an alert with the given title and text.
3590
3643
 
@@ -3612,12 +3665,12 @@ class Run:
3612
3665
  if self._backend and self._backend.interface:
3613
3666
  self._backend.interface.publish_alert(title, text, level_str, wait_duration)
3614
3667
 
3615
- def __enter__(self) -> "Run":
3668
+ def __enter__(self) -> Run:
3616
3669
  return self
3617
3670
 
3618
3671
  def __exit__(
3619
3672
  self,
3620
- exc_type: Type[BaseException],
3673
+ exc_type: type[BaseException],
3621
3674
  exc_val: BaseException,
3622
3675
  exc_tb: TracebackType,
3623
3676
  ) -> bool:
@@ -3641,7 +3694,7 @@ class Run:
3641
3694
  @property
3642
3695
  @_run_decorator._noop_on_finish()
3643
3696
  @_run_decorator._attach
3644
- def _system_metrics(self) -> Dict[str, List[Tuple[datetime, float]]]:
3697
+ def _system_metrics(self) -> dict[str, list[tuple[datetime, float]]]:
3645
3698
  """Returns a dictionary of system metrics.
3646
3699
 
3647
3700
  Returns:
@@ -3650,7 +3703,7 @@ class Run:
3650
3703
 
3651
3704
  def pb_to_dict(
3652
3705
  system_metrics_pb: wandb.proto.wandb_internal_pb2.GetSystemMetricsResponse,
3653
- ) -> Dict[str, List[Tuple[datetime, float]]]:
3706
+ ) -> dict[str, list[tuple[datetime, float]]]:
3654
3707
  res = {}
3655
3708
 
3656
3709
  for metric, records in system_metrics_pb.system_metrics.items():
@@ -3691,8 +3744,8 @@ class Run:
3691
3744
  @staticmethod
3692
3745
  def _header(
3693
3746
  *,
3694
- settings: "Settings",
3695
- printer: Union["PrinterTerm", "PrinterJupyter"],
3747
+ settings: Settings,
3748
+ printer: printer.Printer,
3696
3749
  ) -> None:
3697
3750
  Run._header_wandb_version_info(settings=settings, printer=printer)
3698
3751
  Run._header_sync_info(settings=settings, printer=printer)
@@ -3701,22 +3754,20 @@ class Run:
3701
3754
  @staticmethod
3702
3755
  def _header_wandb_version_info(
3703
3756
  *,
3704
- settings: "Settings",
3705
- printer: Union["PrinterTerm", "PrinterJupyter"],
3757
+ settings: Settings,
3758
+ printer: printer.Printer,
3706
3759
  ) -> None:
3707
3760
  if settings.quiet or settings.silent:
3708
3761
  return
3709
3762
 
3710
3763
  # TODO: add this to a higher verbosity level
3711
- printer.display(
3712
- f"Tracking run with wandb version {wandb.__version__}", off=False
3713
- )
3764
+ printer.display(f"Tracking run with wandb version {wandb.__version__}")
3714
3765
 
3715
3766
  @staticmethod
3716
3767
  def _header_sync_info(
3717
3768
  *,
3718
- settings: "Settings",
3719
- printer: Union["PrinterTerm", "PrinterJupyter"],
3769
+ settings: Settings,
3770
+ printer: printer.Printer,
3720
3771
  ) -> None:
3721
3772
  if settings._offline:
3722
3773
  printer.display(
@@ -3728,17 +3779,18 @@ class Run:
3728
3779
  )
3729
3780
  else:
3730
3781
  info = [f"Run data is saved locally in {printer.files(settings.sync_dir)}"]
3731
- if not printer._html:
3782
+ if not printer.supports_html:
3732
3783
  info.append(
3733
3784
  f"Run {printer.code('`wandb offline`')} to turn off syncing."
3734
3785
  )
3735
- printer.display(info, off=settings.quiet or settings.silent)
3786
+ if not settings.quiet and not settings.silent:
3787
+ printer.display(info)
3736
3788
 
3737
3789
  @staticmethod
3738
3790
  def _header_run_info(
3739
3791
  *,
3740
- settings: "Settings",
3741
- printer: Union["PrinterTerm", "PrinterJupyter"],
3792
+ settings: Settings,
3793
+ printer: printer.Printer,
3742
3794
  ) -> None:
3743
3795
  if settings._offline or settings.silent:
3744
3796
  return
@@ -3756,7 +3808,7 @@ class Run:
3756
3808
  if not run_name:
3757
3809
  return
3758
3810
 
3759
- if printer._html:
3811
+ if printer.supports_html:
3760
3812
  if not wandb.jupyter.maybe_display(): # type: ignore
3761
3813
  run_line = f"<strong>{printer.link(run_url, run_name)}</strong>"
3762
3814
  project_line, sweep_line = "", ""
@@ -3775,10 +3827,8 @@ class Run:
3775
3827
  [f"{run_state_str} {run_line} {project_line}", sweep_line],
3776
3828
  )
3777
3829
 
3778
- else:
3779
- printer.display(
3780
- f"{run_state_str} {printer.name(run_name)}", off=not run_name
3781
- )
3830
+ elif run_name:
3831
+ printer.display(f"{run_state_str} {printer.name(run_name)}")
3782
3832
 
3783
3833
  if not settings.quiet:
3784
3834
  # TODO: add verbosity levels and add this to higher levels
@@ -3794,11 +3844,13 @@ class Run:
3794
3844
  )
3795
3845
 
3796
3846
  # TODO(settings) use `wandb_settings` (if self.settings.anonymous == "true":)
3797
- if Api().api.settings().get("anonymous") == "true":
3847
+ if run_name and Api().api.settings().get("anonymous") == "true":
3798
3848
  printer.display(
3799
- "Do NOT share these links with anyone. They can be used to claim your runs.",
3849
+ (
3850
+ "Do NOT share these links with anyone."
3851
+ " They can be used to claim your runs."
3852
+ ),
3800
3853
  level="warn",
3801
- off=not run_name,
3802
3854
  )
3803
3855
 
3804
3856
  # ------------------------------------------------------------------------------
@@ -3808,15 +3860,15 @@ class Run:
3808
3860
  # with the service execution path that doesn't have access to the run instance
3809
3861
  @staticmethod
3810
3862
  def _footer(
3811
- sampled_history: Optional["SampledHistoryResponse"] = None,
3812
- final_summary: Optional["GetSummaryResponse"] = None,
3813
- poll_exit_response: Optional[PollExitResponse] = None,
3814
- internal_messages_response: Optional["InternalMessagesResponse"] = None,
3815
- reporter: Optional[Reporter] = None,
3816
- quiet: Optional[bool] = None,
3863
+ sampled_history: SampledHistoryResponse | None = None,
3864
+ final_summary: GetSummaryResponse | None = None,
3865
+ poll_exit_response: PollExitResponse | None = None,
3866
+ internal_messages_response: InternalMessagesResponse | None = None,
3867
+ reporter: Reporter | None = None,
3868
+ quiet: bool | None = None,
3817
3869
  *,
3818
- settings: "Settings",
3819
- printer: Union["PrinterTerm", "PrinterJupyter"],
3870
+ settings: Settings,
3871
+ printer: printer.Printer,
3820
3872
  ) -> None:
3821
3873
  Run._footer_history_summary_info(
3822
3874
  history=sampled_history,
@@ -3848,177 +3900,51 @@ class Run:
3848
3900
  reporter=reporter, quiet=quiet, settings=settings, printer=printer
3849
3901
  )
3850
3902
 
3851
- # fixme: Temporary hack until we move to rich which allows multiple spinners
3852
3903
  @staticmethod
3853
- def _footer_file_pusher_status_info(
3854
- poll_exit_responses: Optional[
3855
- Union[PollExitResponse, List[Optional[PollExitResponse]]]
3856
- ] = None,
3857
- *,
3858
- printer: Union["PrinterTerm", "PrinterJupyter"],
3859
- ) -> None:
3860
- if not poll_exit_responses:
3861
- return
3862
- if isinstance(poll_exit_responses, PollExitResponse):
3863
- Run._footer_single_run_file_pusher_status_info(
3864
- poll_exit_responses, printer=printer
3865
- )
3866
- elif isinstance(poll_exit_responses, list):
3867
- poll_exit_responses_list = poll_exit_responses
3868
- assert all(
3869
- response is None or isinstance(response, PollExitResponse)
3870
- for response in poll_exit_responses_list
3871
- )
3872
- if len(poll_exit_responses_list) == 0:
3873
- return
3874
- elif len(poll_exit_responses_list) == 1:
3875
- Run._footer_single_run_file_pusher_status_info(
3876
- poll_exit_responses_list[0], printer=printer
3877
- )
3878
- else:
3879
- Run._footer_multiple_runs_file_pusher_status_info(
3880
- poll_exit_responses_list, printer=printer
3881
- )
3882
- else:
3883
- logger.error(
3884
- f"Got the type `{type(poll_exit_responses)}` for `poll_exit_responses`. "
3885
- "Expected either None, PollExitResponse or a List[Union[PollExitResponse, None]]"
3886
- )
3887
-
3888
- @staticmethod
3889
- def _footer_single_run_file_pusher_status_info(
3890
- poll_exit_response: Optional[PollExitResponse] = None,
3904
+ def _footer_sync_info(
3905
+ poll_exit_response: PollExitResponse | None = None,
3906
+ quiet: bool | None = None,
3891
3907
  *,
3892
- printer: Union["PrinterTerm", "PrinterJupyter"],
3908
+ settings: Settings,
3909
+ printer: printer.Printer,
3893
3910
  ) -> None:
3894
- # todo: is this same as settings._offline?
3895
- if not poll_exit_response:
3911
+ if settings.silent:
3896
3912
  return
3897
3913
 
3898
- stats = poll_exit_response.pusher_stats
3899
-
3900
- megabyte = wandb.util.POW_2_BYTES[2][1]
3901
- line = (
3902
- f"{stats.uploaded_bytes / megabyte:.3f} MB"
3903
- f" of {stats.total_bytes / megabyte:.3f} MB uploaded"
3904
- )
3905
- if stats.deduped_bytes > 0:
3906
- line += f" ({stats.deduped_bytes / megabyte:.3f} MB deduped)"
3907
- line += "\r"
3908
-
3909
- if stats.total_bytes > 0:
3910
- printer.progress_update(line, stats.uploaded_bytes / stats.total_bytes)
3911
- else:
3912
- printer.progress_update(line, 1.0)
3913
-
3914
- if poll_exit_response.done:
3915
- printer.progress_close()
3916
-
3917
- if stats.total_bytes > 0:
3918
- dedupe_fraction = stats.deduped_bytes / float(stats.total_bytes)
3919
- else:
3920
- dedupe_fraction = 0
3921
-
3922
- if stats.deduped_bytes > 0.01:
3914
+ if settings._offline:
3915
+ if not quiet and not settings.quiet:
3923
3916
  printer.display(
3924
- f"W&B sync reduced upload amount by {dedupe_fraction:.1%}"
3917
+ [
3918
+ "You can sync this run to the cloud by running:",
3919
+ printer.code(f"wandb sync {settings.sync_dir}"),
3920
+ ],
3925
3921
  )
3926
-
3927
- @staticmethod
3928
- def _footer_multiple_runs_file_pusher_status_info(
3929
- poll_exit_responses: List[Optional[PollExitResponse]],
3930
- *,
3931
- printer: Union["PrinterTerm", "PrinterJupyter"],
3932
- ) -> None:
3933
- # todo: is this same as settings._offline?
3934
- if not all(poll_exit_responses):
3935
3922
  return
3936
3923
 
3937
- megabyte = wandb.util.POW_2_BYTES[2][1]
3938
- total_files: int = sum(
3939
- sum(
3940
- [
3941
- response.file_counts.wandb_count,
3942
- response.file_counts.media_count,
3943
- response.file_counts.artifact_count,
3944
- response.file_counts.other_count,
3945
- ]
3924
+ info = []
3925
+ if settings.run_name and settings.run_url:
3926
+ info.append(
3927
+ f"{printer.emoji('rocket')} View run {printer.name(settings.run_name)} at: {printer.link(settings.run_url)}"
3946
3928
  )
3947
- for response in poll_exit_responses
3948
- if response is not None and response.file_counts is not None
3949
- )
3950
- uploaded = sum(
3951
- response.pusher_stats.uploaded_bytes
3952
- for response in poll_exit_responses
3953
- if response is not None and response.pusher_stats is not None
3954
- )
3955
- total = sum(
3956
- response.pusher_stats.total_bytes
3957
- for response in poll_exit_responses
3958
- if response is not None and response.pusher_stats is not None
3959
- )
3960
-
3961
- line = (
3962
- f"Processing {len(poll_exit_responses)} runs with {total_files} files "
3963
- f"({uploaded/megabyte :.2f} MB/{total/megabyte :.2f} MB)\r"
3964
- )
3965
- # line = "{}{:<{max_len}}\r".format(line, " ", max_len=(80 - len(line)))
3966
- printer.progress_update(line) # type:ignore[call-arg]
3967
-
3968
- done = all(
3969
- [
3970
- poll_exit_response.done
3971
- for poll_exit_response in poll_exit_responses
3972
- if poll_exit_response
3973
- ]
3974
- )
3975
- if done:
3976
- printer.progress_close()
3977
-
3978
- @staticmethod
3979
- def _footer_sync_info(
3980
- poll_exit_response: Optional[PollExitResponse] = None,
3981
- quiet: Optional[bool] = None,
3982
- *,
3983
- settings: "Settings",
3984
- printer: Union["PrinterTerm", "PrinterJupyter"],
3985
- ) -> None:
3986
- if settings.silent:
3987
- return
3988
-
3989
- if settings._offline:
3990
- printer.display(
3991
- [
3992
- "You can sync this run to the cloud by running:",
3993
- printer.code(f"wandb sync {settings.sync_dir}"),
3994
- ],
3995
- off=(quiet or settings.quiet),
3929
+ if settings.project_url:
3930
+ info.append(
3931
+ f"{printer.emoji('star')} View project at: {printer.link(settings.project_url)}"
3996
3932
  )
3997
- else:
3998
- info = []
3999
- if settings.run_name and settings.run_url:
4000
- info.append(
4001
- f"{printer.emoji('rocket')} View run {printer.name(settings.run_name)} at: {printer.link(settings.run_url)}"
4002
- )
4003
- if settings.project_url:
4004
- info.append(
4005
- f"{printer.emoji('star')} View project at: {printer.link(settings.project_url)}"
4006
- )
4007
- if poll_exit_response and poll_exit_response.file_counts:
4008
- logger.info("logging synced files")
4009
- file_counts = poll_exit_response.file_counts
4010
- info.append(
4011
- f"Synced {file_counts.wandb_count} W&B file(s), {file_counts.media_count} media file(s), "
4012
- f"{file_counts.artifact_count} artifact file(s) and {file_counts.other_count} other file(s)",
4013
- )
4014
- printer.display(info)
3933
+ if poll_exit_response and poll_exit_response.file_counts:
3934
+ logger.info("logging synced files")
3935
+ file_counts = poll_exit_response.file_counts
3936
+ info.append(
3937
+ f"Synced {file_counts.wandb_count} W&B file(s), {file_counts.media_count} media file(s), "
3938
+ f"{file_counts.artifact_count} artifact file(s) and {file_counts.other_count} other file(s)",
3939
+ )
3940
+ printer.display(info)
4015
3941
 
4016
3942
  @staticmethod
4017
3943
  def _footer_log_dir_info(
4018
- quiet: Optional[bool] = None,
3944
+ quiet: bool | None = None,
4019
3945
  *,
4020
- settings: "Settings",
4021
- printer: Union["PrinterTerm", "PrinterJupyter"],
3946
+ settings: Settings,
3947
+ printer: printer.Printer,
4022
3948
  ) -> None:
4023
3949
  if (quiet or settings.quiet) or settings.silent:
4024
3950
  return
@@ -4032,12 +3958,12 @@ class Run:
4032
3958
 
4033
3959
  @staticmethod
4034
3960
  def _footer_history_summary_info(
4035
- history: Optional["SampledHistoryResponse"] = None,
4036
- summary: Optional["GetSummaryResponse"] = None,
4037
- quiet: Optional[bool] = None,
3961
+ history: SampledHistoryResponse | None = None,
3962
+ summary: GetSummaryResponse | None = None,
3963
+ quiet: bool | None = None,
4038
3964
  *,
4039
- settings: "Settings",
4040
- printer: Union["PrinterTerm", "PrinterJupyter"],
3965
+ settings: Settings,
3966
+ printer: printer.Printer,
4041
3967
  ) -> None:
4042
3968
  if (quiet or settings.quiet) or settings.silent:
4043
3969
  return
@@ -4103,11 +4029,11 @@ class Run:
4103
4029
 
4104
4030
  @staticmethod
4105
4031
  def _footer_internal_messages(
4106
- internal_messages_response: Optional["InternalMessagesResponse"] = None,
4107
- quiet: Optional[bool] = None,
4032
+ internal_messages_response: InternalMessagesResponse | None = None,
4033
+ quiet: bool | None = None,
4108
4034
  *,
4109
- settings: "Settings",
4110
- printer: Union["PrinterTerm", "PrinterJupyter"],
4035
+ settings: Settings,
4036
+ printer: printer.Printer,
4111
4037
  ) -> None:
4112
4038
  if (quiet or settings.quiet) or settings.silent:
4113
4039
  return
@@ -4121,9 +4047,9 @@ class Run:
4121
4047
  @staticmethod
4122
4048
  def _footer_notify_wandb_core(
4123
4049
  *,
4124
- quiet: Optional[bool] = None,
4125
- settings: "Settings",
4126
- printer: Union["PrinterTerm", "PrinterJupyter"],
4050
+ quiet: bool | None = None,
4051
+ settings: Settings,
4052
+ printer: printer.Printer,
4127
4053
  ) -> None:
4128
4054
  """Prints a message advertising the upcoming core release."""
4129
4055
  if quiet or not settings._require_legacy_service:
@@ -4138,11 +4064,11 @@ class Run:
4138
4064
 
4139
4065
  @staticmethod
4140
4066
  def _footer_reporter_warn_err(
4141
- reporter: Optional[Reporter] = None,
4142
- quiet: Optional[bool] = None,
4067
+ reporter: Reporter | None = None,
4068
+ quiet: bool | None = None,
4143
4069
  *,
4144
- settings: "Settings",
4145
- printer: Union["PrinterTerm", "PrinterJupyter"],
4070
+ settings: Settings,
4071
+ printer: printer.Printer,
4146
4072
  ) -> None:
4147
4073
  if (quiet or settings.quiet) or settings.silent:
4148
4074
  return
@@ -4168,10 +4094,10 @@ class Run:
4168
4094
  # We define this outside of the run context to support restoring before init
4169
4095
  def restore(
4170
4096
  name: str,
4171
- run_path: Optional[str] = None,
4097
+ run_path: str | None = None,
4172
4098
  replace: bool = False,
4173
- root: Optional[str] = None,
4174
- ) -> Union[None, TextIO]:
4099
+ root: str | None = None,
4100
+ ) -> None | TextIO:
4175
4101
  """Download the specified file from cloud storage.
4176
4102
 
4177
4103
  File is placed into the current directory or run directory.
@@ -4229,7 +4155,7 @@ except AttributeError:
4229
4155
  pass
4230
4156
 
4231
4157
 
4232
- def finish(exit_code: Optional[int] = None, quiet: Optional[bool] = None) -> None:
4158
+ def finish(exit_code: int | None = None, quiet: bool | None = None) -> None:
4233
4159
  """Mark a run as finished, and finish uploading all data.
4234
4160
 
4235
4161
  This is used when creating multiple runs in the same process.