lightning-sdk 2025.8.6rc2__py3-none-any.whl → 2025.11.5__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.
- lightning_sdk/__init__.py +13 -7
- lightning_sdk/__version__.py +3 -0
- lightning_sdk/agents.py +2 -1
- lightning_sdk/ai_hub.py +2 -1
- lightning_sdk/api/__init__.py +2 -0
- lightning_sdk/api/base_studio_api.py +13 -9
- lightning_sdk/api/cloud_account_api.py +31 -10
- lightning_sdk/api/deployment_api.py +27 -1
- lightning_sdk/api/job_api.py +16 -12
- lightning_sdk/api/license_api.py +26 -59
- lightning_sdk/api/llm_api.py +25 -2
- lightning_sdk/api/mmt_api.py +5 -2
- lightning_sdk/api/studio_api.py +252 -55
- lightning_sdk/api/teamspace_api.py +106 -30
- lightning_sdk/api/user_api.py +56 -2
- lightning_sdk/api/utils.py +108 -18
- lightning_sdk/base_studio.py +51 -27
- lightning_sdk/cli/__init__.py +1 -0
- lightning_sdk/cli/base_studio/__init__.py +10 -0
- lightning_sdk/cli/base_studio/list.py +43 -0
- lightning_sdk/cli/config/__init__.py +14 -0
- lightning_sdk/cli/config/get.py +57 -0
- lightning_sdk/cli/config/set.py +92 -0
- lightning_sdk/cli/config/show.py +9 -0
- lightning_sdk/cli/entrypoint.py +71 -71
- lightning_sdk/cli/groups.py +56 -0
- lightning_sdk/cli/job/__init__.py +7 -0
- lightning_sdk/cli/{clusters_menu.py → legacy/clusters_menu.py} +9 -6
- lightning_sdk/cli/{configure.py → legacy/configure.py} +2 -2
- lightning_sdk/cli/{connect.py → legacy/connect.py} +2 -2
- lightning_sdk/cli/{create.py → legacy/create.py} +12 -14
- lightning_sdk/cli/{delete.py → legacy/delete.py} +5 -5
- lightning_sdk/cli/legacy/deploy/__init__.py +0 -0
- lightning_sdk/cli/{deploy → legacy/deploy}/_auth.py +5 -6
- lightning_sdk/cli/{deploy → legacy/deploy}/devbox.py +8 -2
- lightning_sdk/cli/{deploy → legacy/deploy}/serve.py +19 -8
- lightning_sdk/cli/{download.py → legacy/download.py} +14 -15
- lightning_sdk/cli/legacy/entrypoint.py +110 -0
- lightning_sdk/cli/{generate.py → legacy/generate.py} +1 -1
- lightning_sdk/cli/{inspection.py → legacy/inspection.py} +1 -1
- lightning_sdk/cli/{job_and_mmt_action.py → legacy/job_and_mmt_action.py} +6 -6
- lightning_sdk/cli/{jobs_menu.py → legacy/jobs_menu.py} +1 -1
- lightning_sdk/cli/{list.py → legacy/list.py} +12 -13
- lightning_sdk/cli/{mmts_menu.py → legacy/mmts_menu.py} +1 -1
- lightning_sdk/cli/{open.py → legacy/open.py} +4 -4
- lightning_sdk/cli/{start.py → legacy/start.py} +1 -0
- lightning_sdk/cli/{stop.py → legacy/stop.py} +1 -1
- lightning_sdk/cli/{switch.py → legacy/switch.py} +1 -0
- lightning_sdk/cli/{teamspace_menu.py → legacy/teamspace_menu.py} +1 -1
- lightning_sdk/cli/{upload.py → legacy/upload.py} +7 -8
- lightning_sdk/cli/license/__init__.py +14 -0
- lightning_sdk/cli/license/get.py +15 -0
- lightning_sdk/cli/license/list.py +45 -0
- lightning_sdk/cli/license/set.py +13 -0
- lightning_sdk/cli/mmt/__init__.py +7 -0
- lightning_sdk/cli/studio/__init__.py +24 -0
- lightning_sdk/cli/studio/connect.py +139 -0
- lightning_sdk/cli/studio/create.py +96 -0
- lightning_sdk/cli/studio/delete.py +49 -0
- lightning_sdk/cli/studio/list.py +85 -0
- lightning_sdk/cli/studio/ssh.py +64 -0
- lightning_sdk/cli/studio/start.py +115 -0
- lightning_sdk/cli/studio/stop.py +45 -0
- lightning_sdk/cli/studio/switch.py +66 -0
- lightning_sdk/cli/utils/__init__.py +7 -0
- lightning_sdk/cli/utils/cloud_account_map.py +10 -0
- lightning_sdk/cli/utils/get_base_studio.py +24 -0
- lightning_sdk/cli/utils/handle_machine_and_gpus_args.py +69 -0
- lightning_sdk/cli/utils/logging.py +122 -0
- lightning_sdk/cli/utils/owner_selection.py +110 -0
- lightning_sdk/cli/utils/resolve.py +28 -0
- lightning_sdk/cli/utils/richt_print.py +35 -0
- lightning_sdk/cli/utils/save_to_config.py +27 -0
- lightning_sdk/cli/utils/ssh_connection.py +59 -0
- lightning_sdk/cli/utils/studio_selection.py +113 -0
- lightning_sdk/cli/utils/teamspace_selection.py +125 -0
- lightning_sdk/cli/vm/__init__.py +20 -0
- lightning_sdk/cli/vm/create.py +33 -0
- lightning_sdk/cli/vm/delete.py +25 -0
- lightning_sdk/cli/vm/list.py +30 -0
- lightning_sdk/cli/vm/ssh.py +31 -0
- lightning_sdk/cli/vm/start.py +60 -0
- lightning_sdk/cli/vm/stop.py +25 -0
- lightning_sdk/cli/vm/switch.py +38 -0
- lightning_sdk/constants.py +1 -0
- lightning_sdk/deployment/__init__.py +2 -0
- lightning_sdk/deployment/deployment.py +17 -2
- lightning_sdk/helpers.py +56 -37
- lightning_sdk/job/base.py +21 -6
- lightning_sdk/job/job.py +13 -0
- lightning_sdk/job/v1.py +11 -0
- lightning_sdk/job/v2.py +12 -0
- lightning_sdk/lightning_cloud/login.py +320 -10
- lightning_sdk/lightning_cloud/openapi/__init__.py +113 -3
- lightning_sdk/lightning_cloud/openapi/api/__init__.py +3 -0
- lightning_sdk/lightning_cloud/openapi/api/assistants_service_api.py +713 -75
- lightning_sdk/lightning_cloud/openapi/api/auth_service_api.py +376 -0
- lightning_sdk/lightning_cloud/openapi/api/billing_service_api.py +191 -1
- lightning_sdk/lightning_cloud/openapi/api/cloud_space_environment_template_service_api.py +5 -1
- lightning_sdk/lightning_cloud/openapi/api/cloud_space_service_api.py +420 -0
- lightning_sdk/lightning_cloud/openapi/api/cloudy_service_api.py +0 -97
- lightning_sdk/lightning_cloud/openapi/api/cluster_service_api.py +420 -0
- lightning_sdk/lightning_cloud/openapi/api/data_connection_service_api.py +101 -0
- lightning_sdk/lightning_cloud/openapi/api/incidents_service_api.py +1058 -0
- lightning_sdk/lightning_cloud/openapi/api/jobs_service_api.py +121 -0
- lightning_sdk/lightning_cloud/openapi/api/k8_s_cluster_service_api.py +1742 -94
- lightning_sdk/lightning_cloud/openapi/api/markets_service_api.py +145 -0
- lightning_sdk/lightning_cloud/openapi/api/models_store_api.py +4 -4
- lightning_sdk/lightning_cloud/openapi/api/product_license_service_api.py +108 -108
- lightning_sdk/lightning_cloud/openapi/api/projects_service_api.py +105 -0
- lightning_sdk/lightning_cloud/openapi/api/schedules_service_api.py +347 -0
- lightning_sdk/lightning_cloud/openapi/api/sdk_command_history_service_api.py +141 -0
- lightning_sdk/lightning_cloud/openapi/api/storage_service_api.py +761 -1
- lightning_sdk/lightning_cloud/openapi/configuration.py +3 -19
- lightning_sdk/lightning_cloud/openapi/models/__init__.py +110 -3
- lightning_sdk/lightning_cloud/openapi/models/assistant_id_conversations_body.py +15 -15
- lightning_sdk/lightning_cloud/openapi/models/cloudspace_id_visibility_body.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/cluster_id_kubernetestemplates_body.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/cluster_id_metrics_body.py +131 -1
- lightning_sdk/lightning_cloud/openapi/models/create.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/create_machine_request_represents_the_request_to_create_a_machine.py +461 -0
- lightning_sdk/lightning_cloud/openapi/models/deployments_id_body.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/externalv1_cloud_space_instance_status.py +105 -1
- lightning_sdk/lightning_cloud/openapi/models/externalv1_user_status.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/id_codeconfig_body.py +3 -81
- lightning_sdk/lightning_cloud/openapi/models/id_fork_body1.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/id_render_body.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/id_sleepconfig_body.py +175 -0
- lightning_sdk/lightning_cloud/openapi/models/id_transfer_body.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/incident_id_messages_body.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/incidents_id_body.py +279 -0
- lightning_sdk/lightning_cloud/openapi/models/job_id_reportroutingtelemetry_body.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/kubernetestemplates_id_body.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/license_key_validate_body.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/message_id_actions_body.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/messages_message_id_body.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/metricsstream_create_body.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/models_model_id_body.py +109 -31
- lightning_sdk/lightning_cloud/openapi/models/models_model_id_body1.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/orgs_id_body.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/project_id_storage_body.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/project_id_storagetransfers_body.py +175 -0
- lightning_sdk/lightning_cloud/openapi/models/project_tab_management_messages.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/projects_id_body.py +79 -1
- lightning_sdk/lightning_cloud/openapi/models/storage_complete_body.py +41 -15
- lightning_sdk/lightning_cloud/openapi/models/storagetransfers_validate_body.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/update1.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/uploads_upload_id_body1.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/user_id_affiliatelinks_body.py +107 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_abort_storage_transfer_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_aggregated_pod_metrics.py +799 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_ai_pod_v1.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_assistant_session_daily_aggregated.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_billing_tier.py +0 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cancel_running_cloud_space_instance_transfer_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_cloud_provider.py +2 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_cloud_space_environment_template_config.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cloud_space_specialized_view.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_cloud_space_transfer_metadata.py +147 -17
- lightning_sdk/lightning_cloud/openapi/models/v1_cloudflare_v1.py +3 -29
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_accelerator.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_capacity_reservation.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_metrics.py +1527 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_security_options.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_spec.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_cluster_type.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_container_metrics.py +21 -21
- lightning_sdk/lightning_cloud/openapi/models/v1_create_incident_request.py +305 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_create_license_request.py +175 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_create_machine_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_create_model_metrics_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_create_project_request.py +79 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_create_sdk_command_history_request.py +253 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_create_sdk_command_history_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_daily_model_metrics.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_data_connection.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_delete_incident_message_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_delete_incident_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_delete_kubernetes_template_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_delete_license_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_delete_machine_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_deployment.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_external_cluster_spec.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_external_search_user.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_filestore_data_connection.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_filesystem_metric.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_firewall_rule.py +175 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_cloud_space_required_balance_status_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_cloud_space_transfer_estimate_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_latest_model_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_machine_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_market_pricing_response.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_model_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_temp_bucket_credentials_response.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_get_user_response.py +41 -15
- lightning_sdk/lightning_cloud/openapi/models/v1_google_cloud_direct_v1.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_group_node_metrics.py +1215 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_group_pod_metrics.py +1241 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_guest_login_request.py +177 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_guest_login_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_guest_user.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident.py +565 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident_detail.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident_event.py +591 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident_message.py +253 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident_severity.py +105 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_incident_type.py +108 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_job.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_job_spec.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_k8s_incident_indexes.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_kai_scheduler_queue_metrics.py +627 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_kubernetes_aws_config.py +279 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_kubernetes_direct_settings_v1.py +253 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_kubernetes_direct_v1.py +107 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_kubernetes_template.py +357 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_kubernetes_template_property.py +227 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_lambda_labs_direct_v1.py +29 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_license.py +227 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_lightning_elastic_cluster_v1.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_aggregated_pod_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_cluster_metric_timestamps_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_cluster_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_cluster_namespace_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_cluster_namespace_user_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_conversation_message_actions_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_filesystem_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_group_pod_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_incident_events_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_incident_messages_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_incidents_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_kai_scheduler_queues_metrics_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_kubernetes_templates_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/{v1_list_product_licenses_response.py → v1_list_license_response.py} +16 -16
- lightning_sdk/lightning_cloud/openapi/models/v1_list_machines_response.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_platform_notifications_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_schedule_runs_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_list_storage_transfers_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_lustre_data_connection.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_machine.py +617 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_machine_direct_v1.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_magic_link_login_request.py +1 -27
- lightning_sdk/lightning_cloud/openapi/models/v1_magic_link_login_response.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_managed_model.py +29 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_market_price.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_membership.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_message_action.py +279 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_metrics_stream.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_model_metrics.py +175 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_namespace_metrics.py +591 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_namespace_user_metrics.py +435 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_nebius_direct_v1.py +29 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_node_metrics.py +361 -23
- lightning_sdk/lightning_cloud/openapi/models/v1_notification_type.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_organization.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_pause_storage_transfer_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_platform_notification.py +279 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_pod_metrics.py +335 -23
- lightning_sdk/lightning_cloud/openapi/models/v1_project.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_project_cluster_binding.py +27 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_project_membership.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_project_settings.py +53 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_project_tab.py +149 -0
- lightning_sdk/lightning_cloud/openapi/models/{v1_product_license_check_response.py → v1_purchase_annual_upsell_request.py} +23 -23
- lightning_sdk/lightning_cloud/openapi/models/v1_purchase_annual_upsell_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_quote_annual_upsell_response.py +227 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_render_kubernetes_template_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_report_deployment_routing_telemetry_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_required_balance_reason.py +107 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_reset_api_key_request.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_reset_api_key_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_resource_visibility.py +1 -27
- lightning_sdk/lightning_cloud/openapi/models/v1_resume_storage_transfer_response.py +97 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_routing_telemetry.py +79 -1
- lightning_sdk/lightning_cloud/openapi/models/v1_rule_resource.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_schedule_run.py +357 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_sdk_command_history_severity.py +104 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_sdk_command_history_type.py +104 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_secret_type.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_server_alert_type.py +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_slack_notifier.py +201 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_slack_notifier_type.py +105 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_storage_asset.py +133 -3
- lightning_sdk/lightning_cloud/openapi/models/v1_storage_transfer.py +435 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_storage_transfer_status.py +108 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_token_login_request.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_token_login_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_token_owner_type.py +104 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_update_cloud_space_instance_config_request.py +3 -81
- lightning_sdk/lightning_cloud/openapi/models/v1_update_project_tab_order_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_update_user_request.py +41 -15
- lightning_sdk/lightning_cloud/openapi/models/v1_user_features.py +444 -600
- lightning_sdk/lightning_cloud/openapi/models/v1_validate_license_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_validate_storage_transfer_response.py +123 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_voltage_park_direct_v1.py +29 -3
- lightning_sdk/lightning_cloud/rest_client.py +48 -45
- lightning_sdk/lightning_cloud/utils/data_connection.py +51 -1
- lightning_sdk/llm/llm.py +175 -56
- lightning_sdk/llm/public_assistants.py +44 -7
- lightning_sdk/machine.py +21 -2
- lightning_sdk/mmt/base.py +7 -0
- lightning_sdk/mmt/mmt.py +11 -3
- lightning_sdk/mmt/v1.py +3 -1
- lightning_sdk/mmt/v2.py +4 -0
- lightning_sdk/owner.py +2 -1
- lightning_sdk/pipeline/steps.py +6 -0
- lightning_sdk/plugin.py +6 -1
- lightning_sdk/studio.py +294 -53
- lightning_sdk/teamspace.py +167 -7
- lightning_sdk/user.py +19 -1
- lightning_sdk/utils/config.py +179 -0
- lightning_sdk/utils/license.py +13 -0
- lightning_sdk/utils/logging.py +79 -0
- lightning_sdk/utils/names.py +1179 -0
- lightning_sdk/utils/progress.py +283 -0
- lightning_sdk/utils/resolve.py +82 -7
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/METADATA +2 -1
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/RECORD +328 -169
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/entry_points.txt +1 -0
- lightning_sdk/lightning_cloud/openapi/models/v1_product_license.py +0 -435
- lightning_sdk/services/license.py +0 -363
- /lightning_sdk/cli/{deploy → legacy}/__init__.py +0 -0
- /lightning_sdk/cli/{ai_hub.py → legacy/ai_hub.py} +0 -0
- /lightning_sdk/cli/{docker_cli.py → legacy/docker_cli.py} +0 -0
- /lightning_sdk/cli/{exceptions.py → legacy/exceptions.py} +0 -0
- /lightning_sdk/cli/{run.py → legacy/run.py} +0 -0
- /lightning_sdk/cli/{studios_menu.py → legacy/studios_menu.py} +0 -0
- /lightning_sdk/cli/{coloring.py → utils/coloring.py} +0 -0
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/LICENSE +0 -0
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/WHEEL +0 -0
- {lightning_sdk-2025.8.6rc2.dist-info → lightning_sdk-2025.11.5.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,122 @@
|
|
|
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.__version__ import __version__
|
|
16
|
+
from lightning_sdk.cli.utils import rich_to_str
|
|
17
|
+
from lightning_sdk.constants import _LIGHTNING_DEBUG
|
|
18
|
+
from lightning_sdk.lightning_cloud.openapi.models.v1_create_sdk_command_history_request import (
|
|
19
|
+
V1CreateSDKCommandHistoryRequest,
|
|
20
|
+
)
|
|
21
|
+
from lightning_sdk.lightning_cloud.openapi.models.v1_sdk_command_history_severity import V1SDKCommandHistorySeverity
|
|
22
|
+
from lightning_sdk.lightning_cloud.openapi.models.v1_sdk_command_history_type import V1SDKCommandHistoryType
|
|
23
|
+
from lightning_sdk.lightning_cloud.rest_client import LightningClient
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def _log_command(message: str = "", duration: int = 0, error: Optional[str] = None) -> None:
|
|
27
|
+
original_command = " ".join(shlex.quote(arg) for arg in sys.argv)
|
|
28
|
+
client = LightningClient(retry=False, max_tries=0)
|
|
29
|
+
|
|
30
|
+
body = V1CreateSDKCommandHistoryRequest(
|
|
31
|
+
command=original_command,
|
|
32
|
+
duration=duration,
|
|
33
|
+
message=f"VERSION: {__version__} | {message}",
|
|
34
|
+
project_id=None,
|
|
35
|
+
severity=V1SDKCommandHistorySeverity.INFO,
|
|
36
|
+
type=V1SDKCommandHistoryType.CLI,
|
|
37
|
+
)
|
|
38
|
+
|
|
39
|
+
if error:
|
|
40
|
+
body.severity = V1SDKCommandHistorySeverity.WARNING if error == "0" else V1SDKCommandHistorySeverity.ERROR
|
|
41
|
+
body.message = body.message + f" | Error: {error}"
|
|
42
|
+
|
|
43
|
+
# limit characters
|
|
44
|
+
body.message = body.message[:1000]
|
|
45
|
+
|
|
46
|
+
with suppress(Exception):
|
|
47
|
+
client.s_dk_command_history_service_create_sdk_command_history(body=body)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def _notify_exception(exception_type: Type[BaseException], value: BaseException, tb: TracebackType) -> None:
|
|
51
|
+
"""CLI won't show tracebacks, just print the exception message."""
|
|
52
|
+
message = str(value.args[0]) if value.args else str(value) or "An unknown error occurred"
|
|
53
|
+
|
|
54
|
+
error_text = Text()
|
|
55
|
+
error_text.append(f"{exception_type.__name__}: ", style="bold red")
|
|
56
|
+
error_text.append(message, style="white")
|
|
57
|
+
|
|
58
|
+
renderables = [error_text]
|
|
59
|
+
|
|
60
|
+
if _LIGHTNING_DEBUG:
|
|
61
|
+
tb_text = "".join(traceback.format_exception(exception_type, value, tb))
|
|
62
|
+
renderables.append(Text("\n\nFull traceback:\n", style="bold yellow"))
|
|
63
|
+
renderables.append(Syntax(tb_text, "python", theme="monokai light", line_numbers=False, word_wrap=True))
|
|
64
|
+
else:
|
|
65
|
+
renderables.append(Text("\n\n🐞 To view the full traceback, set: LIGHTNING_DEBUG=1"))
|
|
66
|
+
|
|
67
|
+
renderables.append(Text("\n📘 Need help? Run: lightning <command> --help", style="cyan"))
|
|
68
|
+
|
|
69
|
+
text = rich_to_str(Panel(Group(*renderables), title="⚡ Lightning CLI Error", border_style="red"))
|
|
70
|
+
click.echo(text, color=True)
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
def logging_excepthook(exception_type: Type[BaseException], value: BaseException, tb: TracebackType) -> None:
|
|
74
|
+
try:
|
|
75
|
+
tb_str = "".join(traceback.format_exception(exception_type, value, tb))
|
|
76
|
+
ctx = click.get_current_context(silent=True)
|
|
77
|
+
command_context = ctx.command_path if ctx else "outside_command_context"
|
|
78
|
+
|
|
79
|
+
message = (
|
|
80
|
+
f"Command: {command_context} | Type: {exception_type.__name__!s} | Value: {value!s} | Traceback: {tb_str}"
|
|
81
|
+
)
|
|
82
|
+
_log_command(message=message)
|
|
83
|
+
finally:
|
|
84
|
+
_notify_exception(exception_type, value, tb)
|
|
85
|
+
|
|
86
|
+
|
|
87
|
+
class CommandLoggingGroup(click.Group):
|
|
88
|
+
def _format_ctx(self, ctx: click.Context) -> str:
|
|
89
|
+
parts = []
|
|
90
|
+
for k, v in ctx.params.items():
|
|
91
|
+
if v is True:
|
|
92
|
+
parts.append(f"--{k}")
|
|
93
|
+
elif v is False or v is None:
|
|
94
|
+
continue
|
|
95
|
+
else:
|
|
96
|
+
parts.append(f"--{k} {v}")
|
|
97
|
+
params = " ".join(parts)
|
|
98
|
+
args = " ".join(ctx.args)
|
|
99
|
+
return (
|
|
100
|
+
f"""Commands: {ctx.command_path} | Subcommand: {ctx.invoked_subcommand} | Params: {params} | Args:{args}"""
|
|
101
|
+
)
|
|
102
|
+
|
|
103
|
+
def invoke(self, ctx: click.Context) -> any:
|
|
104
|
+
"""Overrides the default invoke to wrap command execution with tracking."""
|
|
105
|
+
start_time = time()
|
|
106
|
+
error_message = None
|
|
107
|
+
|
|
108
|
+
try:
|
|
109
|
+
return super().invoke(ctx)
|
|
110
|
+
except click.ClickException as e:
|
|
111
|
+
error_message = str(e)
|
|
112
|
+
e.show()
|
|
113
|
+
ctx.exit(e.exit_code)
|
|
114
|
+
except Exception as e:
|
|
115
|
+
error_message = str(e)
|
|
116
|
+
raise
|
|
117
|
+
finally:
|
|
118
|
+
_log_command(
|
|
119
|
+
message=self._format_ctx(ctx),
|
|
120
|
+
duration=int(time() - start_time),
|
|
121
|
+
error=error_message,
|
|
122
|
+
)
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import os
|
|
2
|
+
from contextlib import suppress
|
|
3
|
+
from typing import Dict, List, Optional, TypedDict
|
|
4
|
+
|
|
5
|
+
import click
|
|
6
|
+
from simple_term_menu import TerminalMenu
|
|
7
|
+
|
|
8
|
+
from lightning_sdk.cli.legacy.exceptions import StudioCliError
|
|
9
|
+
from lightning_sdk.organization import Organization
|
|
10
|
+
from lightning_sdk.owner import Owner
|
|
11
|
+
from lightning_sdk.user import User
|
|
12
|
+
from lightning_sdk.utils.resolve import ApiException, _get_authed_user, _resolve_org, _resolve_user
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
class _OwnerMenuType(TypedDict):
|
|
16
|
+
name: str
|
|
17
|
+
is_org: bool
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
class OwnerMenu:
|
|
21
|
+
"""This class is used to select a teamspace owner (org/user) from a list of possible owners.
|
|
22
|
+
|
|
23
|
+
It can be used to select an owner from a list of possible owners, or to resolve an owner from a name.
|
|
24
|
+
"""
|
|
25
|
+
|
|
26
|
+
def _get_owner_from_interactive_menu(self, possible_owners: Dict[str, _OwnerMenuType]) -> _OwnerMenuType:
|
|
27
|
+
owner_ids = sorted(possible_owners.keys())
|
|
28
|
+
terminal_menu = self._prepare_terminal_menu_owners([possible_owners[k] for k in owner_ids])
|
|
29
|
+
terminal_menu.show()
|
|
30
|
+
|
|
31
|
+
selected_id = owner_ids[terminal_menu.chosen_menu_index]
|
|
32
|
+
return possible_owners[selected_id]
|
|
33
|
+
|
|
34
|
+
def _get_owner_from_name(self, owner: str, possible_owners: Dict[str, _OwnerMenuType]) -> _OwnerMenuType:
|
|
35
|
+
for _, ts in possible_owners.items():
|
|
36
|
+
if ts["name"]:
|
|
37
|
+
return ts
|
|
38
|
+
|
|
39
|
+
click.echo(f"Could not find Owner {owner}, please select it from the list:")
|
|
40
|
+
return self._get_owner_from_interactive_menu(possible_owners)
|
|
41
|
+
|
|
42
|
+
@staticmethod
|
|
43
|
+
def _prepare_terminal_menu_owners(
|
|
44
|
+
possible_owners: List[_OwnerMenuType], title: Optional[str] = None
|
|
45
|
+
) -> TerminalMenu:
|
|
46
|
+
if title is None:
|
|
47
|
+
title = "Please select a Teamspace-Owner out of the following:"
|
|
48
|
+
|
|
49
|
+
return TerminalMenu(
|
|
50
|
+
[f"{to['name']} ({'Organization' if to['is_org'] else 'User'})" for to in possible_owners],
|
|
51
|
+
title=title,
|
|
52
|
+
clear_menu_on_exit=True,
|
|
53
|
+
)
|
|
54
|
+
|
|
55
|
+
@staticmethod
|
|
56
|
+
def _get_possible_owners(user: User) -> Dict[str, _OwnerMenuType]:
|
|
57
|
+
user_api = user._user_api
|
|
58
|
+
|
|
59
|
+
orgs = user_api._get_organizations_for_authed_user()
|
|
60
|
+
owners: Dict[str, _OwnerMenuType] = {user.id: {"name": user.name, "is_org": False}}
|
|
61
|
+
|
|
62
|
+
for org in orgs:
|
|
63
|
+
owners[org.id] = {"name": org.name, "is_org": True}
|
|
64
|
+
|
|
65
|
+
return owners
|
|
66
|
+
|
|
67
|
+
def __call__(self, owner: Optional[str] = None) -> Owner:
|
|
68
|
+
try:
|
|
69
|
+
# try to resolve the teamspace from the name, environment or config
|
|
70
|
+
resolved_owner = None
|
|
71
|
+
with suppress(ApiException, ValueError, RuntimeError):
|
|
72
|
+
resolved_owner = _resolve_org(owner)
|
|
73
|
+
|
|
74
|
+
if resolved_owner is not None:
|
|
75
|
+
return resolved_owner
|
|
76
|
+
|
|
77
|
+
with suppress(ApiException, ValueError, RuntimeError):
|
|
78
|
+
resolved_owner = _resolve_user(owner)
|
|
79
|
+
|
|
80
|
+
if resolved_owner is not None:
|
|
81
|
+
return resolved_owner
|
|
82
|
+
|
|
83
|
+
if os.environ.get("LIGHTNING_NON_INTERACTIVE", "0") == "1":
|
|
84
|
+
raise ValueError(
|
|
85
|
+
"Owner selection is not supported in non-interactive mode. Please provide an owner name."
|
|
86
|
+
)
|
|
87
|
+
|
|
88
|
+
# if the owner is not resolved, try to get the owner from the interactive menu
|
|
89
|
+
# this could mean that either no owner was provided or the provided owner is not valid
|
|
90
|
+
user = _get_authed_user()
|
|
91
|
+
|
|
92
|
+
possible_owners = self._get_possible_owners(user)
|
|
93
|
+
if owner is None:
|
|
94
|
+
owner_dict = self._get_owner_from_interactive_menu(possible_owners=possible_owners)
|
|
95
|
+
|
|
96
|
+
else:
|
|
97
|
+
owner_dict = self._get_owner_from_name(owner=owner, possible_owners=possible_owners)
|
|
98
|
+
|
|
99
|
+
if owner_dict.get("is_org", False):
|
|
100
|
+
return Organization(owner_dict.get("name", None))
|
|
101
|
+
|
|
102
|
+
return User(owner_dict.get("name", None))
|
|
103
|
+
except KeyboardInterrupt:
|
|
104
|
+
raise KeyboardInterrupt from None
|
|
105
|
+
|
|
106
|
+
except Exception as e:
|
|
107
|
+
raise StudioCliError(
|
|
108
|
+
f"Could not find the given Teamspace-Owner {owner}. "
|
|
109
|
+
"Please contact Lightning AI directly to resolve this issue."
|
|
110
|
+
) from e
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
from typing import Optional
|
|
2
|
+
|
|
3
|
+
from lightning_sdk.teamspace import Teamspace
|
|
4
|
+
from lightning_sdk.utils.resolve import _resolve_teamspace
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
def resolve_teamspace_owner_name_format(teamspace_name: Optional[str]) -> Optional[Teamspace]:
|
|
8
|
+
teamspace_resolved = None
|
|
9
|
+
if teamspace_name is None:
|
|
10
|
+
return _resolve_teamspace(None, None, None)
|
|
11
|
+
|
|
12
|
+
splits = teamspace_name.split("/")
|
|
13
|
+
if len(splits) == 1:
|
|
14
|
+
try:
|
|
15
|
+
teamspace_resolved = _resolve_teamspace(teamspace_name, None, None)
|
|
16
|
+
except Exception:
|
|
17
|
+
teamspace_resolved = None
|
|
18
|
+
|
|
19
|
+
elif len(splits) == 2:
|
|
20
|
+
try:
|
|
21
|
+
try:
|
|
22
|
+
teamspace_resolved = _resolve_teamspace(splits[1], splits[0], None)
|
|
23
|
+
except Exception:
|
|
24
|
+
teamspace_resolved = _resolve_teamspace(splits[1], None, splits[0])
|
|
25
|
+
except Exception:
|
|
26
|
+
teamspace_resolved = None
|
|
27
|
+
|
|
28
|
+
return teamspace_resolved
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import os
|
|
2
|
+
from typing import Any
|
|
3
|
+
|
|
4
|
+
from rich.console import Console
|
|
5
|
+
|
|
6
|
+
from lightning_sdk.studio import Studio
|
|
7
|
+
from lightning_sdk.utils.resolve import _get_studio_url
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
def rich_to_str(*renderables: Any) -> str:
|
|
11
|
+
with open(os.devnull, "w") as f:
|
|
12
|
+
console = Console(file=f, record=True)
|
|
13
|
+
console.print(*renderables)
|
|
14
|
+
return console.export_text(styles=True)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
# not supported on all terminals (e.g. older ones). in that case it's a name without a link
|
|
18
|
+
# see https://gist.github.com/egmontkob/eb114294efbcd5adb1944c9f3cb5feda#hyperlinks-aka-html-like-anchors-in-terminal-emulators
|
|
19
|
+
# for details and
|
|
20
|
+
# https://github.com/Alhadis/OSC8-Adoption/blob/main/README.md for status of OSC8 adoption
|
|
21
|
+
def studio_name_link(studio: Studio, to_ascii: bool = True) -> str:
|
|
22
|
+
"""Hyperlink a studio name.
|
|
23
|
+
|
|
24
|
+
Args:
|
|
25
|
+
studio: the studio whose name to print and link to the studio url
|
|
26
|
+
to_ascii: whether return a plain ascii string with characters for linking converted to ascii as well.
|
|
27
|
+
if False, returns the rich markup directly.
|
|
28
|
+
"""
|
|
29
|
+
url = _get_studio_url(studio)
|
|
30
|
+
|
|
31
|
+
studio_link_markup = f"[link={url}]{studio.name}[/link]"
|
|
32
|
+
if not to_ascii:
|
|
33
|
+
return studio_link_markup
|
|
34
|
+
|
|
35
|
+
return rich_to_str(studio_link_markup).strip("\n")
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
from lightning_sdk.organization import Organization
|
|
2
|
+
from lightning_sdk.studio import Studio
|
|
3
|
+
from lightning_sdk.teamspace import Teamspace
|
|
4
|
+
from lightning_sdk.utils.config import Config, DefaultConfigKeys
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
def save_teamspace_to_config(teamspace: Teamspace, overwrite: bool = False) -> None:
|
|
8
|
+
saved = _save_to_config_if_not_exists(DefaultConfigKeys.teamspace_name, teamspace.name, overwrite)
|
|
9
|
+
saved = _save_to_config_if_not_exists(DefaultConfigKeys.teamspace_owner, teamspace.owner.name, overwrite=saved)
|
|
10
|
+
saved = _save_to_config_if_not_exists(
|
|
11
|
+
DefaultConfigKeys.teamspace_owner_type,
|
|
12
|
+
"organization" if isinstance(teamspace.owner, Organization) else "user",
|
|
13
|
+
overwrite=saved,
|
|
14
|
+
)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
def save_studio_to_config(studio: Studio, overwrite: bool = False) -> None:
|
|
18
|
+
saved = _save_to_config_if_not_exists(DefaultConfigKeys.studio, studio.name, overwrite)
|
|
19
|
+
save_teamspace_to_config(studio.teamspace, overwrite=saved)
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def _save_to_config_if_not_exists(key: str, value: str, overwrite: bool = False) -> bool:
|
|
23
|
+
cfg = Config()
|
|
24
|
+
if overwrite or cfg.get(key) is None:
|
|
25
|
+
cfg.set(key, value)
|
|
26
|
+
return True
|
|
27
|
+
return False
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import os
|
|
2
|
+
import platform
|
|
3
|
+
import uuid
|
|
4
|
+
from pathlib import Path
|
|
5
|
+
from typing import Optional
|
|
6
|
+
|
|
7
|
+
from lightning_sdk.lightning_cloud.login import Auth
|
|
8
|
+
from lightning_sdk.utils.config import _DEFAULT_CONFIG_FILE_PATH
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
def configure_ssh_internal(force_download: bool = False) -> str:
|
|
12
|
+
"""Internal function to configure SSH without Click decorators."""
|
|
13
|
+
auth = Auth()
|
|
14
|
+
auth.authenticate()
|
|
15
|
+
return download_ssh_keys(auth.api_key, force_download=force_download)
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
def download_ssh_keys(
|
|
19
|
+
api_key: Optional[str],
|
|
20
|
+
force_download: bool = False,
|
|
21
|
+
ssh_key_name: str = "lightning_rsa",
|
|
22
|
+
) -> str:
|
|
23
|
+
"""Download the SSH key for a User."""
|
|
24
|
+
ssh_private_key_path = os.path.join(os.path.expanduser(os.path.dirname(_DEFAULT_CONFIG_FILE_PATH)), ssh_key_name)
|
|
25
|
+
|
|
26
|
+
os.makedirs(os.path.dirname(ssh_private_key_path), exist_ok=True)
|
|
27
|
+
|
|
28
|
+
if not os.path.isfile(ssh_private_key_path) or force_download:
|
|
29
|
+
key_id = str(uuid.uuid4())
|
|
30
|
+
download_file(
|
|
31
|
+
f"https://lightning.ai/setup/ssh-gen?t={api_key}&id={key_id}&machineName={platform.node()}",
|
|
32
|
+
Path(ssh_private_key_path),
|
|
33
|
+
overwrite=True,
|
|
34
|
+
chmod=0o600,
|
|
35
|
+
)
|
|
36
|
+
download_file(
|
|
37
|
+
f"https://lightning.ai/setup/ssh-public?t={api_key}&id={key_id}",
|
|
38
|
+
Path(ssh_private_key_path + ".pub"),
|
|
39
|
+
overwrite=True,
|
|
40
|
+
)
|
|
41
|
+
|
|
42
|
+
return ssh_private_key_path
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def download_file(url: str, local_path: Path, overwrite: bool = True, chmod: Optional[int] = None) -> None:
|
|
46
|
+
"""Download a file from a URL."""
|
|
47
|
+
import requests
|
|
48
|
+
|
|
49
|
+
if os.path.isfile(local_path) and not overwrite:
|
|
50
|
+
raise FileExistsError(f"The file {local_path} already exists and overwrite is set to False.")
|
|
51
|
+
|
|
52
|
+
response = requests.get(url, stream=True)
|
|
53
|
+
response.raise_for_status()
|
|
54
|
+
|
|
55
|
+
with open(local_path, "wb") as file:
|
|
56
|
+
for chunk in response.iter_content(chunk_size=8192):
|
|
57
|
+
file.write(chunk)
|
|
58
|
+
if chmod is not None:
|
|
59
|
+
os.chmod(local_path, 0o600)
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import os
|
|
2
|
+
from contextlib import suppress
|
|
3
|
+
from typing import Dict, List, Optional, Union
|
|
4
|
+
|
|
5
|
+
import click
|
|
6
|
+
from simple_term_menu import TerminalMenu
|
|
7
|
+
|
|
8
|
+
from lightning_sdk.cli.legacy.exceptions import StudioCliError
|
|
9
|
+
from lightning_sdk.studio import VM, Studio
|
|
10
|
+
from lightning_sdk.teamspace import Teamspace
|
|
11
|
+
from lightning_sdk.utils.resolve import _get_authed_user
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class StudiosMenu:
|
|
15
|
+
"""This class is used to select a studio from a list of possible studios within a teamspace.
|
|
16
|
+
|
|
17
|
+
It can be used to select a studio from a list of possible studios, or to resolve a studio from a name.
|
|
18
|
+
"""
|
|
19
|
+
|
|
20
|
+
def __init__(self, teamspace: Teamspace, vm: bool = False) -> None:
|
|
21
|
+
"""Initialize the StudiosMenu with a teamspace.
|
|
22
|
+
|
|
23
|
+
Args:
|
|
24
|
+
teamspace: The teamspace to list studios from
|
|
25
|
+
"""
|
|
26
|
+
self.teamspace = teamspace
|
|
27
|
+
self.vm = vm
|
|
28
|
+
|
|
29
|
+
def _get_studio_from_interactive_menu(self, possible_studios: Dict[str, Union[Studio, VM]]) -> Union[Studio, VM]:
|
|
30
|
+
studio_names = sorted(possible_studios.keys())
|
|
31
|
+
terminal_menu = self._prepare_terminal_menu_studios(studio_names, vm=self.vm)
|
|
32
|
+
terminal_menu.show()
|
|
33
|
+
|
|
34
|
+
selected_name = studio_names[terminal_menu.chosen_menu_index]
|
|
35
|
+
return possible_studios[selected_name]
|
|
36
|
+
|
|
37
|
+
def _get_studio_from_name(self, studio: str, possible_studios: Dict[str, Union[Studio, VM]]) -> Union[Studio, VM]:
|
|
38
|
+
if studio in possible_studios:
|
|
39
|
+
return possible_studios[studio]
|
|
40
|
+
|
|
41
|
+
click.echo(f"Could not find {'VM' if self.vm else 'Studio'} {studio}, please select it from the list:")
|
|
42
|
+
return self._get_studio_from_interactive_menu(possible_studios)
|
|
43
|
+
|
|
44
|
+
@staticmethod
|
|
45
|
+
def _prepare_terminal_menu_studios(
|
|
46
|
+
studio_names: List[str], title: Optional[str] = None, vm: bool = False
|
|
47
|
+
) -> TerminalMenu:
|
|
48
|
+
if title is None:
|
|
49
|
+
title = f"Please select a {'VM' if vm else 'Studio'} out of the following:"
|
|
50
|
+
|
|
51
|
+
return TerminalMenu(studio_names, title=title, clear_menu_on_exit=True)
|
|
52
|
+
|
|
53
|
+
def _get_possible_studios(self) -> Dict[str, Union[Studio, VM]]:
|
|
54
|
+
"""Get all available studios in the teamspace."""
|
|
55
|
+
studios: Dict[str, Union[Studio, VM]] = {}
|
|
56
|
+
|
|
57
|
+
user = _get_authed_user()
|
|
58
|
+
teamspace_studios = self.teamspace.vms if self.vm else self.teamspace.studios
|
|
59
|
+
for studio in teamspace_studios:
|
|
60
|
+
if studio._studio.user_id == user.id:
|
|
61
|
+
studios[studio.name] = studio
|
|
62
|
+
return studios
|
|
63
|
+
|
|
64
|
+
def __call__(self, studio: Optional[str] = None) -> Union[Studio, VM]:
|
|
65
|
+
"""Select a studio from the teamspace.
|
|
66
|
+
|
|
67
|
+
Args:
|
|
68
|
+
studio: Optional studio name to select. If not provided, will show interactive menu.
|
|
69
|
+
|
|
70
|
+
Returns:
|
|
71
|
+
Selected Studio/VM object
|
|
72
|
+
|
|
73
|
+
Raises:
|
|
74
|
+
StudioCliError: If studio selection fails
|
|
75
|
+
"""
|
|
76
|
+
try:
|
|
77
|
+
# try to resolve the studio from the name, environment or config
|
|
78
|
+
resolved_studio = None
|
|
79
|
+
|
|
80
|
+
selected_cls = VM if self.vm else Studio
|
|
81
|
+
|
|
82
|
+
with suppress(Exception):
|
|
83
|
+
resolved_studio = selected_cls(name=studio, teamspace=self.teamspace, create_ok=False)
|
|
84
|
+
|
|
85
|
+
if resolved_studio is not None:
|
|
86
|
+
return resolved_studio
|
|
87
|
+
|
|
88
|
+
if os.environ.get("LIGHTNING_NON_INTERACTIVE", "0") == "1" and studio is None:
|
|
89
|
+
raise ValueError(
|
|
90
|
+
f"{'VM' if self.vm else 'Studio'} selection is not supported in non-interactive mode. "
|
|
91
|
+
"Please provide a studio name."
|
|
92
|
+
)
|
|
93
|
+
|
|
94
|
+
click.echo(f"Listing studios in teamspace {self.teamspace.owner.name}/{self.teamspace.name}...")
|
|
95
|
+
|
|
96
|
+
possible_studios = self._get_possible_studios()
|
|
97
|
+
|
|
98
|
+
if not possible_studios:
|
|
99
|
+
raise ValueError(f"No studios found in teamspace {self.teamspace.name}")
|
|
100
|
+
|
|
101
|
+
if studio is None:
|
|
102
|
+
return self._get_studio_from_interactive_menu(possible_studios)
|
|
103
|
+
|
|
104
|
+
return self._get_studio_from_name(studio, possible_studios)
|
|
105
|
+
|
|
106
|
+
except KeyboardInterrupt:
|
|
107
|
+
raise KeyboardInterrupt from None
|
|
108
|
+
|
|
109
|
+
except Exception as e:
|
|
110
|
+
raise StudioCliError(
|
|
111
|
+
f"Could not resolve a {'VM' if self.vm else 'Studio'}. "
|
|
112
|
+
"Please pass it as an argument or contact Lightning AI directly to resolve this issue."
|
|
113
|
+
) from e
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import os
|
|
2
|
+
from contextlib import suppress
|
|
3
|
+
from typing import Dict, List, Optional
|
|
4
|
+
|
|
5
|
+
import click
|
|
6
|
+
from simple_term_menu import TerminalMenu
|
|
7
|
+
|
|
8
|
+
from lightning_sdk.cli.legacy.exceptions import StudioCliError
|
|
9
|
+
from lightning_sdk.cli.utils.resolve import resolve_teamspace_owner_name_format
|
|
10
|
+
from lightning_sdk.teamspace import Organization, Teamspace
|
|
11
|
+
from lightning_sdk.user import Owner, User
|
|
12
|
+
from lightning_sdk.utils.resolve import ApiException, _get_authed_user, _resolve_teamspace
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
class TeamspacesMenu:
|
|
16
|
+
"""This class is used to select a teamspace from a list of possible teamspaces.
|
|
17
|
+
|
|
18
|
+
It can be used to select a teamspace from a list of possible teamspaces, or to resolve a teamspace from a name.
|
|
19
|
+
"""
|
|
20
|
+
|
|
21
|
+
def __init__(self, owner: Optional[Owner] = None) -> None:
|
|
22
|
+
self._owner: Owner = owner
|
|
23
|
+
|
|
24
|
+
def _get_teamspace_from_interactive_menu(self, possible_teamspaces: Dict[str, str]) -> str:
|
|
25
|
+
teamspace_ids = sorted(possible_teamspaces.keys())
|
|
26
|
+
terminal_menu = self._prepare_terminal_menu_teamspaces(
|
|
27
|
+
[possible_teamspaces[k] for k in teamspace_ids], self._owner
|
|
28
|
+
)
|
|
29
|
+
terminal_menu.show()
|
|
30
|
+
|
|
31
|
+
selected_id = teamspace_ids[terminal_menu.chosen_menu_index]
|
|
32
|
+
return possible_teamspaces[selected_id]
|
|
33
|
+
|
|
34
|
+
def _get_teamspace_from_name(self, teamspace: str, possible_teamspaces: Dict[str, str]) -> str:
|
|
35
|
+
for _, ts in possible_teamspaces.items():
|
|
36
|
+
if ts == teamspace:
|
|
37
|
+
return ts
|
|
38
|
+
|
|
39
|
+
click.echo(f"Could not find Teamspace {self._owner.name}/{teamspace}, please select it from the list:")
|
|
40
|
+
return self._get_teamspace_from_interactive_menu(possible_teamspaces)
|
|
41
|
+
|
|
42
|
+
@staticmethod
|
|
43
|
+
def _prepare_terminal_menu_teamspaces(
|
|
44
|
+
possible_teamspaces: List[str],
|
|
45
|
+
owner: Owner,
|
|
46
|
+
title: Optional[str] = None,
|
|
47
|
+
) -> TerminalMenu:
|
|
48
|
+
if title is None:
|
|
49
|
+
title = f"Please select a Teamspace (owned by {owner.name}) out of the following:"
|
|
50
|
+
|
|
51
|
+
return TerminalMenu(possible_teamspaces, title=title, clear_menu_on_exit=True)
|
|
52
|
+
|
|
53
|
+
def _get_possible_teamspaces(self, user: User) -> Dict[str, str]:
|
|
54
|
+
user_api = user._user_api
|
|
55
|
+
|
|
56
|
+
memberships = user_api._get_all_teamspace_memberships(
|
|
57
|
+
user_id=user.id, org_id=self._owner.id if isinstance(self._owner, Organization) else None
|
|
58
|
+
)
|
|
59
|
+
|
|
60
|
+
teamspaces = {}
|
|
61
|
+
# get all teamspace memberships
|
|
62
|
+
for membership in memberships:
|
|
63
|
+
teamspace_id = membership.project_id
|
|
64
|
+
teamspace_name = membership.name
|
|
65
|
+
|
|
66
|
+
if membership.owner_id == self._owner.id:
|
|
67
|
+
teamspaces[teamspace_id] = teamspace_name
|
|
68
|
+
|
|
69
|
+
return teamspaces
|
|
70
|
+
|
|
71
|
+
def __call__(self, teamspace: Optional[str] = None) -> Teamspace:
|
|
72
|
+
resolved_teamspace = None
|
|
73
|
+
resolved_teamspace = resolve_teamspace_owner_name_format(teamspace)
|
|
74
|
+
if resolved_teamspace is not None:
|
|
75
|
+
return resolved_teamspace
|
|
76
|
+
|
|
77
|
+
if self._owner is None:
|
|
78
|
+
# resolve owner if required
|
|
79
|
+
from lightning_sdk.cli.utils.owner_selection import OwnerMenu
|
|
80
|
+
|
|
81
|
+
menu = OwnerMenu()
|
|
82
|
+
self._owner = menu()
|
|
83
|
+
|
|
84
|
+
if isinstance(self._owner, Organization):
|
|
85
|
+
org = self._owner
|
|
86
|
+
user = None
|
|
87
|
+
elif isinstance(self._owner, User):
|
|
88
|
+
org = None
|
|
89
|
+
user = self._owner
|
|
90
|
+
|
|
91
|
+
with suppress(ApiException, ValueError, RuntimeError):
|
|
92
|
+
resolved_teamspace = _resolve_teamspace(teamspace=teamspace, org=org, user=user)
|
|
93
|
+
|
|
94
|
+
if resolved_teamspace is not None:
|
|
95
|
+
return resolved_teamspace
|
|
96
|
+
|
|
97
|
+
if os.environ.get("LIGHTNING_NON_INTERACTIVE", "0") == "1":
|
|
98
|
+
raise ValueError(
|
|
99
|
+
"Teamspace selection is not supported in non-interactive mode. "
|
|
100
|
+
"Please provide a teamspace name in the format of 'owner/teamspace'."
|
|
101
|
+
)
|
|
102
|
+
|
|
103
|
+
# if the teamspace is not resolved, try to get the teamspace from the interactive menu
|
|
104
|
+
# this could mean that either no teamspace was provided or the provided teamspace is not valid
|
|
105
|
+
try:
|
|
106
|
+
auth_user = _get_authed_user()
|
|
107
|
+
|
|
108
|
+
possible_teamspaces = self._get_possible_teamspaces(auth_user)
|
|
109
|
+
if teamspace is None:
|
|
110
|
+
teamspace_name = self._get_teamspace_from_interactive_menu(possible_teamspaces=possible_teamspaces)
|
|
111
|
+
else:
|
|
112
|
+
teamspace_name = self._get_teamspace_from_name(
|
|
113
|
+
teamspace=teamspace, possible_teamspaces=possible_teamspaces
|
|
114
|
+
)
|
|
115
|
+
|
|
116
|
+
return Teamspace(teamspace_name, org=org, user=user)
|
|
117
|
+
|
|
118
|
+
except KeyboardInterrupt:
|
|
119
|
+
raise KeyboardInterrupt from None
|
|
120
|
+
|
|
121
|
+
except Exception as e:
|
|
122
|
+
raise StudioCliError(
|
|
123
|
+
f"Could not find the given Teamspace owned by {self._owner.name}. "
|
|
124
|
+
"Please contact Lightning AI directly to resolve this issue."
|
|
125
|
+
) from e
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import click
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
def register_commands(group: click.Group) -> None:
|
|
5
|
+
"""Register studio commands with the given group."""
|
|
6
|
+
from lightning_sdk.cli.vm.create import create_vm
|
|
7
|
+
from lightning_sdk.cli.vm.delete import delete_vm
|
|
8
|
+
from lightning_sdk.cli.vm.list import list_vms
|
|
9
|
+
from lightning_sdk.cli.vm.ssh import ssh_vm
|
|
10
|
+
from lightning_sdk.cli.vm.start import start_vm
|
|
11
|
+
from lightning_sdk.cli.vm.stop import stop_vm
|
|
12
|
+
from lightning_sdk.cli.vm.switch import switch_vm
|
|
13
|
+
|
|
14
|
+
group.add_command(create_vm)
|
|
15
|
+
group.add_command(delete_vm)
|
|
16
|
+
group.add_command(list_vms)
|
|
17
|
+
group.add_command(ssh_vm)
|
|
18
|
+
group.add_command(start_vm)
|
|
19
|
+
group.add_command(stop_vm)
|
|
20
|
+
group.add_command(switch_vm)
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
from typing import Optional
|
|
2
|
+
|
|
3
|
+
import click
|
|
4
|
+
|
|
5
|
+
from lightning_sdk.cli.studio.create import create_impl
|
|
6
|
+
from lightning_sdk.machine import CloudProvider
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
@click.command("create")
|
|
10
|
+
@click.option("--name", help="The name of the VM to create. If not provided, a random name will be generated.")
|
|
11
|
+
@click.option("--teamspace", help="Override default teamspace (format: owner/teamspace)")
|
|
12
|
+
@click.option(
|
|
13
|
+
"--cloud-provider",
|
|
14
|
+
help="The cloud provider to start the VM on. Defaults to teamspace default.",
|
|
15
|
+
type=click.Choice(m.name for m in list(CloudProvider)),
|
|
16
|
+
)
|
|
17
|
+
@click.option(
|
|
18
|
+
"--cloud-account",
|
|
19
|
+
help="The cloud account to create the VM on. Defaults to teamspace default.",
|
|
20
|
+
type=click.STRING,
|
|
21
|
+
)
|
|
22
|
+
def create_vm(
|
|
23
|
+
name: Optional[str] = None,
|
|
24
|
+
teamspace: Optional[str] = None,
|
|
25
|
+
cloud_provider: Optional[str] = None,
|
|
26
|
+
cloud_account: Optional[str] = None,
|
|
27
|
+
) -> None:
|
|
28
|
+
"""Create a new VM.
|
|
29
|
+
|
|
30
|
+
Example:
|
|
31
|
+
lightning vm create
|
|
32
|
+
"""
|
|
33
|
+
create_impl(name=name, teamspace=teamspace, cloud_provider=cloud_provider, cloud_account=cloud_account, vm=True)
|