Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Action: Check the status of K8s CronJob pods #888

Merged
merged 4 commits into from
Oct 12, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
24 changes: 24 additions & 0 deletions Kubernetes/legos/k8s_check_cronjob_pod_status/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
[<img align="left" src="https://unskript.com/assets/favicon.png" width="100" height="100" style="padding-right: 5px">]
(https://unskript.com/assets/favicon.png)
<h1>Checks the status of CronJob pods</h1>

## Description
This action checks the status of CronJob pods

## Lego Details
k8s_check_cronjob_pod_status(handle, namespace: str="")
handle: Object of type unSkript K8S Connector.
namespace: Namespace where the CronJob is deployed.


## Lego Input
This Lego takes inputs handle namespace(Optional)

## Lego Output
Here is a sample output.
<img src="./1.png">
<img src="./2.png">

## See it in Action

You can see this Lego in action following this link [unSkript Live](https://us.app.unskript.io)
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
{
"action_title": "Check the status of K8s CronJob pods",
"action_description": "This action checks the status of CronJob pods",
"action_type": "LEGO_TYPE_K8S",
"action_entry_function": "k8s_check_cronjob_pod_status",
"action_needs_credential": true,
"action_output_type": "ACTION_OUTPUT_TYPE_LIST",
"action_is_check": true,
"action_next_hop": [
""
],
"action_next_hop_parameter_mapping": {},
"action_supports_iteration": true,
"action_supports_poll": true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
from __future__ import annotations

#
# Copyright (c) 2023 unSkript.com
# All rights reserved.
#
from datetime import datetime, timezone
from kubernetes import client
from typing import Tuple, Optional
from pydantic import BaseModel, Field
from croniter import croniter


class InputSchema(BaseModel):
namespace: Optional[str] = Field(..., description='k8s Namespace', title='Namespace')


def k8s_check_cronjob_pod_status_printer(output):
status, issues = output
if status:
print("CronJobs are running as expected.")
else:
for issue in issues:
print(f"CronJob '{issue['cronjob_name']}' Alert: {issue['message']}")


def k8s_check_cronjob_pod_status(handle, namespace: str='') -> Tuple:
"""
Checks the status of the CronJob pods.

:type handle: object
:param handle: The Kubernetes client handle.

:type name: str
:param namespace: Namespace where the CronJob is deployed.

:return: A tuple where the first item has the status if the check and second has a list of failed objects.
"""
# Initialize the K8s API clients
batch_v1 = client.BatchV1Api(api_client=handle)
batch_v1beta1 = client.BatchV1beta1Api(api_client=handle)
core_v1 = client.CoreV1Api(api_client=handle)

issues = {"NotAssociated": [], "Pending": [], "UnexpectedState": []}

# Get namespaces to check
if namespace:
namespaces = [namespace]
else:
ns_obj = core_v1.list_namespace()
namespaces = [ns.metadata.name for ns in ns_obj.items]

for ns in namespaces:
# Fetch all CronJobs in the namespace
cronjobs = batch_v1beta1.list_namespaced_cron_job(ns).items

for cronjob in cronjobs:
schedule = cronjob.spec.schedule

# Calculate the next expected run
now = datetime.now(timezone.utc)
iter = croniter(schedule, now)
next_run = iter.get_next(datetime)
time_to_next_run = next_run - now

# Fetch the most recent Job associated with the CronJob
jobs = batch_v1.list_namespaced_job(ns) # Fetch all jobs, and then filter by prefix.

associated_jobs = [job for job in jobs.items if job.metadata.name.startswith(cronjob.metadata.name)]
if not associated_jobs:
issues["NotAssociated"].append({"pod_name": cronjob.metadata.name, "namespace": ns})
continue

latest_job = sorted(associated_jobs, key=lambda x: x.status.start_time, reverse=True)[0]

# Check job's pods for any issues
pods = core_v1.list_namespaced_pod(ns, label_selector=f"job-name={latest_job.metadata.name}")

for pod in pods.items:
if pod.status.phase == 'Pending' and now - pod.status.start_time > time_to_next_run:
issues["Pending"].append({"pod_name": pod.metadata.name, "namespace": ns})
elif pod.status.phase not in ['Running', 'Succeeded']:
issues["UnexpectedState"].append({"pod_name": pod.metadata.name, "namespace": ns, "state": pod.status.phase})

if all(not val for val in issues.values()):
return (True, None)
else:
return (False, issues)




Loading