lightning-sdk 2025.10.8__py3-none-any.whl → 2025.10.22__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (96) hide show
  1. lightning_sdk/__init__.py +6 -3
  2. lightning_sdk/api/base_studio_api.py +13 -9
  3. lightning_sdk/api/cloud_account_api.py +0 -2
  4. lightning_sdk/api/license_api.py +26 -59
  5. lightning_sdk/api/studio_api.py +15 -2
  6. lightning_sdk/base_studio.py +30 -17
  7. lightning_sdk/cli/base_studio/list.py +1 -3
  8. lightning_sdk/cli/entrypoint.py +8 -34
  9. lightning_sdk/cli/studio/connect.py +42 -92
  10. lightning_sdk/cli/studio/create.py +23 -1
  11. lightning_sdk/cli/studio/start.py +12 -2
  12. lightning_sdk/cli/utils/get_base_studio.py +24 -0
  13. lightning_sdk/cli/utils/handle_machine_and_gpus_args.py +71 -0
  14. lightning_sdk/cli/utils/logging.py +121 -0
  15. lightning_sdk/cli/utils/ssh_connection.py +1 -1
  16. lightning_sdk/constants.py +1 -0
  17. lightning_sdk/helpers.py +53 -34
  18. lightning_sdk/job/job.py +5 -0
  19. lightning_sdk/job/v1.py +8 -0
  20. lightning_sdk/job/v2.py +8 -0
  21. lightning_sdk/lightning_cloud/login.py +260 -10
  22. lightning_sdk/lightning_cloud/openapi/__init__.py +30 -3
  23. lightning_sdk/lightning_cloud/openapi/api/__init__.py +1 -0
  24. lightning_sdk/lightning_cloud/openapi/api/assistants_service_api.py +19 -19
  25. lightning_sdk/lightning_cloud/openapi/api/auth_service_api.py +97 -0
  26. lightning_sdk/lightning_cloud/openapi/api/cloud_space_service_api.py +105 -0
  27. lightning_sdk/lightning_cloud/openapi/api/k8_s_cluster_service_api.py +1463 -240
  28. lightning_sdk/lightning_cloud/openapi/api/product_license_service_api.py +108 -108
  29. lightning_sdk/lightning_cloud/openapi/api/sdk_command_history_service_api.py +141 -0
  30. lightning_sdk/lightning_cloud/openapi/models/__init__.py +29 -3
  31. lightning_sdk/lightning_cloud/openapi/models/cloudspace_id_visibility_body.py +27 -1
  32. lightning_sdk/lightning_cloud/openapi/models/cluster_id_metrics_body.py +53 -1
  33. lightning_sdk/lightning_cloud/openapi/models/create_machine_request_represents_the_request_to_create_a_machine.py +27 -1
  34. lightning_sdk/lightning_cloud/openapi/models/deployments_id_body.py +27 -1
  35. lightning_sdk/lightning_cloud/openapi/models/externalv1_cloud_space_instance_status.py +79 -1
  36. lightning_sdk/lightning_cloud/openapi/models/id_fork_body1.py +27 -1
  37. lightning_sdk/lightning_cloud/openapi/models/id_transfer_body.py +53 -1
  38. lightning_sdk/lightning_cloud/openapi/models/incident_id_messages_body.py +149 -0
  39. lightning_sdk/lightning_cloud/openapi/models/incidents_id_body.py +279 -0
  40. lightning_sdk/lightning_cloud/openapi/models/license_key_validate_body.py +123 -0
  41. lightning_sdk/lightning_cloud/openapi/models/messages_message_id_body.py +149 -0
  42. lightning_sdk/lightning_cloud/openapi/models/project_id_incidents_body.py +279 -0
  43. lightning_sdk/lightning_cloud/openapi/models/projects_id_body.py +27 -1
  44. lightning_sdk/lightning_cloud/openapi/models/storage_complete_body.py +15 -15
  45. lightning_sdk/lightning_cloud/openapi/models/v1_cloud_space_transfer_metadata.py +53 -1
  46. lightning_sdk/lightning_cloud/openapi/models/v1_create_license_request.py +175 -0
  47. lightning_sdk/lightning_cloud/openapi/models/v1_create_project_request.py +27 -1
  48. lightning_sdk/lightning_cloud/openapi/models/v1_create_sdk_command_history_request.py +253 -0
  49. lightning_sdk/lightning_cloud/openapi/models/v1_create_sdk_command_history_response.py +97 -0
  50. lightning_sdk/lightning_cloud/openapi/models/v1_delete_incident_message_response.py +97 -0
  51. lightning_sdk/lightning_cloud/openapi/models/v1_delete_incident_response.py +97 -0
  52. lightning_sdk/lightning_cloud/openapi/models/v1_delete_license_response.py +97 -0
  53. lightning_sdk/lightning_cloud/openapi/models/v1_deployment.py +27 -1
  54. lightning_sdk/lightning_cloud/openapi/models/v1_external_cluster_spec.py +27 -1
  55. lightning_sdk/lightning_cloud/openapi/models/v1_get_cloud_space_transfer_estimate_response.py +149 -0
  56. lightning_sdk/lightning_cloud/openapi/models/v1_group_pod_metrics.py +1241 -0
  57. lightning_sdk/lightning_cloud/openapi/models/v1_incident.py +565 -0
  58. lightning_sdk/lightning_cloud/openapi/models/v1_incident_detail.py +149 -0
  59. lightning_sdk/lightning_cloud/openapi/models/v1_incident_event.py +27 -1
  60. lightning_sdk/lightning_cloud/openapi/models/v1_incident_message.py +253 -0
  61. lightning_sdk/lightning_cloud/openapi/models/v1_incident_type.py +1 -0
  62. lightning_sdk/lightning_cloud/openapi/models/v1_job.py +53 -1
  63. lightning_sdk/lightning_cloud/openapi/models/v1_job_spec.py +27 -1
  64. lightning_sdk/lightning_cloud/openapi/models/v1_kai_scheduler_queue_metrics.py +627 -0
  65. lightning_sdk/lightning_cloud/openapi/models/v1_license.py +227 -0
  66. lightning_sdk/lightning_cloud/openapi/models/v1_list_group_pod_metrics_response.py +123 -0
  67. lightning_sdk/lightning_cloud/openapi/models/v1_list_incident_messages_response.py +149 -0
  68. lightning_sdk/lightning_cloud/openapi/models/v1_list_incidents_response.py +149 -0
  69. lightning_sdk/lightning_cloud/openapi/models/v1_list_kai_scheduler_queues_metrics_response.py +123 -0
  70. lightning_sdk/lightning_cloud/openapi/models/{v1_list_product_licenses_response.py → v1_list_license_response.py} +16 -16
  71. lightning_sdk/lightning_cloud/openapi/models/v1_machine.py +79 -1
  72. lightning_sdk/lightning_cloud/openapi/models/v1_membership.py +27 -1
  73. lightning_sdk/lightning_cloud/openapi/models/v1_project_membership.py +27 -1
  74. lightning_sdk/lightning_cloud/openapi/models/v1_project_settings.py +27 -1
  75. lightning_sdk/lightning_cloud/openapi/models/v1_resource_visibility.py +1 -27
  76. lightning_sdk/lightning_cloud/openapi/models/v1_sdk_command_history_severity.py +104 -0
  77. lightning_sdk/lightning_cloud/openapi/models/v1_sdk_command_history_type.py +104 -0
  78. lightning_sdk/lightning_cloud/openapi/models/v1_server_alert_type.py +1 -0
  79. lightning_sdk/lightning_cloud/openapi/models/v1_slack_notifier.py +53 -1
  80. lightning_sdk/lightning_cloud/openapi/models/v1_token_login_request.py +123 -0
  81. lightning_sdk/lightning_cloud/openapi/models/v1_token_login_response.py +123 -0
  82. lightning_sdk/lightning_cloud/openapi/models/v1_token_owner_type.py +104 -0
  83. lightning_sdk/lightning_cloud/openapi/models/v1_user_features.py +209 -131
  84. lightning_sdk/lightning_cloud/openapi/models/{v1_product_license_check_response.py → v1_validate_license_response.py} +21 -21
  85. lightning_sdk/lightning_cloud/rest_client.py +48 -45
  86. lightning_sdk/machine.py +2 -1
  87. lightning_sdk/studio.py +22 -2
  88. lightning_sdk/utils/license.py +13 -0
  89. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/METADATA +1 -1
  90. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/RECORD +94 -64
  91. lightning_sdk/lightning_cloud/openapi/models/v1_product_license.py +0 -435
  92. lightning_sdk/services/license.py +0 -363
  93. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/LICENSE +0 -0
  94. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/WHEEL +0 -0
  95. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/entry_points.txt +0 -0
  96. {lightning_sdk-2025.10.8.dist-info → lightning_sdk-2025.10.22.dist-info}/top_level.txt +0 -0
@@ -4,6 +4,7 @@ from typing import Optional
4
4
 
5
5
  import click
6
6
 
7
+ from lightning_sdk.cli.utils.handle_machine_and_gpus_args import handle_machine_and_gpus_args
7
8
  from lightning_sdk.cli.utils.richt_print import studio_name_link
8
9
  from lightning_sdk.cli.utils.save_to_config import save_studio_to_config
9
10
  from lightning_sdk.cli.utils.studio_selection import StudiosMenu
@@ -32,8 +33,7 @@ from lightning_sdk.studio import VM, Studio
32
33
  @click.option(
33
34
  "--cloud-provider",
34
35
  help=(
35
- "The cloud provider to start the studio on. Defaults to teamspace default. "
36
- "Only used if --create is specified."
36
+ "The cloud provider to start the studio on. Defaults to teamspace default. Only used if --create is specified."
37
37
  ),
38
38
  type=click.Choice(m.name for m in list(CloudProvider)),
39
39
  )
@@ -42,11 +42,17 @@ from lightning_sdk.studio import VM, Studio
42
42
  help="The cloud account to start the studio on. Defaults to teamspace default. Only used if --create is specified.",
43
43
  type=click.STRING,
44
44
  )
45
+ @click.option(
46
+ "--gpus",
47
+ help="The number and type of GPUs to start the studio on (format: TYPE:COUNT, e.g. L4:4)",
48
+ type=click.STRING,
49
+ )
45
50
  def start_studio(
46
51
  name: Optional[str] = None,
47
52
  teamspace: Optional[str] = None,
48
53
  create: bool = False,
49
54
  machine: str = "CPU",
55
+ gpus: Optional[str] = None,
50
56
  interruptible: bool = False,
51
57
  cloud_provider: Optional[str] = None,
52
58
  cloud_account: Optional[str] = None,
@@ -62,6 +68,7 @@ def start_studio(
62
68
  teamspace=teamspace,
63
69
  create=create,
64
70
  machine=machine,
71
+ gpus=gpus,
65
72
  interruptible=interruptible,
66
73
  cloud_provider=cloud_provider,
67
74
  cloud_account=cloud_account,
@@ -74,6 +81,7 @@ def start_impl(
74
81
  teamspace: Optional[str],
75
82
  create: bool,
76
83
  machine: str,
84
+ gpus: Optional[str],
77
85
  interruptible: bool,
78
86
  cloud_provider: Optional[str],
79
87
  cloud_account: Optional[str],
@@ -98,6 +106,8 @@ def start_impl(
98
106
  cloud_account=cloud_account,
99
107
  )
100
108
 
109
+ machine = handle_machine_and_gpus_args(machine, gpus)
110
+
101
111
  save_studio_to_config(studio)
102
112
 
103
113
  Studio.show_progress = True
@@ -0,0 +1,24 @@
1
+ from typing import Optional
2
+
3
+ from lightning_sdk.base_studio import BaseStudio
4
+
5
+
6
+ def get_base_studio_id(studio_type: Optional[str]) -> Optional[str]:
7
+ base_studios = BaseStudio()
8
+ base_studios = base_studios.list()
9
+ template_id = None
10
+
11
+ if base_studios and len(base_studios):
12
+ # if not specified by user, use the first existing template studio
13
+ template_id = base_studios[0].id
14
+ # else, try to match the provided studio_type to base studio name
15
+ if studio_type:
16
+ normalized_studio_type = studio_type.lower().replace(" ", "-")
17
+ match = next(
18
+ (s for s in base_studios if s.name.lower().replace(" ", "-") == normalized_studio_type),
19
+ None,
20
+ )
21
+ if match:
22
+ template_id = match.id
23
+
24
+ return template_id
@@ -0,0 +1,71 @@
1
+ from typing import Dict, Optional, Set
2
+
3
+ import click
4
+
5
+ from lightning_sdk.machine import Machine
6
+
7
+ DEFAULT_MACHINE = "CPU"
8
+
9
+
10
+ def _split_gpus_spec(gpus: str) -> tuple[str, int]:
11
+ machine_name, machine_val = gpus.split(":", 1)
12
+ machine_name = machine_name.strip()
13
+ machine_val = machine_val.strip()
14
+
15
+ if not machine_val.isdigit() or int(machine_val) <= 0:
16
+ raise ValueError(f"Invalid GPU count '{machine_val}'. Must be a positive integer.")
17
+
18
+ machine_num = int(machine_val)
19
+ return machine_name, machine_num
20
+
21
+
22
+ def _construct_available_gpus(machine_options: Dict[str, str]) -> Set[str]:
23
+ # returns available gpus:count
24
+ available_gpus = set()
25
+ for v in machine_options.values():
26
+ if "_X_" in v:
27
+ gpu_type_num = v.replace("_X_", ":")
28
+ available_gpus.add(gpu_type_num)
29
+ else:
30
+ available_gpus.add(v)
31
+ return available_gpus
32
+
33
+
34
+ def _get_machine_from_gpus(gpus: str) -> Machine:
35
+ machine_name = gpus
36
+ machine_num = 1
37
+
38
+ if ":" in gpus:
39
+ machine_name, machine_num = _split_gpus_spec(gpus)
40
+
41
+ machine_options = {
42
+ m.name.lower(): m.name for m in Machine.__dict__.values() if isinstance(m, Machine) and m._include_in_cli
43
+ }
44
+
45
+ if machine_num == 1:
46
+ # e.g. gpus=L4 or gpus=L4:1
47
+ gpu_key = machine_name.lower()
48
+ try:
49
+ return machine_options[gpu_key]
50
+ except KeyError:
51
+ available = ", ".join(_construct_available_gpus(machine_options))
52
+ raise ValueError(f"Invalid GPU type '{machine_name}'. Available options: {available}") from None
53
+
54
+ # Else: e.g. gpus=L4:4
55
+ gpu_key = f"{machine_name.lower()}_x_{machine_num}"
56
+ try:
57
+ return machine_options[gpu_key]
58
+ except KeyError:
59
+ available = ", ".join(_construct_available_gpus(machine_options))
60
+ raise ValueError(f"Invalid GPU configuration '{gpus}'. Available options: {available}") from None
61
+
62
+
63
+ def handle_machine_and_gpus_args(machine: Optional[str], gpus: Optional[str]) -> str:
64
+ if machine and gpus:
65
+ raise click.UsageError("Options --machine and --gpus are mutually exclusive. Provide only one.")
66
+ elif gpus:
67
+ machine = _get_machine_from_gpus(gpus.strip())
68
+ elif not machine:
69
+ machine = DEFAULT_MACHINE
70
+
71
+ return machine
@@ -0,0 +1,121 @@
1
+ import shlex
2
+ import sys
3
+ import traceback
4
+ from contextlib import suppress
5
+ from time import time
6
+ from types import TracebackType
7
+ from typing import Optional, Type
8
+
9
+ import click
10
+ from rich.console import Group
11
+ from rich.panel import Panel
12
+ from rich.syntax import Syntax
13
+ from rich.text import Text
14
+
15
+ from lightning_sdk.cli.utils import rich_to_str
16
+ from lightning_sdk.constants import _LIGHTNING_DEBUG
17
+ from lightning_sdk.lightning_cloud.openapi.models.v1_create_sdk_command_history_request import (
18
+ V1CreateSDKCommandHistoryRequest,
19
+ )
20
+ from lightning_sdk.lightning_cloud.openapi.models.v1_sdk_command_history_severity import V1SDKCommandHistorySeverity
21
+ from lightning_sdk.lightning_cloud.openapi.models.v1_sdk_command_history_type import V1SDKCommandHistoryType
22
+ from lightning_sdk.lightning_cloud.rest_client import LightningClient
23
+
24
+
25
+ def _log_command(message: str = "", duration: int = 0, error: Optional[str] = None) -> None:
26
+ original_command = " ".join(shlex.quote(arg) for arg in sys.argv)
27
+ client = LightningClient(retry=False, max_tries=0)
28
+
29
+ body = V1CreateSDKCommandHistoryRequest(
30
+ command=original_command,
31
+ duration=duration,
32
+ message=message,
33
+ project_id=None,
34
+ severity=V1SDKCommandHistorySeverity.INFO,
35
+ type=V1SDKCommandHistoryType.CLI,
36
+ )
37
+
38
+ if error:
39
+ body.severity = V1SDKCommandHistorySeverity.WARNING if error == "0" else V1SDKCommandHistorySeverity.ERROR
40
+ body.message = body.message + f" | Error: {error}"
41
+
42
+ # limit characters
43
+ body.message = body.message[:1000]
44
+
45
+ with suppress(Exception):
46
+ client.s_dk_command_history_service_create_sdk_command_history(body=body)
47
+
48
+
49
+ def _notify_exception(exception_type: Type[BaseException], value: BaseException, tb: TracebackType) -> None:
50
+ """CLI won't show tracebacks, just print the exception message."""
51
+ message = str(value.args[0]) if value.args else str(value) or "An unknown error occurred"
52
+
53
+ error_text = Text()
54
+ error_text.append(f"{exception_type.__name__}: ", style="bold red")
55
+ error_text.append(message, style="white")
56
+
57
+ renderables = [error_text]
58
+
59
+ if _LIGHTNING_DEBUG:
60
+ tb_text = "".join(traceback.format_exception(exception_type, value, tb))
61
+ renderables.append(Text("\n\nFull traceback:\n", style="bold yellow"))
62
+ renderables.append(Syntax(tb_text, "python", theme="monokai light", line_numbers=False, word_wrap=True))
63
+ else:
64
+ renderables.append(Text("\n\n🐞 To view the full traceback, set: LIGHTNING_DEBUG=1"))
65
+
66
+ renderables.append(Text("\n📘 Need help? Run: lightning <command> --help", style="cyan"))
67
+
68
+ text = rich_to_str(Panel(Group(*renderables), title="⚡ Lightning CLI Error", border_style="red"))
69
+ click.echo(text, color=True)
70
+
71
+
72
+ def logging_excepthook(exception_type: Type[BaseException], value: BaseException, tb: TracebackType) -> None:
73
+ try:
74
+ tb_str = "".join(traceback.format_exception(exception_type, value, tb))
75
+ ctx = click.get_current_context(silent=True)
76
+ command_context = ctx.command_path if ctx else "outside_command_context"
77
+
78
+ message = (
79
+ f"Command: {command_context} | Type: {exception_type.__name__!s} | Value: {value!s} | Traceback: {tb_str}"
80
+ )
81
+ _log_command(message=message)
82
+ finally:
83
+ _notify_exception(exception_type, value, tb)
84
+
85
+
86
+ class CommandLoggingGroup(click.Group):
87
+ def _format_ctx(self, ctx: click.Context) -> str:
88
+ parts = []
89
+ for k, v in ctx.params.items():
90
+ if v is True:
91
+ parts.append(f"--{k}")
92
+ elif v is False or v is None:
93
+ continue
94
+ else:
95
+ parts.append(f"--{k} {v}")
96
+ params = " ".join(parts)
97
+ args = " ".join(ctx.args)
98
+ return (
99
+ f"""Commands: {ctx.command_path} | Subcommand: {ctx.invoked_subcommand} | Params: {params} | Args:{args}"""
100
+ )
101
+
102
+ def invoke(self, ctx: click.Context) -> any:
103
+ """Overrides the default invoke to wrap command execution with tracking."""
104
+ start_time = time()
105
+ error_message = None
106
+
107
+ try:
108
+ return super().invoke(ctx)
109
+ except click.ClickException as e:
110
+ error_message = str(e)
111
+ e.show()
112
+ ctx.exit(e.exit_code)
113
+ except Exception as e:
114
+ error_message = str(e)
115
+ raise
116
+ finally:
117
+ _log_command(
118
+ message=self._format_ctx(ctx),
119
+ duration=int(time() - start_time),
120
+ error=error_message,
121
+ )
@@ -16,7 +16,7 @@ def configure_ssh_internal(force_download: bool = False) -> str:
16
16
 
17
17
 
18
18
  def download_ssh_keys(
19
- api_key: str | None,
19
+ api_key: Optional[str],
20
20
  force_download: bool = False,
21
21
  ssh_key_name: str = "lightning_rsa",
22
22
  ) -> str:
@@ -29,3 +29,4 @@ class Store:
29
29
 
30
30
 
31
31
  __GLOBAL_LIGHTNING_UNIQUE_IDS_STORE__ = Store()
32
+ _LIGHTNING_DISABLE_VERSION_CHECK = int(os.getenv("LIGHTNING_DISABLE_VERSION_CHECK", "0"))
lightning_sdk/helpers.py CHANGED
@@ -1,4 +1,3 @@
1
- import functools
2
1
  import importlib
3
2
  import os
4
3
  import sys
@@ -10,48 +9,68 @@ import tqdm
10
9
  import tqdm.std
11
10
  from packaging import version as packaging_version
12
11
 
13
- __package_name__ = "lightning-sdk"
12
+ from lightning_sdk.constants import _LIGHTNING_DISABLE_VERSION_CHECK
14
13
 
15
14
 
16
- @functools.lru_cache(maxsize=1)
17
- def _get_newer_version(curr_version: str) -> Optional[str]:
18
- """Check PyPI for newer versions of ``lightning-sdk``.
19
-
20
- Returning the newest version if different from the current or ``None`` otherwise.
15
+ class VersionChecker:
16
+ """Handles version checking and upgrade prompts for lightning-sdk.
21
17
 
18
+ This class ensures that version check warnings are only shown once per session,
19
+ preventing duplicate warnings in multithreaded scenarios.
22
20
  """
23
- if packaging_version.parse(curr_version).is_prerelease:
24
- return None
25
- try:
26
- response = requests.get(f"https://pypi.org/pypi/{__package_name__}/json")
27
- response_json = response.json()
28
- releases = response_json["releases"]
29
- if curr_version not in releases:
30
- # Always return None if not installed from PyPI (e.g. dev versions)
21
+
22
+ def __init__(self, package_name: str = "lightning-sdk") -> None:
23
+ self.package_name = package_name
24
+ self._warning_shown = False
25
+ self._cached_version: Optional[str] = None
26
+
27
+ def _get_newer_version(self, curr_version: str) -> Optional[str]:
28
+ """Check PyPI for newer versions of ``lightning-sdk``.
29
+
30
+ Returning the newest version if different from the current or ``None`` otherwise.
31
+ """
32
+ if self._cached_version is not None:
33
+ return self._cached_version
34
+
35
+ if _LIGHTNING_DISABLE_VERSION_CHECK == 1 or packaging_version.parse(curr_version).is_prerelease:
36
+ self._cached_version = None
31
37
  return None
32
- latest_version = response_json["info"]["version"]
33
- parsed_version = packaging_version.parse(latest_version)
34
- is_invalid = response_json["info"]["yanked"] or parsed_version.is_devrelease or parsed_version.is_prerelease
35
- return None if curr_version == latest_version or is_invalid else latest_version
36
- except requests.exceptions.RequestException:
37
- return None
38
38
 
39
+ try:
40
+ response = requests.get(f"https://pypi.org/pypi/{self.package_name}/json")
41
+ response_json = response.json()
42
+ releases = response_json["releases"]
43
+ if curr_version not in releases:
44
+ # Always return None if not installed from PyPI (e.g. dev versions)
45
+ self._cached_version = None
46
+ return None
47
+ latest_version = response_json["info"]["version"]
48
+ parsed_version = packaging_version.parse(latest_version)
49
+ is_invalid = response_json["info"]["yanked"] or parsed_version.is_devrelease or parsed_version.is_prerelease
50
+ self._cached_version = None if curr_version == latest_version or is_invalid else latest_version
51
+ return self._cached_version
52
+ except requests.exceptions.RequestException:
53
+ self._cached_version = None
54
+ return None
39
55
 
40
- def _check_version_and_prompt_upgrade(curr_version: str) -> None:
41
- """Checks that the current version of ``lightning-sdk`` is the latest on PyPI.
56
+ def check_and_prompt_upgrade(self, curr_version: str) -> None:
57
+ """Checks that the current version of ``lightning-sdk`` is the latest on PyPI.
42
58
 
43
- If not, warn the user to upgrade ``lightning-sdk``.
59
+ If not, warn the user to upgrade ``lightning-sdk``.
60
+ Tracks if the warning has already been shown in this session to avoid duplicate warnings.
61
+ """
62
+ if self._warning_shown:
63
+ return
44
64
 
45
- """
46
- new_version = _get_newer_version(curr_version)
47
- if new_version:
48
- warnings.warn(
49
- f"A newer version of {__package_name__} is available ({new_version}). "
50
- f"Please consider upgrading with `pip install -U {__package_name__}`. "
51
- "Not all platform functionality can be guaranteed to work with the current version.",
52
- UserWarning,
53
- )
54
- return
65
+ new_version = self._get_newer_version(curr_version)
66
+ if new_version:
67
+ warnings.warn(
68
+ f"A newer version of {self.package_name} is available ({new_version}). "
69
+ f"Please consider upgrading with `pip install -U {self.package_name}`. "
70
+ "Not all platform functionality can be guaranteed to work with the current version.",
71
+ UserWarning,
72
+ )
73
+ self._warning_shown = True
55
74
 
56
75
 
57
76
  def _set_tqdm_envvars_noninteractive() -> None:
lightning_sdk/job/job.py CHANGED
@@ -268,6 +268,11 @@ class Job(_BaseJob):
268
268
  """The machine type the job is running on."""
269
269
  return self._internal_job.machine
270
270
 
271
+ @property
272
+ def public_ip(self) -> Optional[str]:
273
+ """The public IP address of the machine the job is running on."""
274
+ return self._internal_job.public_ip
275
+
271
276
  @property
272
277
  def artifact_path(self) -> Optional[str]:
273
278
  """Path to the artifacts created by the job within the distributed teamspace filesystem."""
lightning_sdk/job/v1.py CHANGED
@@ -181,6 +181,14 @@ class _JobV1(_BaseJob):
181
181
  """Get the machine the job is running on."""
182
182
  return self.work.machine
183
183
 
184
+ @property
185
+ def public_ip(self) -> Optional[str]:
186
+ """Get the public IP of the machine the job is running on."""
187
+ try:
188
+ return self._job.status.ip_address
189
+ except AttributeError:
190
+ return None
191
+
184
192
  @property
185
193
  def name(self) -> str:
186
194
  """The name of the job."""
lightning_sdk/job/v2.py CHANGED
@@ -173,6 +173,14 @@ class _JobV2(_BaseJob):
173
173
  _get_org_id(self.teamspace),
174
174
  )
175
175
 
176
+ @property
177
+ def public_ip(self) -> Optional[str]:
178
+ """Get the public IP of the machine the job is running on."""
179
+ try:
180
+ return self._job.public_ip_address
181
+ except AttributeError:
182
+ return None
183
+
176
184
  @property
177
185
  def artifact_path(self) -> Optional[str]:
178
186
  """The path to the artifacts of the job within the distributed teamspace filesystem."""