qontract-reconcile 0.10.1rc822__py3-none-any.whl → 0.10.1rc824__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.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: qontract-reconcile
3
- Version: 0.10.1rc822
3
+ Version: 0.10.1rc824
4
4
  Summary: Collection of tools to reconcile services with their desired state as defined in the app-interface DB.
5
5
  Home-page: https://github.com/app-sre/qontract-reconcile
6
6
  Author: Red Hat App-SRE Team
@@ -438,7 +438,7 @@ reconcile/skupper_network/models.py,sha256=DNTI7HZv-rqY42GIIxyRuvroHLvdH6rJerjIq
438
438
  reconcile/skupper_network/reconciler.py,sha256=XS-1oKBr_1l3dYUAVqUH6gCHg1G5ZuOfY_7fgGVAiFA,9996
439
439
  reconcile/skupper_network/site_controller.py,sha256=A3K-62BjJ5HiFVydV0ouGoD1NwrO7XhAH15BHAcS9fk,1550
440
440
  reconcile/statuspage/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
441
- reconcile/statuspage/atlassian.py,sha256=odYbYX8l3oST7AYm66Q4MiaxSbvDzsqEO6VBYJggxkc,17420
441
+ reconcile/statuspage/atlassian.py,sha256=5awxH9ovbtgis_N4aabBvrlo2ZyLNLaO6ezIYDgVq0Y,18571
442
442
  reconcile/statuspage/integration.py,sha256=hsazrQMceJbr61nEkJLxJbHhudTGtFuH0mlCo66-2ug,711
443
443
  reconcile/statuspage/page.py,sha256=WHDwV2PXEo4WwI2EgPOkS6j_T7geZEDXTgSaqpDo75U,5101
444
444
  reconcile/statuspage/state.py,sha256=HD9EOoKm_nEqCMLIwW809En3cq5VhyzKJPUbsh-bae8,1617
@@ -784,7 +784,7 @@ tools/app_interface_metrics_exporter.py,sha256=zkwkxdAUAxjdc-pzx2_oJXG25fo0Fnyd5
784
784
  tools/app_interface_reporter.py,sha256=upA-J-n-HXHKVDINRuMR7vTt-iJvQORKUVi9D3leQto,17738
785
785
  tools/glitchtip_access_reporter.py,sha256=oPBnk_YoDuljU3v0FaChzOwwnk4vap1xEE67QEjzdqs,2948
786
786
  tools/glitchtip_access_revalidation.py,sha256=8kbBJk04mkq28kWoRDDkfCGIF3GRg3pJrFAh1sW0dbk,2821
787
- tools/qontract_cli.py,sha256=_hap0giaBiU0o7SbHRmFsZ3yqXq7ULjK_omuifKBUqo,117165
787
+ tools/qontract_cli.py,sha256=_-ianEfVdOU8ZCDjZ-jLfkNMtQT2qw3uwjO-XnTRsKw,117904
788
788
  tools/sd_app_sre_alert_report.py,sha256=e9vAdyenUz2f5c8-z-5WY0wv-SJ9aePKDH2r4IwB6pc,5063
789
789
  tools/template_validation.py,sha256=-U-lTGeLaci8yWPEblCJeev2DOlY1jM9QOOh-O1zts8,3376
790
790
  tools/cli_commands/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
@@ -811,8 +811,8 @@ tools/test/test_app_interface_metrics_exporter.py,sha256=SX7qL3D1SIRKFo95FoQztvf
811
811
  tools/test/test_qontract_cli.py,sha256=_D61RFGAN5x44CY1tYbouhlGXXABwYfxKSWSQx3Jrss,4941
812
812
  tools/test/test_sd_app_sre_alert_report.py,sha256=v363r9zM7__0kR5K6mvJoGFcM9BvE33fWAayrqkpojA,2116
813
813
  tools/test/test_sre_checkpoints.py,sha256=SKqPPTl9ua0RFdSSofnoQX-JZE6dFLO3LRhfQzqtfh8,2607
814
- qontract_reconcile-0.10.1rc822.dist-info/METADATA,sha256=PmZRp4zeELlKOlfpqmgMwLrUqzMhI0QPLN_a4rgFDyY,2314
815
- qontract_reconcile-0.10.1rc822.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
816
- qontract_reconcile-0.10.1rc822.dist-info/entry_points.txt,sha256=GKQqCl2j2X1BJQ69een6rHcR26PmnxnONLNOQB-nRjY,491
817
- qontract_reconcile-0.10.1rc822.dist-info/top_level.txt,sha256=l5ISPoXzt0SdR4jVdkfa7RPSKNc8zAHYWAnR-Dw8Ey8,24
818
- qontract_reconcile-0.10.1rc822.dist-info/RECORD,,
814
+ qontract_reconcile-0.10.1rc824.dist-info/METADATA,sha256=BuGUQL722lQm08tOKAkp1LW3RG0QP8YMMcot9jZ4DIM,2314
815
+ qontract_reconcile-0.10.1rc824.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
816
+ qontract_reconcile-0.10.1rc824.dist-info/entry_points.txt,sha256=GKQqCl2j2X1BJQ69een6rHcR26PmnxnONLNOQB-nRjY,491
817
+ qontract_reconcile-0.10.1rc824.dist-info/top_level.txt,sha256=l5ISPoXzt0SdR4jVdkfa7RPSKNc8zAHYWAnR-Dw8Ey8,24
818
+ qontract_reconcile-0.10.1rc824.dist-info/RECORD,,
@@ -56,9 +56,9 @@ class AtlassianRawMaintenance(BaseModel):
56
56
  scheduled_until: str
57
57
  incident_updates: list[AtlassianRawMaintenanceUpdate]
58
58
  components: list[AtlassianRawComponent]
59
- auto_transition_deliver_notifications_at_end: bool
60
- auto_transition_deliver_notifications_at_start: bool
61
- scheduled_remind_prior: bool
59
+ auto_transition_deliver_notifications_at_end: Optional[bool]
60
+ auto_transition_deliver_notifications_at_start: Optional[bool]
61
+ scheduled_remind_prior: Optional[bool]
62
62
 
63
63
 
64
64
  class AtlassianAPI:
@@ -301,8 +301,15 @@ class AtlassianStatusPageProvider:
301
301
 
302
302
  # component status
303
303
  desired_component_status = desired.desired_component_status()
304
- status_update_required = desired_component_status is not None and (
305
- not current or desired_component_status != current.status
304
+ active_maintenance_affecting_component = [
305
+ m
306
+ for m in self.active_maintenances
307
+ if desired.display_name in [c.name for c in m.components]
308
+ ]
309
+ status_update_required = (
310
+ desired_component_status is not None
311
+ and (not current or desired_component_status != current.status)
312
+ and not active_maintenance_affecting_component
306
313
  )
307
314
 
308
315
  # shortcut execution if there is nothing to do
@@ -326,7 +333,7 @@ class AtlassianStatusPageProvider:
326
333
  component_id=current_component.id,
327
334
  )
328
335
 
329
- # validte the component and check if the current state needs to be updated
336
+ # validate the component and check if the current state needs to be updated
330
337
  needs_update = self.should_apply(desired, current_component)
331
338
  if not needs_update:
332
339
  return
@@ -456,12 +463,26 @@ class AtlassianStatusPageProvider:
456
463
  ]
457
464
 
458
465
  def create_maintenance(self, maintenance: StatusMaintenance) -> None:
466
+ component_ids: list[str] = []
467
+ for sc in maintenance.components:
468
+ current_component, _ = self.lookup_component(sc)
469
+ if current_component:
470
+ component_ids.append(current_component.id)
459
471
  data = {
460
472
  "name": maintenance.name,
461
473
  "status": "scheduled",
462
474
  "scheduled_for": maintenance.schedule_start,
463
475
  "scheduled_until": maintenance.schedule_end,
464
476
  "body": maintenance.message,
477
+ "scheduled_remind_prior": maintenance.announcements.remind_subscribers,
478
+ "scheduled_auto_transition": True,
479
+ "scheduled_auto_in_progress": True,
480
+ "scheduled_auto_completed": True,
481
+ "component_ids": component_ids,
482
+ "auto_transition_to_maintenance_state": True,
483
+ "auto_transition_to_operational_state": True,
484
+ "auto_transition_deliver_notifications_at_start": maintenance.announcements.notify_subscribers_on_start,
485
+ "auto_transition_deliver_notifications_at_end": maintenance.announcements.notify_subscribers_on_completion,
465
486
  }
466
487
  incident_id = self._api.create_incident(data)
467
488
  self._bind_component(
tools/qontract_cli.py CHANGED
@@ -2619,17 +2619,57 @@ def maintenances(ctx):
2619
2619
  print_output(ctx.obj["options"], data, columns)
2620
2620
 
2621
2621
 
2622
+ class MigrationStatusCount:
2623
+ def __init__(self, app: str) -> None:
2624
+ self.app = app
2625
+ self._source = 0
2626
+ self._target = 0
2627
+
2628
+ def inc(self, source_or_target: str) -> None:
2629
+ match source_or_target:
2630
+ case "source":
2631
+ self._source += 1
2632
+ case "target":
2633
+ self._target += 1
2634
+ case _:
2635
+ raise ValueError("hcp migration label must be source or target")
2636
+
2637
+ @property
2638
+ def classic(self) -> int:
2639
+ return self._source
2640
+
2641
+ @property
2642
+ def hcp(self) -> int:
2643
+ return self._target
2644
+
2645
+ @property
2646
+ def total(self) -> int:
2647
+ return self.classic + self.hcp
2648
+
2649
+ @property
2650
+ def progress(self) -> float:
2651
+ return round(self.hcp / self.total * 100, 0)
2652
+
2653
+ @property
2654
+ def item(self) -> dict[str, Any]:
2655
+ return {
2656
+ "app": self.app,
2657
+ "classic": self.classic or "0",
2658
+ "hcp": self.hcp or "0",
2659
+ "progress": self.progress or "0",
2660
+ }
2661
+
2662
+
2622
2663
  @get.command()
2623
2664
  @click.pass_context
2624
2665
  def hcp_migration_status(ctx):
2625
- counts: dict[str, dict[str, int]] = {}
2666
+ counts: dict[str, MigrationStatusCount] = {}
2667
+ total_count = MigrationStatusCount("total")
2626
2668
  saas_files = get_saas_files()
2627
2669
  for sf in saas_files:
2628
2670
  if sf.publish_job_logs:
2629
2671
  # ignore post deployment test saas files
2630
2672
  continue
2631
- app = sf.app.parent_app.name if sf.app.parent_app else sf.app.name
2632
- counts.setdefault(app, {"source": 0, "target": 0})
2633
2673
  for rt in sf.resource_templates:
2634
2674
  if rt.provider == "directory" or "dashboard" in rt.name:
2635
2675
  # ignore grafana dashboards
@@ -2644,26 +2684,15 @@ def hcp_migration_status(ctx):
2644
2684
  if t.delete:
2645
2685
  continue
2646
2686
  if hcp_migration := t.namespace.cluster.labels.get("hcp_migration"):
2647
- counts[app][hcp_migration] += 1
2648
-
2649
- data = []
2650
- for a, c in counts.items():
2651
- source = c["source"]
2652
- target = c["target"]
2653
- item = {}
2654
- item["app"] = a
2655
- item["classic"] = source or "0"
2656
- item["hcp"] = target or "0"
2657
- total = source + target
2658
- if total == 0:
2659
- continue
2660
- progress = round(target / total * 100, 2) or "0"
2661
- item["progress"] = progress
2662
- data.append(item)
2663
-
2664
- summary_completed = len([d for d in data if d["progress"] == 100])
2665
- print(f"SUMMARY: {summary_completed} / {len(data)} COMPLETED")
2687
+ app = sf.app.parent_app.name if sf.app.parent_app else sf.app.name
2688
+ counts.setdefault(app, MigrationStatusCount(app))
2689
+ counts[app].inc(hcp_migration)
2690
+ total_count.inc(hcp_migration)
2666
2691
 
2692
+ data = [c.item for c in counts.values()]
2693
+ print(
2694
+ f"SUMMARY: {total_count.hcp} / {total_count.total} COMPLETED ({total_count.progress}%)"
2695
+ )
2667
2696
  columns = ["app", "classic", "hcp", "progress"]
2668
2697
  print_output(ctx.obj["options"], data, columns)
2669
2698