Metadata-Version: 2.4
Name: palpluss
Version: 0.1.0
Summary: Official PalPluss Python SDK
Project-URL: Homepage, https://palpluss.com
Project-URL: Documentation, https://docs.palpluss.com
Project-URL: Repository, https://github.com/palpluss/palpluss-sdk
Project-URL: Bug Tracker, https://github.com/palpluss/palpluss-sdk/issues
Project-URL: Changelog, https://github.com/palpluss/palpluss-sdk/blob/main/packages/python/CHANGELOG.md
Author-email: PalPluss <dev@palpluss.com>
License: MIT
Keywords: b2c,kenya,mobile-money,mpesa,palpluss,payments,sdk,stk-push
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Topic :: Office/Business :: Financial
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Typing :: Typed
Requires-Python: >=3.9
Requires-Dist: httpx>=0.27.0
Provides-Extra: dev
Requires-Dist: build>=1.2.0; extra == 'dev'
Requires-Dist: mypy>=1.9.0; extra == 'dev'
Requires-Dist: pytest-asyncio>=0.23.0; extra == 'dev'
Requires-Dist: pytest-httpx>=0.30.0; extra == 'dev'
Requires-Dist: pytest>=8.0.0; extra == 'dev'
Requires-Dist: ruff>=0.4.0; extra == 'dev'
Requires-Dist: twine>=5.0.0; extra == 'dev'
Description-Content-Type: text/markdown

# PalPluss Python SDK

[![PyPI version](https://img.shields.io/pypi/v/palpluss.svg)](https://pypi.org/project/palpluss/)
[![Python](https://img.shields.io/pypi/pyversions/palpluss.svg)](https://pypi.org/project/palpluss/)
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)

Official Python SDK for the [PalPluss](https://palpluss.com) payments API — accept M-Pesa STK Push payments, send B2C payouts, and manage your service wallet from Python.

## Requirements

- Python >= 3.9
- `httpx` >= 0.27

## Installation

```bash
pip install palpluss
```

## Quick start

```python
from palpluss import PalPluss

client = PalPluss(api_key="pk_live_your_api_key")

# STK Push (Lipa na M-Pesa)
result = client.stk_push(
    amount=500,
    phone="254712345678",
    account_reference="ORDER-001",
)
print(result["transactionId"])  # "tx_..."
print(result["status"])         # "PENDING"
```

### Async usage

```python
import asyncio
from palpluss import AsyncPalPluss

async def main():
    async with AsyncPalPluss(api_key="pk_live_your_api_key") as client:
        result = await client.stk_push(amount=500, phone="254712345678")
        print(result["transactionId"])

asyncio.run(main())
```

## Configuration

| Parameter | Type | Default | Description |
|---|---|---|---|
| `api_key` | `str` | `PALPLUSS_API_KEY` env var | Your PalPluss API key |
| `timeout` | `float` | `30.0` | Request timeout in seconds |
| `auto_retry_on_rate_limit` | `bool` | `True` | Auto-retry on HTTP 429 |
| `max_retries` | `int` | `3` | Max retry attempts |

Set `PALPLUSS_BASE_URL` to override the API base URL (e.g. for sandbox).

## API Reference

### STK Push

```python
result = client.stk_push(
    amount=500,                         # Required: amount in KES
    phone="254712345678",               # Required: recipient phone
    account_reference="ORDER-001",      # Optional
    transaction_desc="Payment",         # Optional
    channel_id="ch_...",               # Optional
    callback_url="https://...",         # Optional
    credential_id="cred_...",           # Optional
)
```

### B2C Payout

```python
result = client.b2c_payout(
    amount=1000,
    phone="254712345678",
    reference="PAY-001",                # Optional
    description="Salary",               # Optional
    idempotency_key="my-unique-key",    # Optional — auto-generated if omitted
)
```

### Service Wallet

```python
# Get balance
balance = client.get_service_balance()
print(balance["availableBalance"])

# Topup
topup = client.service_topup(
    amount=5000,
    phone="254712345678",
    idempotency_key="topup-001",        # Optional — caller must provide for safe retry
)
```

### Transactions

```python
# Get single transaction
tx = client.get_transaction("tx_...")

# List transactions (cursor-based pagination)
page = client.list_transactions(limit=20, status="SUCCESS", type="STK")
print(page["items"])
print(page["next_cursor"])  # pass to next call, or None when exhausted

# Pagination
cursor = page["next_cursor"]
while cursor:
    page = client.list_transactions(limit=20, cursor=cursor)
    print(page["items"])
    cursor = page["next_cursor"]
```

### Payment Wallet Channels

```python
# Create
channel = client.create_channel(
    type="PAYBILL",
    shortcode="123456",
    name="Main Paybill",
)

# Update
channel = client.update_channel("ch_...", name="Updated Name")

# Delete
client.delete_channel("ch_...")
```

### Webhooks

```python
from palpluss import parse_webhook_payload

# In your webhook endpoint handler:
def webhook_handler(request_body: str):
    payload = parse_webhook_payload(request_body)
    print(payload["event_type"])           # "transaction.success"
    print(payload["transaction"]["status"]) # "SUCCESS"
```

## Error handling

```python
from palpluss import PalPluss, PalPlussApiError, RateLimitError

client = PalPluss(api_key="pk_live_...")

try:
    result = client.stk_push(amount=500, phone="bad-phone")
except RateLimitError as e:
    print(f"Rate limited, retry after {e.retry_after}s")
except PalPlussApiError as e:
    print(f"API error [{e.code}] {e.http_status}: {e}")
    print(f"Request ID: {e.request_id}")
```

### Error attributes

| Attribute | Type | Description |
|---|---|---|
| `message` | `str` | Human-readable error message |
| `code` | `str` | Machine-readable code (e.g. `INVALID_PHONE`) |
| `http_status` | `int` | HTTP status code |
| `details` | `dict` | Additional error context |
| `request_id` | `str \| None` | Trace ID for support |

`RateLimitError` additionally exposes `retry_after: int | None` (seconds).

## Context manager

```python
# Sync
with PalPluss(api_key="pk_live_...") as client:
    result = client.stk_push(amount=500, phone="254712345678")

# Async
async with AsyncPalPluss(api_key="pk_live_...") as client:
    result = await client.stk_push(amount=500, phone="254712345678")
```

Or call `.close()` / `await .close()` manually when done.

## Development

```bash
# Install with dev dependencies
pip install -e ".[dev]"

# Run tests
pytest

# Type check
mypy palpluss

# Lint
ruff check palpluss
```

## Design

- **Flat public API** — one obvious way to do each thing
- **Zero magic** — no hidden side effects
- **Typed** — full type annotations, TypedDict responses, ships with `py.typed`
- **Transport isolation** — HTTP concerns in `palpluss.http`
- **No phone normalization** — pass numbers as-is
- **No SDK-side validation** — trust the server

## Links

- [PalPluss website](https://palpluss.com)
- [API documentation](https://docs.palpluss.com)
- [GitHub repository](https://github.com/palpluss/palpluss-sdk)
- [Report an issue](https://github.com/palpluss/palpluss-sdk/issues)
