qontract-reconcile 0.10.1rc824__py3-none-any.whl → 0.10.1rc826__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.1rc824
3
+ Version: 0.10.1rc826
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
@@ -445,7 +445,7 @@ reconcile/statuspage/state.py,sha256=HD9EOoKm_nEqCMLIwW809En3cq5VhyzKJPUbsh-bae8
445
445
  reconcile/statuspage/status.py,sha256=mfRJ_tW7jM4_Vy_1cc8C0fKJEoA2GwrA3gJeV1KImAw,2834
446
446
  reconcile/statuspage/integrations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
447
447
  reconcile/statuspage/integrations/components.py,sha256=49KHd_E9AdRvcEA6n75q1McZv2LfN-hRsW-WA7dgw9g,2651
448
- reconcile/statuspage/integrations/maintenances.py,sha256=7gmMS9HwORMWMMBtib6pNGLXR6sAH2aW-HXKHJn6BN4,3083
448
+ reconcile/statuspage/integrations/maintenances.py,sha256=nG45b8A_ir9Ivtr4zH_VpHMahq8wev_M5igqK6T0hP0,4225
449
449
  reconcile/templates/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
450
450
  reconcile/templates/aws_access_key_email.j2,sha256=2MUr1ERmyISzKgHqsWYLd-1Wbl-peUa-FsGUS-JLUFc,238
451
451
  reconcile/templates/email.yml.j2,sha256=OZgczNRgXPj2gVYTgwQyHAQrMGu7xp-e4W1rX19GcrU,690
@@ -539,7 +539,7 @@ reconcile/test/test_terraform_tgw_attachments.py,sha256=rHZHUtDxewpKsRj3nfm2bZ2J
539
539
  reconcile/test/test_terraform_users.py,sha256=XOAfGvITCJPI1LTlISmHbA4ONMQMkxYUMTsny7pQCFw,4319
540
540
  reconcile/test/test_terraform_vpc_peerings.py,sha256=ubcsKh0TrUIwuI1-W3ETIgzsFvzAyeoFmEJFC-IK6JY,20538
541
541
  reconcile/test/test_terraform_vpc_peerings_build_desired_state.py,sha256=DAfpb12I0PlqnuVUHK2vh4LH4d1OylT3H2GE_3TGZZI,47852
542
- reconcile/test/test_three_way_diff_strategy.py,sha256=2fjEqE2w4pIzKq18PRcADTSe01aGwsZfMGloU8xfNaE,3346
542
+ reconcile/test/test_three_way_diff_strategy.py,sha256=0QY2hzOrTVnQxDFbdOJBOIIHEKKOA5RmGftT0QXABeY,3697
543
543
  reconcile/test/test_utils_jinja2.py,sha256=TpzQlpFnLGzNEZp5WOh0o7AuBiGEktqO4MuwiiJW2YY,3895
544
544
  reconcile/test/test_vault_replication.py,sha256=wlc4jm9f8P641UvvxIFFFc5_unJysNkOVrKJscjhQr0,16867
545
545
  reconcile/test/test_vault_utils.py,sha256=vbJnc89XAuE07qbTuWxHM5o9F6R9SO5aHXA38fwxT7A,1122
@@ -654,7 +654,7 @@ reconcile/utils/oc_connection_parameters.py,sha256=85slrnDigYwYmzhyceVkMElWzFArp
654
654
  reconcile/utils/oc_filters.py,sha256=R2Lf3fo0jQCeE62Ygeo_KN24XbAosq0QbjimYG6qHI4,1402
655
655
  reconcile/utils/oc_map.py,sha256=nT69J5pdPeIDnIYjD9fwY6GkE3BMQCf-AF0rmHJuUNw,9068
656
656
  reconcile/utils/ocm_base_client.py,sha256=X8qkPXfpfJdBKBtFv7zyGD33HNAEBJL8owf-ykrt-Ts,6469
657
- reconcile/utils/openshift_resource.py,sha256=Gxhc3oyYDCfMix7RDVvNAZX9O1bQII0KAjGsC5pu7oA,24831
657
+ reconcile/utils/openshift_resource.py,sha256=l5VLvwZ1LRi3l8InMdJS0oc1w5juhgwpWD2Yg7rXubc,24763
658
658
  reconcile/utils/openssl.py,sha256=QVvhzhpChq_4Daf_5wE1qeZJr4thg3DDjJPn4bOPD4E,365
659
659
  reconcile/utils/output.py,sha256=I_kXYyPcN1mlZmX16ZnLNGkhhwnal640GIdIaGJd4wE,2026
660
660
  reconcile/utils/pagerduty_api.py,sha256=fcSAUez6w51woDvbm0plJW2qSw6_NXQs1Fit_KTNitc,7653
@@ -679,7 +679,7 @@ reconcile/utils/structs.py,sha256=LcbLEg8WxfRqM6nW7NhcWN0YeqF7SQzxOgntmLs1SgY,35
679
679
  reconcile/utils/template.py,sha256=wTvRU4AnAV_o042tD4Mwls2dwWMuk7MKnde3MaCjaYg,331
680
680
  reconcile/utils/terraform_client.py,sha256=mZEKpu6nbfiQd60wRkc8-5sljBTUTOgaAKnF89itMzU,32085
681
681
  reconcile/utils/terrascript_aws_client.py,sha256=VlvIHgrZRiMFVgx6a8ZHxoiJoDwqbtOmsZFnwwNrdL0,273199
682
- reconcile/utils/three_way_diff_strategy.py,sha256=nyqeQsLCoPI6e16k2CF3b9KNgQLU-rPf5RtfdUfVMwE,4468
682
+ reconcile/utils/three_way_diff_strategy.py,sha256=nnTk4VDex1GIEqryFVfATU1AeRwzXyoPl_FX6zzmRFI,4643
683
683
  reconcile/utils/throughput.py,sha256=iP4UWAe2LVhDo69mPPmgo9nQ7RxHD6_GS8MZe-aSiuM,344
684
684
  reconcile/utils/vault.py,sha256=AYGG5aDJ7CSVhTFdZowfEg3iSQWenoAt676aGjHQMX8,14978
685
685
  reconcile/utils/vaultsecretref.py,sha256=3Ed2uBy36TzSvL0B-l4FoWQqB2SbBKDKEuUPIO608Bo,931
@@ -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.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,,
814
+ qontract_reconcile-0.10.1rc826.dist-info/METADATA,sha256=bf-WVmAA-y5rqqFgz1O-fs_ZZJp46ypvmDu0EB2u8F8,2314
815
+ qontract_reconcile-0.10.1rc826.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
816
+ qontract_reconcile-0.10.1rc826.dist-info/entry_points.txt,sha256=GKQqCl2j2X1BJQ69een6rHcR26PmnxnONLNOQB-nRjY,491
817
+ qontract_reconcile-0.10.1rc826.dist-info/top_level.txt,sha256=l5ISPoXzt0SdR4jVdkfa7RPSKNc8zAHYWAnR-Dw8Ey8,24
818
+ qontract_reconcile-0.10.1rc826.dist-info/RECORD,,
@@ -1,10 +1,12 @@
1
1
  import logging
2
2
  import sys
3
- from datetime import datetime, timezone
3
+ from datetime import datetime, timedelta, timezone
4
4
 
5
+ from reconcile.slack_base import slackapi_from_queries
5
6
  from reconcile.statuspage.atlassian import AtlassianStatusPageProvider
6
7
  from reconcile.statuspage.integration import get_binding_state, get_status_pages
7
8
  from reconcile.statuspage.page import StatusMaintenance
9
+ from reconcile.statuspage.state import S3ComponentBindingState
8
10
  from reconcile.utils.differ import diff_iterables
9
11
  from reconcile.utils.runtime.integration import (
10
12
  NoParams,
@@ -44,6 +46,25 @@ class StatusPageMaintenancesIntegration(QontractReconcileIntegration[NoParams]):
44
46
  f"Delete StatusPage Maintenance is not supported at this time: {d.name}"
45
47
  )
46
48
 
49
+ def notify(
50
+ self,
51
+ dry_run: bool,
52
+ desired_state: list[StatusMaintenance],
53
+ binding_state: S3ComponentBindingState,
54
+ ) -> None:
55
+ now = datetime.now(timezone.utc)
56
+ slack = slackapi_from_queries(QONTRACT_INTEGRATION, init_usergroups=False)
57
+ for m in desired_state:
58
+ scheduled_start = datetime.fromisoformat(m.schedule_start)
59
+ if now <= scheduled_start <= now + timedelta(hours=1):
60
+ state_key = f"notifications/{m.name}"
61
+ if binding_state.state.exists(state_key):
62
+ continue
63
+ logging.info(f"Notify StatusPage Maintenance: {m.name}")
64
+ if not dry_run:
65
+ slack.chat_post_message(m.message)
66
+ binding_state.state.add(f"notifications/{m.name}")
67
+
47
68
  def run(self, dry_run: bool = False) -> None:
48
69
  binding_state = get_binding_state(self.name, self.secret_reader)
49
70
  pages = get_status_pages()
@@ -75,6 +96,11 @@ class StatusPageMaintenancesIntegration(QontractReconcileIntegration[NoParams]):
75
96
  current_state=current_state,
76
97
  provider=page_provider,
77
98
  )
99
+ self.notify(
100
+ dry_run=dry_run,
101
+ desired_state=desired_state,
102
+ binding_state=binding_state,
103
+ )
78
104
  except Exception:
79
105
  logging.exception(f"failed to reconcile statuspage {p.name}")
80
106
  error = True
@@ -109,3 +109,13 @@ def test_3wpd_change_empty_env_value_should_not_apply(deployment):
109
109
  ]
110
110
 
111
111
  assert three_way_diff_using_hash(c_item, d_item) is True
112
+
113
+
114
+ def test_3wpd_diff_detects_missing_annotation(deployment):
115
+ d_item = OR(deployment, "", "")
116
+ d_item.body["metadata"]["annotations"]["new-annotation"] = "test-value"
117
+ c_item = d_item.annotate(canonicalize=False)
118
+
119
+ del c_item.body["metadata"]["annotations"]["new-annotation"]
120
+
121
+ assert three_way_diff_using_hash(c_item, d_item) is False
@@ -17,7 +17,6 @@ from pydantic import BaseModel
17
17
  from reconcile.utils.metrics import GaugeMetric
18
18
 
19
19
  SECRET_MAX_KEY_LENGTH = 253
20
- LAC_ANNOTATION = "kubectl.kubernetes.io/last-applied-configuration"
21
20
 
22
21
 
23
22
  class ResourceKeyExistsError(Exception):
@@ -20,6 +20,8 @@ NORMALIZE_COMPARE_EXCLUDED_ATTRS = {
20
20
  "managedFields",
21
21
  "namespace",
22
22
  }
23
+ K8S_ANNOTATION_LAC = "kubectl.kubernetes.io/last-applied-configuration"
24
+ NORMALIZE_IGNORE_ANNOTATIONS = QONTRACT_ANNOTATIONS | {K8S_ANNOTATION_LAC}
23
25
 
24
26
 
25
27
  def _normalize_secret(secret: OR) -> None:
@@ -59,9 +61,9 @@ def normalize_object(item: OR) -> OR:
59
61
  validate_k8s_object=False,
60
62
  )
61
63
 
62
- annotations = n.body.get("annotations", {})
64
+ annotations = n.body.get("metadata", {}).get("annotations", {})
63
65
  metadata["annotations"] = {
64
- k: v for k, v in annotations.items() if k not in QONTRACT_ANNOTATIONS
66
+ k: v for k, v in annotations.items() if k not in NORMALIZE_IGNORE_ANNOTATIONS
65
67
  }
66
68
 
67
69
  # Run normalizers on Kinds with special needs