supervisely 6.73.438__py3-none-any.whl → 6.73.513__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 (203) hide show
  1. supervisely/__init__.py +137 -1
  2. supervisely/_utils.py +81 -0
  3. supervisely/annotation/annotation.py +8 -2
  4. supervisely/annotation/json_geometries_map.py +14 -11
  5. supervisely/annotation/label.py +80 -3
  6. supervisely/api/annotation_api.py +14 -11
  7. supervisely/api/api.py +59 -38
  8. supervisely/api/app_api.py +11 -2
  9. supervisely/api/dataset_api.py +74 -12
  10. supervisely/api/entities_collection_api.py +10 -0
  11. supervisely/api/entity_annotation/figure_api.py +52 -4
  12. supervisely/api/entity_annotation/object_api.py +3 -3
  13. supervisely/api/entity_annotation/tag_api.py +63 -12
  14. supervisely/api/guides_api.py +210 -0
  15. supervisely/api/image_api.py +72 -1
  16. supervisely/api/labeling_job_api.py +83 -1
  17. supervisely/api/labeling_queue_api.py +33 -7
  18. supervisely/api/module_api.py +9 -0
  19. supervisely/api/project_api.py +71 -26
  20. supervisely/api/storage_api.py +3 -1
  21. supervisely/api/task_api.py +13 -2
  22. supervisely/api/team_api.py +4 -3
  23. supervisely/api/video/video_annotation_api.py +119 -3
  24. supervisely/api/video/video_api.py +65 -14
  25. supervisely/api/video/video_figure_api.py +24 -11
  26. supervisely/app/__init__.py +1 -1
  27. supervisely/app/content.py +23 -7
  28. supervisely/app/development/development.py +18 -2
  29. supervisely/app/fastapi/__init__.py +1 -0
  30. supervisely/app/fastapi/custom_static_files.py +1 -1
  31. supervisely/app/fastapi/multi_user.py +105 -0
  32. supervisely/app/fastapi/subapp.py +88 -42
  33. supervisely/app/fastapi/websocket.py +77 -9
  34. supervisely/app/singleton.py +21 -0
  35. supervisely/app/v1/app_service.py +18 -2
  36. supervisely/app/v1/constants.py +7 -1
  37. supervisely/app/widgets/__init__.py +6 -0
  38. supervisely/app/widgets/activity_feed/__init__.py +0 -0
  39. supervisely/app/widgets/activity_feed/activity_feed.py +239 -0
  40. supervisely/app/widgets/activity_feed/style.css +78 -0
  41. supervisely/app/widgets/activity_feed/template.html +22 -0
  42. supervisely/app/widgets/card/card.py +20 -0
  43. supervisely/app/widgets/classes_list_selector/classes_list_selector.py +121 -9
  44. supervisely/app/widgets/classes_list_selector/template.html +60 -93
  45. supervisely/app/widgets/classes_mapping/classes_mapping.py +13 -12
  46. supervisely/app/widgets/classes_table/classes_table.py +1 -0
  47. supervisely/app/widgets/deploy_model/deploy_model.py +56 -35
  48. supervisely/app/widgets/dialog/dialog.py +12 -0
  49. supervisely/app/widgets/dialog/template.html +2 -1
  50. supervisely/app/widgets/ecosystem_model_selector/ecosystem_model_selector.py +1 -1
  51. supervisely/app/widgets/experiment_selector/experiment_selector.py +8 -0
  52. supervisely/app/widgets/fast_table/fast_table.py +184 -60
  53. supervisely/app/widgets/fast_table/template.html +1 -1
  54. supervisely/app/widgets/heatmap/__init__.py +0 -0
  55. supervisely/app/widgets/heatmap/heatmap.py +564 -0
  56. supervisely/app/widgets/heatmap/script.js +533 -0
  57. supervisely/app/widgets/heatmap/style.css +233 -0
  58. supervisely/app/widgets/heatmap/template.html +21 -0
  59. supervisely/app/widgets/modal/__init__.py +0 -0
  60. supervisely/app/widgets/modal/modal.py +198 -0
  61. supervisely/app/widgets/modal/template.html +10 -0
  62. supervisely/app/widgets/object_class_view/object_class_view.py +3 -0
  63. supervisely/app/widgets/radio_tabs/radio_tabs.py +18 -2
  64. supervisely/app/widgets/radio_tabs/template.html +1 -0
  65. supervisely/app/widgets/select/select.py +6 -3
  66. supervisely/app/widgets/select_class/__init__.py +0 -0
  67. supervisely/app/widgets/select_class/select_class.py +363 -0
  68. supervisely/app/widgets/select_class/template.html +50 -0
  69. supervisely/app/widgets/select_cuda/select_cuda.py +22 -0
  70. supervisely/app/widgets/select_dataset_tree/select_dataset_tree.py +65 -7
  71. supervisely/app/widgets/select_tag/__init__.py +0 -0
  72. supervisely/app/widgets/select_tag/select_tag.py +352 -0
  73. supervisely/app/widgets/select_tag/template.html +64 -0
  74. supervisely/app/widgets/select_team/select_team.py +37 -4
  75. supervisely/app/widgets/select_team/template.html +4 -5
  76. supervisely/app/widgets/select_user/__init__.py +0 -0
  77. supervisely/app/widgets/select_user/select_user.py +270 -0
  78. supervisely/app/widgets/select_user/template.html +13 -0
  79. supervisely/app/widgets/select_workspace/select_workspace.py +59 -10
  80. supervisely/app/widgets/select_workspace/template.html +9 -12
  81. supervisely/app/widgets/table/table.py +68 -13
  82. supervisely/app/widgets/tree_select/tree_select.py +2 -0
  83. supervisely/aug/aug.py +6 -2
  84. supervisely/convert/base_converter.py +1 -0
  85. supervisely/convert/converter.py +2 -2
  86. supervisely/convert/image/csv/csv_converter.py +24 -15
  87. supervisely/convert/image/image_converter.py +3 -1
  88. supervisely/convert/image/image_helper.py +48 -4
  89. supervisely/convert/image/label_studio/label_studio_converter.py +2 -0
  90. supervisely/convert/image/medical2d/medical2d_helper.py +2 -24
  91. supervisely/convert/image/multispectral/multispectral_converter.py +6 -0
  92. supervisely/convert/image/pascal_voc/pascal_voc_converter.py +8 -5
  93. supervisely/convert/image/pascal_voc/pascal_voc_helper.py +7 -0
  94. supervisely/convert/pointcloud/kitti_3d/kitti_3d_converter.py +33 -3
  95. supervisely/convert/pointcloud/kitti_3d/kitti_3d_helper.py +12 -5
  96. supervisely/convert/pointcloud/las/las_converter.py +13 -1
  97. supervisely/convert/pointcloud/las/las_helper.py +110 -11
  98. supervisely/convert/pointcloud/nuscenes_conv/nuscenes_converter.py +27 -16
  99. supervisely/convert/pointcloud/pointcloud_converter.py +91 -3
  100. supervisely/convert/pointcloud_episodes/nuscenes_conv/nuscenes_converter.py +58 -22
  101. supervisely/convert/pointcloud_episodes/nuscenes_conv/nuscenes_helper.py +21 -47
  102. supervisely/convert/video/__init__.py +1 -0
  103. supervisely/convert/video/multi_view/__init__.py +0 -0
  104. supervisely/convert/video/multi_view/multi_view.py +543 -0
  105. supervisely/convert/video/sly/sly_video_converter.py +359 -3
  106. supervisely/convert/video/video_converter.py +24 -4
  107. supervisely/convert/volume/dicom/dicom_converter.py +13 -5
  108. supervisely/convert/volume/dicom/dicom_helper.py +30 -18
  109. supervisely/geometry/constants.py +1 -0
  110. supervisely/geometry/geometry.py +4 -0
  111. supervisely/geometry/helpers.py +5 -1
  112. supervisely/geometry/oriented_bbox.py +676 -0
  113. supervisely/geometry/polyline_3d.py +110 -0
  114. supervisely/geometry/rectangle.py +2 -1
  115. supervisely/io/env.py +76 -1
  116. supervisely/io/fs.py +21 -0
  117. supervisely/nn/benchmark/base_evaluator.py +104 -11
  118. supervisely/nn/benchmark/instance_segmentation/evaluator.py +1 -8
  119. supervisely/nn/benchmark/object_detection/evaluator.py +20 -4
  120. supervisely/nn/benchmark/object_detection/vis_metrics/pr_curve.py +10 -5
  121. supervisely/nn/benchmark/semantic_segmentation/evaluator.py +34 -16
  122. supervisely/nn/benchmark/semantic_segmentation/vis_metrics/confusion_matrix.py +1 -1
  123. supervisely/nn/benchmark/semantic_segmentation/vis_metrics/frequently_confused.py +1 -1
  124. supervisely/nn/benchmark/semantic_segmentation/vis_metrics/overview.py +1 -1
  125. supervisely/nn/benchmark/visualization/evaluation_result.py +66 -4
  126. supervisely/nn/inference/cache.py +43 -18
  127. supervisely/nn/inference/gui/serving_gui_template.py +5 -2
  128. supervisely/nn/inference/inference.py +916 -222
  129. supervisely/nn/inference/inference_request.py +55 -10
  130. supervisely/nn/inference/predict_app/gui/classes_selector.py +83 -12
  131. supervisely/nn/inference/predict_app/gui/gui.py +676 -488
  132. supervisely/nn/inference/predict_app/gui/input_selector.py +205 -26
  133. supervisely/nn/inference/predict_app/gui/model_selector.py +2 -4
  134. supervisely/nn/inference/predict_app/gui/output_selector.py +46 -6
  135. supervisely/nn/inference/predict_app/gui/settings_selector.py +756 -59
  136. supervisely/nn/inference/predict_app/gui/tags_selector.py +1 -1
  137. supervisely/nn/inference/predict_app/gui/utils.py +236 -119
  138. supervisely/nn/inference/predict_app/predict_app.py +2 -2
  139. supervisely/nn/inference/session.py +43 -35
  140. supervisely/nn/inference/tracking/bbox_tracking.py +118 -35
  141. supervisely/nn/inference/tracking/point_tracking.py +5 -1
  142. supervisely/nn/inference/tracking/tracker_interface.py +10 -1
  143. supervisely/nn/inference/uploader.py +139 -12
  144. supervisely/nn/live_training/__init__.py +7 -0
  145. supervisely/nn/live_training/api_server.py +111 -0
  146. supervisely/nn/live_training/artifacts_utils.py +243 -0
  147. supervisely/nn/live_training/checkpoint_utils.py +229 -0
  148. supervisely/nn/live_training/dynamic_sampler.py +44 -0
  149. supervisely/nn/live_training/helpers.py +14 -0
  150. supervisely/nn/live_training/incremental_dataset.py +146 -0
  151. supervisely/nn/live_training/live_training.py +497 -0
  152. supervisely/nn/live_training/loss_plateau_detector.py +111 -0
  153. supervisely/nn/live_training/request_queue.py +52 -0
  154. supervisely/nn/model/model_api.py +9 -0
  155. supervisely/nn/model/prediction.py +2 -1
  156. supervisely/nn/model/prediction_session.py +26 -14
  157. supervisely/nn/prediction_dto.py +19 -1
  158. supervisely/nn/tracker/base_tracker.py +11 -1
  159. supervisely/nn/tracker/botsort/botsort_config.yaml +0 -1
  160. supervisely/nn/tracker/botsort/tracker/mc_bot_sort.py +7 -4
  161. supervisely/nn/tracker/botsort_tracker.py +94 -65
  162. supervisely/nn/tracker/utils.py +4 -5
  163. supervisely/nn/tracker/visualize.py +93 -93
  164. supervisely/nn/training/gui/classes_selector.py +16 -1
  165. supervisely/nn/training/gui/train_val_splits_selector.py +52 -31
  166. supervisely/nn/training/train_app.py +46 -31
  167. supervisely/project/data_version.py +115 -51
  168. supervisely/project/download.py +1 -1
  169. supervisely/project/pointcloud_episode_project.py +37 -8
  170. supervisely/project/pointcloud_project.py +30 -2
  171. supervisely/project/project.py +14 -2
  172. supervisely/project/project_meta.py +27 -1
  173. supervisely/project/project_settings.py +32 -18
  174. supervisely/project/versioning/__init__.py +1 -0
  175. supervisely/project/versioning/common.py +20 -0
  176. supervisely/project/versioning/schema_fields.py +35 -0
  177. supervisely/project/versioning/video_schema.py +221 -0
  178. supervisely/project/versioning/volume_schema.py +87 -0
  179. supervisely/project/video_project.py +717 -15
  180. supervisely/project/volume_project.py +623 -5
  181. supervisely/template/experiment/experiment.html.jinja +4 -4
  182. supervisely/template/experiment/experiment_generator.py +14 -21
  183. supervisely/template/live_training/__init__.py +0 -0
  184. supervisely/template/live_training/header.html.jinja +96 -0
  185. supervisely/template/live_training/live_training.html.jinja +51 -0
  186. supervisely/template/live_training/live_training_generator.py +464 -0
  187. supervisely/template/live_training/sly-style.css +402 -0
  188. supervisely/template/live_training/template.html.jinja +18 -0
  189. supervisely/versions.json +28 -26
  190. supervisely/video/sampling.py +39 -20
  191. supervisely/video/video.py +41 -12
  192. supervisely/video_annotation/video_figure.py +38 -4
  193. supervisely/video_annotation/video_object.py +29 -4
  194. supervisely/volume/stl_converter.py +2 -0
  195. supervisely/worker_api/agent_rpc.py +24 -1
  196. supervisely/worker_api/rpc_servicer.py +31 -7
  197. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info}/METADATA +58 -40
  198. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info}/RECORD +203 -155
  199. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info}/WHEEL +1 -1
  200. supervisely_lib/__init__.py +6 -1
  201. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info}/entry_points.txt +0 -0
  202. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info/licenses}/LICENSE +0 -0
  203. {supervisely-6.73.438.dist-info → supervisely-6.73.513.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,464 @@
1
+ from __future__ import annotations
2
+
3
+ import os
4
+ import re
5
+ import math
6
+ from datetime import datetime
7
+ from pathlib import Path
8
+ from typing import Dict, Optional
9
+ import plotly.graph_objects as go # pylint: disable=import-error
10
+ from plotly.subplots import make_subplots # pylint: disable=import-error
11
+
12
+ import supervisely as sly
13
+ from supervisely import Api, ProjectMeta, logger
14
+ from supervisely.template.base_generator import BaseGenerator
15
+ from supervisely.imaging.color import rgb2hex
16
+
17
+
18
+ class LiveTrainingGenerator(BaseGenerator):
19
+ """
20
+ Generator for Live training session reports.
21
+
22
+ Logs:
23
+ - Model hyperparameters
24
+ - Training loss graphs
25
+ - Checkpoints
26
+ - Dataset size over time
27
+ """
28
+
29
+ def __init__(
30
+ self,
31
+ api: Api,
32
+ session_info: dict,
33
+ model_config: dict,
34
+ model_meta: ProjectMeta,
35
+ task_type: str,
36
+ output_dir: str = "./live_training_report",
37
+ team_id: Optional[int] = None,
38
+
39
+ ):
40
+ """
41
+ Initialize Live training generator.
42
+
43
+ :param api: Supervisely API instance
44
+ :param session_info: Session metadata (session_id, start_time, project_id, etc.)
45
+ :param model_config: Model configuration (hyperparameters, backbone, etc.)
46
+ :param model_meta: Model metadata with classes
47
+ :param output_dir: Local output directory
48
+ :param team_id: Team ID
49
+ """
50
+ super().__init__(api, output_dir)
51
+ self.team_id = team_id or sly.env.team_id()
52
+ self.session_info = session_info
53
+ self.model_config = model_config
54
+ self.model_meta = model_meta
55
+ self.task_type = task_type
56
+ self._slug_map = {
57
+ "semantic segmentation": "supervisely-ecosystem/live-training---semantic-segmentation",
58
+ "object detection": "supervisely-ecosystem/live-training---object-detection",
59
+ }
60
+ self.slug = self._slug_map[task_type]
61
+
62
+ # Validate required fields
63
+ self._validate_session_info()
64
+
65
+ def _validate_session_info(self):
66
+ """Validate that session_info contains required fields"""
67
+ required = ["session_id", "project_id", "start_time"]
68
+ missing = [k for k in required if k not in self.session_info]
69
+ if missing:
70
+ raise ValueError(f"Missing required fields in session_info: {missing}")
71
+
72
+ def _report_url(self, server_address: str, template_id: int) -> str:
73
+ """Generate URL to open the Live training report"""
74
+ return f"{server_address}/nn/experiments/{template_id}"
75
+
76
+ def context(self) -> dict:
77
+ return {
78
+ "env": self._get_env_context(),
79
+ "session": self._get_session_context(),
80
+ "model": self._get_model_context(),
81
+ "training": self._get_training_context(),
82
+ "dataset": self._get_dataset_context(),
83
+ "widgets": self._get_widgets_context(),
84
+ "resources": self._get_resources_context(),
85
+ }
86
+
87
+ def _get_env_context(self) -> dict:
88
+ """Environment info"""
89
+ return {
90
+ "server_address": self.api.server_address,
91
+ }
92
+
93
+ def _get_session_context(self) -> dict:
94
+ session_id = self.session_info["session_id"]
95
+ project_id = self.session_info["project_id"]
96
+ artifacts_dir = self.session_info.get("artifacts_dir", "")
97
+ task_id = self.session_info.get("task_id", session_id)
98
+
99
+ project_info = self.api.project.get_info_by_id(project_id)
100
+ project_url = f"{self.api.server_address}/projects/{project_id}/datasets"
101
+ artifacts_url = f"{self.api.server_address}/files/?path={artifacts_dir}" if artifacts_dir else None
102
+
103
+ return {
104
+ "id": session_id,
105
+ "task_id": task_id,
106
+ "name": self.session_info.get("session_name", f"Session {session_id}"),
107
+ "start_time": self.session_info["start_time"],
108
+ "duration": self.session_info.get("duration"),
109
+ "current_iteration": self.session_info.get("current_iteration", 0),
110
+ "artifacts_url": artifacts_url,
111
+ "artifacts_dir": artifacts_dir,
112
+ "project": {
113
+ "id": project_id,
114
+ "name": project_info.name if project_info else "Unknown",
115
+ "url": project_url if project_info else None,
116
+ },
117
+ "status": self.session_info.get("status", "running"),
118
+ }
119
+
120
+ @staticmethod
121
+ def parse_hyperparameters(config_path: str) -> dict:
122
+ """
123
+ Parse hyperparameters from MMEngine config file.
124
+
125
+ :param config_path: Path to config.py
126
+ :return: Dict with extracted hyperparameters
127
+ """
128
+ # TODO: only basic parsing for segmentation
129
+ hyperparams = {}
130
+
131
+ if not os.path.exists(config_path):
132
+ return hyperparams
133
+
134
+ with open(config_path, 'r') as f:
135
+ content = f.read()
136
+
137
+ # Extract crop_size
138
+ match = re.search(r'crop_size\s*=\s*\((\d+),\s*(\d+)\)', content)
139
+ if match:
140
+ hyperparams['crop_size'] = f"({match.group(1)}, {match.group(2)})"
141
+
142
+ # Extract learning rate
143
+ match = re.search(r'lr=([0-9.e-]+)', content)
144
+ if match:
145
+ hyperparams['learning_rate'] = float(match.group(1))
146
+
147
+ # Extract batch_size
148
+ match = re.search(r'batch_size=(\d+)', content)
149
+ if match:
150
+ hyperparams['batch_size'] = int(match.group(1))
151
+
152
+ # Extract max_epochs
153
+ match = re.search(r'max_epochs\s*=\s*(\d+)', content)
154
+ if match:
155
+ hyperparams['max_epochs'] = int(match.group(1))
156
+
157
+ # Extract weight_decay
158
+ match = re.search(r'weight_decay=([0-9.e-]+)', content)
159
+ if match:
160
+ hyperparams['weight_decay'] = float(match.group(1))
161
+
162
+ # Extract optimizer
163
+ match = re.search(r"optimizer=dict\(type='(\w+)'", content)
164
+ if match:
165
+ hyperparams['optimizer'] = match.group(1)
166
+
167
+ return hyperparams
168
+
169
+ def _get_model_context(self) -> dict:
170
+ """Model configuration info"""
171
+ classes = [cls.name for cls in self.model_meta.obj_classes if cls.name != "_background_"]
172
+ display_name = self.model_config.get("display_name", self.model_config.get("model_name", "Unknown"))
173
+
174
+ return {
175
+ "name": display_name,
176
+ "backbone": self.model_config.get("backbone", "N/A"),
177
+ "num_classes": len(classes),
178
+ "classes": classes,
179
+ "classes_short": classes[:3] + (["..."] if len(classes) > 3 else []),
180
+ "config_file": self.model_config.get("config_file", "N/A"),
181
+ "task_type": self.model_config.get("task_type", "Live Training"),
182
+ }
183
+
184
+ def _get_training_context(self) -> dict:
185
+ """Training logs and checkpoints"""
186
+ logs_path = self.session_info.get("logs_dir")
187
+ logs_url = None
188
+ if logs_path:
189
+ logs_url = f"{self.api.server_address}/files/?path={logs_path}"
190
+
191
+ checkpoints = []
192
+ artifacts_dir = self.session_info.get("artifacts_dir", "")
193
+ for ckpt in self.session_info.get("checkpoints", []):
194
+ checkpoint = {
195
+ "name": ckpt["name"],
196
+ "iteration": ckpt["iteration"],
197
+ "loss": ckpt.get("loss"),
198
+ "url": f"{self.api.server_address}/files/?path={artifacts_dir}/checkpoints/{ckpt['name']}",
199
+ }
200
+ checkpoints.append(checkpoint)
201
+
202
+ # Get total iterations from loss_history or checkpoints
203
+ loss_history = self.session_info.get("loss_history", [])
204
+ # Handle both old (list) and new (dict) formats
205
+ if isinstance(loss_history, list) and loss_history:
206
+ total_iterations = loss_history[-1]["iteration"]
207
+ elif isinstance(loss_history, dict):
208
+ # Get max step from any metric
209
+ total_iterations = max(
210
+ (item["step"] for metric_data in loss_history.values() for item in metric_data),
211
+ default=0
212
+ ) if loss_history else 0
213
+ else:
214
+ total_iterations = max([c["iteration"] for c in self.session_info.get("checkpoints", [])]) if self.session_info.get("checkpoints") else 0
215
+
216
+
217
+ return {
218
+ "total_iterations": total_iterations,
219
+ "device": self.session_info.get("device", "N/A"),
220
+ "session_url": self.session_info.get("session_url"),
221
+ "checkpoints": checkpoints,
222
+ "logs": {
223
+ "path": logs_path,
224
+ "url": logs_url,
225
+ },
226
+ }
227
+
228
+ def _get_dataset_context(self) -> dict:
229
+ """Dataset info"""
230
+ return {
231
+ "current_size": self.session_info.get("dataset_size", 0),
232
+ "initial_samples": self.session_info.get("initial_samples", 0),
233
+ }
234
+
235
+ def _get_training_plots_html(self) -> Optional[str]:
236
+ """
237
+ Generate HTML for training loss plot.
238
+ Currently returns None - to be implemented later with actual loss data.
239
+ """
240
+ # TODO: Generate plot from loss history
241
+ # For now return placeholder
242
+ return None
243
+
244
+ def _generate_classes_table(self) -> str:
245
+ """Generate HTML table with class names, shapes and colors.
246
+
247
+ :returns: HTML string with classes table
248
+ :rtype: str
249
+ """
250
+ type_to_icon = {
251
+ sly.AnyGeometry: "zmdi zmdi-shape",
252
+ sly.Rectangle: "zmdi zmdi-crop-din",
253
+ sly.Polygon: "",
254
+ sly.Bitmap: "zmdi zmdi-brush",
255
+ sly.Polyline: "zmdi zmdi-gesture",
256
+ sly.Point: "zmdi zmdi-dot-circle-alt",
257
+ sly.Cuboid: "zmdi zmdi-ungroup", #
258
+ sly.GraphNodes: "zmdi zmdi-grain",
259
+ sly.MultichannelBitmap: "zmdi zmdi-layers",
260
+ }
261
+
262
+ if not hasattr(self.model_meta, "obj_classes"):
263
+ return None
264
+
265
+ if len(self.model_meta.obj_classes) == 0:
266
+ return None
267
+
268
+ html = ['<table class="table">']
269
+ html.append("<thead><tr><th>Class name</th><th>Shape</th></tr></thead>")
270
+ html.append("<tbody>")
271
+
272
+ for obj_class in self.model_meta.obj_classes:
273
+ class_name = obj_class.name
274
+ color_hex = rgb2hex(obj_class.color)
275
+ icon = type_to_icon.get(obj_class.geometry_type, "zmdi zmdi-shape")
276
+
277
+ class_cell = (
278
+ f"<i class='zmdi zmdi-circle' style='color: {color_hex}; margin-right: 5px;'></i>"
279
+ f"<span>{class_name}</span>"
280
+ )
281
+
282
+ if isinstance(icon, str) and icon.startswith("data:image"):
283
+ shape_cell = f"<img src='{icon}' style='height: 15px; margin-right: 2px;'/>"
284
+ else:
285
+ shape_cell = f"<i class='{icon}' style='margin-right: 5px;'></i>"
286
+
287
+ shape_name = obj_class.geometry_type.geometry_name()
288
+ shape_cell += f"<span>{shape_name}</span>"
289
+
290
+ html.append(f"<tr><td>{class_cell}</td><td>{shape_cell}</td></tr>")
291
+
292
+ html.append("</tbody>")
293
+ html.append("</table>")
294
+ return "\n".join(html)
295
+
296
+ def upload_to_artifacts(self, remote_dir: str):
297
+ """
298
+ Upload report to team files.
299
+
300
+ Default path: /live-training/{project_id}_{project_name}/{session_id}/
301
+ """
302
+ # Normalize path - remove trailing slash
303
+ remote_dir = remote_dir.rstrip("/")
304
+ file_info = self.upload(remote_dir, team_id=self.team_id)
305
+ self._report_file_info = file_info
306
+ return file_info
307
+
308
+ def _get_widgets_context(self) -> dict:
309
+ """Generate widgets (tables, plots) for the report"""
310
+ checkpoints_table = self._generate_checkpoints_table()
311
+ training_plot = self._generate_training_plot()
312
+ classes = self._generate_classes_table()
313
+
314
+ return {
315
+ "tables": {
316
+ "checkpoints": checkpoints_table,
317
+ "classes": classes,
318
+ },
319
+ "training_plot": training_plot,
320
+ }
321
+
322
+ def _generate_checkpoints_table(self) -> Optional[str]:
323
+ """Generate HTML table with checkpoints"""
324
+ # Get training context to access checkpoints with URLs
325
+ training_ctx = self._get_training_context()
326
+ checkpoints = training_ctx.get("checkpoints", [])
327
+
328
+ if not checkpoints:
329
+ return None
330
+
331
+ html = ['<table class="table">']
332
+ html.append("<thead><tr><th>Checkpoint Name</th><th>Iteration</th><th>Loss</th><th>Actions</th></tr></thead>")
333
+ html.append("<tbody>")
334
+
335
+ for checkpoint in checkpoints:
336
+ name = checkpoint.get("name", "N/A")
337
+ iteration = checkpoint.get("iteration", "N/A")
338
+ loss = checkpoint.get("loss")
339
+ url = checkpoint.get("url", "")
340
+ loss_str = f"{loss:.6f}" if loss is not None else "N/A"
341
+
342
+ download_link = f'<a href="{url}" target="_blank" class="download-link">Download</a>' if url else ""
343
+
344
+ html.append(f"<tr><td>{name}</td><td>{iteration}</td><td>{loss_str}</td><td>{download_link}</td></tr>")
345
+
346
+ html.append("</tbody>")
347
+ html.append("</table>")
348
+ return "\n".join(html)
349
+
350
+ def _generate_training_plot(self) -> str:
351
+ """Generate training plots grid (like Experiments)"""
352
+ loss_history = self.session_info.get("loss_history", {})
353
+
354
+ if not loss_history or not isinstance(loss_history, dict):
355
+ return "<p>No training data available yet.</p>"
356
+
357
+ # Get all metrics
358
+ metrics = list(loss_history.keys())
359
+ n_metrics = len(metrics)
360
+
361
+ if n_metrics == 0:
362
+ return "<p>No training data available yet.</p>"
363
+
364
+ # Calculate grid size (like in Experiments)
365
+ side = min(4, max(2, math.ceil(math.sqrt(n_metrics))))
366
+ cols = side
367
+ rows = math.ceil(n_metrics / cols)
368
+
369
+ # Create subplots
370
+ fig = make_subplots(rows=rows, cols=cols, subplot_titles=metrics)
371
+
372
+ for idx, metric in enumerate(metrics, start=1):
373
+ data = loss_history[metric]
374
+ if not data:
375
+ continue
376
+
377
+ steps = [item["step"] for item in data]
378
+ values = [item["value"] for item in data]
379
+
380
+ row = (idx - 1) // cols + 1
381
+ col = (idx - 1) % cols + 1
382
+
383
+ fig.add_trace(
384
+ go.Scatter(
385
+ x=steps,
386
+ y=values,
387
+ mode="lines",
388
+ name=metric,
389
+ showlegend=False,
390
+ ),
391
+ row=row,
392
+ col=col,
393
+ )
394
+
395
+ # Special formatting for training rate
396
+ if metric.startswith("lr"):
397
+ fig.update_yaxes(tickformat=".0e", row=row, col=col)
398
+
399
+ fig.update_layout(
400
+ height=300 * rows,
401
+ width=400 * cols,
402
+ showlegend=False,
403
+ )
404
+
405
+ # Save as PNG
406
+ data_dir = os.path.join(self.output_dir, "data")
407
+ os.makedirs(data_dir, exist_ok=True)
408
+ img_path = os.path.join(data_dir, "training_plots_grid.png")
409
+
410
+ try:
411
+ fig.write_image(img_path, engine="kaleido")
412
+ except Exception as e:
413
+ logger.warning(f"Failed to save training plot: {e}")
414
+ return "<p>Failed to generate training plot</p>"
415
+
416
+ # Return Vue image component
417
+ return f'<sly-iw-image src="/data/training_plots_grid.png" :template-base-path="templateBasePath" :options="{{ style: {{ width: \'70%\', height: \'auto\' }} }}" />'
418
+
419
+ def _get_online_training_app_info(self):
420
+ """Get online training app info from ecosystem"""
421
+ try:
422
+ # TODO: only works for public apps.
423
+ # Exception handles only private apps on dev server. Need implement for private apps on any server.
424
+ module_id = self.api.app.get_ecosystem_module_id(self.slug)
425
+ except Exception as e:
426
+ logger.warning(f"Failed to get module ID for slug {self.slug}: {e}.")
427
+ if self.api.server_address.endswith("dev.internal.supervisely.com"):
428
+ logger.warning("Using hardcoded module ID for dev server")
429
+ task2module_map = {
430
+ "object detection": 620,
431
+ "semantic segmentation": 621,
432
+ }
433
+ module_id = task2module_map.get(self.task_type)
434
+ else:
435
+ raise e
436
+ return {
437
+ "slug": self.slug,
438
+ "module_id": module_id,
439
+ }
440
+
441
+ def _get_resources_context(self):
442
+ """Return apps module IDs for buttons"""
443
+ online_training_app = self._get_online_training_app_info()
444
+
445
+ return {
446
+ "apps": {
447
+ "online_training": online_training_app,
448
+ }
449
+ }
450
+
451
+ def get_report(self) -> str:
452
+ """Get report URL after upload"""
453
+ if self._report_file_info is None:
454
+ raise RuntimeError("Report not uploaded yet. Call upload_to_artifacts() first.")
455
+
456
+ # self._report_file_info is file_id (int), not FileInfo object
457
+ file_id = self._report_file_info if isinstance(self._report_file_info, int) else self._report_file_info.id
458
+ return self._report_url(self.api.server_address, file_id)
459
+
460
+ def get_report_id(self) -> int:
461
+ """Get report file ID"""
462
+ if self._report_file_info is None:
463
+ raise RuntimeError("Report not uploaded yet. Call upload_to_artifacts() first.")
464
+ return self._report_file_info.id