Metadata-Version: 2.1
Name: karya-client
Version: 1.0.0
Summary: Python client to interract with Karya; the open sourced distributed job scheduler
Home-page: https://saumya-bhatt.github.io/karya-python-client/
Author: Saumya Bhatt
Author-email: 108668549+SaumyaBhatt106@users.noreply.github.com
Requires-Python: >=3.10,<4.0
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Requires-Dist: httpx (>=0.28.1,<0.29.0)
Requires-Dist: pydantic (>=2.10.3,<3.0.0)
Requires-Dist: ruff (>=0.8.3,<0.9.0)
Requires-Dist: sphinx (>=8.1.3,<9.0.0)
Requires-Dist: sphinx-rtd-theme (>=3.0.2,<4.0.0)
Project-URL: Documentation, https://saumya-bhatt.github.io/karya-python-client/
Project-URL: Repository, https://github.com/Saumya-Bhatt/karya-python-client
Description-Content-Type: text/markdown

# Karya Python Client

This here is the Python client to interract with [Karya - the open sourced distributed job scheduler](https://github.com/Saumya-Bhatt/karya)

- [API Docs](https://saumya-bhatt.github.io/karya-python-client)
- [How to contribute](./.github/CONTRIBUTING.md)

---

## Getting Started

This section highlights the steps to get started with the Karya Python client.

### Installation

```shell
    pip install karya-client
```

The distribution files can also be found here - [Github Release](https://github.com/Saumya-Bhatt/karya-python-client/releases).

### Useage Examples

A list of samples to configure different plans with various actions and hooks can be found [here](https://saumya-bhatt.github.io/karya-python-client/#usage-examples)

### Using the Client

Do refer to the [Client API Documentation](https://saumya-bhatt.github.io/karya-python-client/#module-karya.commons.client) to understand the various methods and classes provided by the client.

1. Create a config object:

   ```python
   from karya.clients.config import ClientConfig
   from karya.entities.enums import Protocol

   ## point this to where the Karya server is running
   config = ClientConfig(
       protocol=Portocol.HTTP,
       host='localhost',
       port=8080
   )

   ## For localsetup, a default config is provided as: ClientConfig.dev()
   ```

2. Create a client object:

   ```python
   from karya.clients import KaryaRestClient

   client = KaryaRestClient(config)
   ```

3. Creat a user. Only a user configured in the Karya server can be used to create a client object.

   ```python
   from karya.clients.requests import CreateUserRequest

   create_user_request = CreateUserRequest(name="python-client")
   user = await client.create_user(create_user_request)
   ```

4. Specify the action that you would want to trigger once the task is scheduled.

   ```python
   from karya.entities.actions import RestApiRequest

   ## For example, we shall be making a POST request to a local server
   action = RestApiRequest(
       protocol=Protocol.HTTPS,  # Use HTTPS for secure communication
       base_url="localhost",  # Base URL for the REST API
       method=Method.POST,  # HTTP method for the request (POST)
       headers={"content-type": "application/json"},  # Set the content type to JSON
       body=RestApiRequest.JsonBody.from_dict(
           {"message": "Hello from python client"}
       ),  # JSON body to send in the request
       timeout=2000,  # Timeout for the request (in milliseconds)
   )
   ```

5. Submit the plan to Karya.

   > `period_time` has to be in the [ISO 8601](https://en.wikipedia.org/wiki/ISO_8601#Durations) format.

   ```python
   from karya.clients.requests SubmitPlanRequest
   from karya.entities.plan_types import Recurring

   # For example, we shall be submitting a recurring plan
   plan_request = SubmitPlanRequest(
       user_id=user.id,  # Use the created user's ID
       description="Make a recurring API call from python client",  # Description of the plan
       period_time="PT7S",  # Time period between each execution (7 seconds)
       max_failure_retry=3,  # Retry count in case of failure
       plan_type=Recurring(  # Define a recurring plan
           end_at=None,  # No specific end time, so it will continue indefinitely
       ),
       action=rest_action,  # The action to be executed as part of the plan (REST API call)
   )

   plan = await client.submit_plan(plan_request)
   ```

6. And you're done! The plan will be executed as per the schedule:

   - The action will be triggered every 7 seconds.
   - The action will make a POST request to `localhost` with the JSON body `{"message": "Hello from python client"}`
   - The request will have a timeout of 2 seconds.

---

## Plan Type

Karya supports the following plan types:

### One Time

This can be used to trigger a delayed action.

```python
from karya.entities.plan_types import OneTime

val plan_type = OneTime()
```

### Recurring

This can be used to trigger an action periodically.

> **NOTE:** If the `endAt` field is not specified, the plan will run indefinitely.

```python
from karya.entities.plan_types import Recurring

plan_type = Recurring(end_at=1734694042) # define time in epoch-milli second
```

---

## Actions

Actions define what Karya should do once it has to execute the plan. The client supports the following actions:

### REST API Request

Make a REST API request to a specified URL with the given parameters.

```python
    rest_action = RestApiRequest(
        protocol=Protocol.HTTPS,  # Use HTTPS for secure communication
        base_url="localhost",  # Base URL for the REST API
        method=Method.POST,  # HTTP method for the request (POST)
        headers={"content-type": "application/json"},  # Set the content type to JSON
        body=RestApiRequest.JsonBody.from_dict(
            {"message": "Hello from python client"}
        ),  # JSON body to send in the request
        timeout=2000,  # Timeout for the request (in milliseconds)
    )
```

### Push to Kafka

Push a message to a Kafka topic.

```python
    kafka_action = KafkaPush(
        topic="test-topic",  # Kafka topic to push the message to
        message="Hello from python client",  # Message to push to the Kafka topic
    )
```

### Send Email

Send an email to a specified email address.

```python
    email_action = EmailRequest(
        recipient="recipient@gmail.com",  # Email recipient
        subject="Karya notification",  # Email subject
        message="Hello from Karya!",  # Email message body
    )
```

### Send a Slack Message

Send a message to a specified Slack channel.

```python
    slack_action = SlackMessage(
        channel="test-channel",  # Slack channel to send the message to
        message="Hello from python client",  # Message to send to the Slack channel
    )
```

### Chain another job

Chain another job to the current job.

```python
    chained_action = ChainedRequest(
        request=SubmitPlanRequest(
            user_id=user.id,  # Use the created user's ID
            description="Make a recurring API call from python client",  # Plan description
            period_time="PT5S",  # Time period between calls (5 seconds)
            max_failure_retry=3,  # Retry count in case of failure
            plan_type=Recurring(end_at=None),  # Recurring plan with no end time
            action=RestApiRequest(
                base_url="eox7wbcodh9parh.m.pipedream.net"
            ),  # API request action
        )
    )
```

---

## Hooks

[API documentation](https://saumya-bhatt.github.io/karya-python-client/index.html#module-karya.commons.entities.models.Hook)

```python
    hook = Hook(
        hook_type=HookType  # Hook type
        action=ActionType,  # Can be any of the actions specified above
    )
```

Hooks are used to trigger actions on certain triggers. The client supports the following hooks:

- `ON_FAILURE`: Trigger an action when the plan fails.
- `ON_COMPLETION`: Trigger an action when the plan completes successfully.

