Metadata-Version: 2.4
Name: sahmk
Version: 0.4.0
Summary: Lightweight Python client for the SAHMK Developer API.
Author-email: Sahmk <developer@sahmk.sa>
License-Expression: MIT
Project-URL: Homepage, https://github.com/sahmk-sa/sahmk-python
Project-URL: Documentation, https://sahmk.sa/developers/docs
Project-URL: Repository, https://github.com/sahmk-sa/sahmk-python
Project-URL: Issues, https://github.com/sahmk-sa/sahmk-python/issues
Keywords: stocks,tadawul,market-data,websocket,api
Classifier: Development Status :: 3 - Alpha
Classifier: Intended Audience :: Developers
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3 :: Only
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: Topic :: Office/Business :: Financial
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: requests>=2.28.0
Requires-Dist: websockets>=12.0
Provides-Extra: dev
Requires-Dist: pytest>=7.0; extra == "dev"
Requires-Dist: pytest-cov>=4.0; extra == "dev"
Requires-Dist: pytest-asyncio>=0.21.0; extra == "dev"
Requires-Dist: responses>=0.23.0; extra == "dev"
Requires-Dist: flake8>=6.0; extra == "dev"
Requires-Dist: black>=23.0; extra == "dev"
Dynamic: license-file

# SAHMK Python SDK

A lightweight Python client for the [SAHMK Developer API](https://sahmk.sa/developers) — real-time and historical Saudi stock market (Tadawul) data.

## Features

- **Real-time quotes** — live prices for 350+ Tadawul stocks
- **Batch quotes** — up to 50 stocks in a single request
- **Historical data** — daily/weekly/monthly OHLCV with custom date ranges
- **Market overview** — TASI index, gainers, losers, volume/value leaders, sectors
- **Company info** — fundamentals, technicals, valuation, analyst consensus (by plan)
- **Financials** — income statements, balance sheets, cash flow
- **Dividends** — history, yield, upcoming payments
- **Events** — AI-generated stock event summaries
- **WebSocket streaming** — real-time price updates pushed to you (Pro+ plan)

## Installation

```bash
pip install sahmk
```

Or clone this repo for local development:

```bash
git clone https://github.com/sahmk-sa/sahmk-python.git
cd sahmk-python
pip install -r requirements.txt
```

## Quick Start

```python
from sahmk import SahmkClient

client = SahmkClient("your_api_key")

# Get a stock quote
quote = client.quote("2222")
print(f"{quote['name_en']}: {quote['price']} SAR ({quote['change_percent']}%)")

# Get market summary
market = client.market_summary()
print(f"TASI: {market['index_value']} ({market['index_change_percent']}%)")

# Batch quotes (Starter+ plan)
result = client.quotes(["2222", "1120", "4191"])
for q in result["quotes"]:
    print(f"{q['symbol']}: {q['price']}")
```

## CLI Quick Start

The package also installs a CLI for instant testing:

```bash
export SAHMK_API_KEY="your_api_key"
sahmk quote 2222
sahmk market gainers --limit 5
sahmk historical 2222 --from 2026-01-01 --to 2026-01-28
```

You can also pass the key directly:

```bash
sahmk quote 2222 --api-key your_api_key
```

## Retries and Rate Limits

The client automatically retries transient failures (429 rate-limit and 5xx server errors) with exponential backoff:

```python
# Defaults: 3 retries, 0.5s backoff factor (0.5s, 1s, 2s delays)
client = SahmkClient("your_api_key")

# Customize retry behavior
client = SahmkClient("your_api_key", retries=5, backoff_factor=1.0)

# Disable retries entirely
client = SahmkClient("your_api_key", retries=0)
```

Rate limit errors include metadata from the API:

```python
from sahmk import SahmkRateLimitError

try:
    quote = client.quote("2222")
except SahmkRateLimitError as e:
    print(f"Rate limited. Retry after: {e.retry_after}s")
    print(f"Remaining: {e.rate_remaining}/{e.rate_limit}")
```

## Get Your API Key

1. Sign up at [sahmk.sa/developers](https://sahmk.sa/developers)
2. Verify your email
3. Go to Dashboard → API Keys → Create Key
4. Copy your key (starts with `shmk_live_` or `shmk_test_`)

## Plans

| Plan | Price | Requests/Day | WebSocket |
|------|-------|-------------|-----------|
| Free | 0 SAR | 100 | - |
| Starter | 149 SAR/mo | 5,000 | - |
| Pro | 499 SAR/mo | 50,000 | Yes |
| Enterprise | [`Contact us`](https://www.sahmk.sa/contactus?type=api-support) | Unlimited | Yes |

## Examples

Check the [`examples/`](examples/) directory:

| File | Description |
|------|-------------|
| [`quote.py`](examples/quote.py) | Get a single stock quote |
| [`batch_quotes.py`](examples/batch_quotes.py) | Fetch multiple stocks at once |
| [`historical.py`](examples/historical.py) | Historical price data |
| [`market_summary.py`](examples/market_summary.py) | Market overview and movers |
| [`websocket_stream.py`](examples/websocket_stream.py) | Real-time WebSocket streaming (Pro+) |

## API Reference

Base URL: `https://app.sahmk.sa/api/v1`

| Endpoint | Plan | Description |
|----------|------|-------------|
| `GET /quote/{symbol}/` | Free | Stock quote |
| `GET /quotes/?symbols=...` | Starter+ | Batch quotes (up to 50) |
| `GET /historical/{symbol}/` | Starter+ | Historical OHLCV data |
| `GET /market/summary/` | Free | Market overview & TASI index |
| `GET /market/gainers/` | Free | Top gainers |
| `GET /market/losers/` | Free | Top losers |
| `GET /market/volume/` | Free | Volume leaders |
| `GET /market/value/` | Free | Value leaders |
| `GET /market/sectors/` | Free | Sector performance |
| `GET /company/{symbol}/` | Free+ | Company info (tiered by plan) |
| `GET /financials/{symbol}/` | Starter+ | Financial statements |
| `GET /dividends/{symbol}/` | Starter+ | Dividend history & yield |
| `GET /events/` | Pro+ | AI-generated stock events |

All endpoints require the `X-API-Key` header.

Full docs: [sahmk.sa/developers/docs](https://sahmk.sa/developers/docs)

Changelog: [CHANGELOG.md](CHANGELOG.md)

Roadmap: [ROADMAP.md](ROADMAP.md)

## WebSocket Streaming (Pro+)

```python
import asyncio
from sahmk import SahmkClient

client = SahmkClient("your_api_key")

async def on_quote(msg):
    symbol = msg["symbol"]
    price = msg["data"]["price"]
    print(f"{symbol}: {price}")

asyncio.run(client.stream(["2222", "1120"], on_quote=on_quote))
```

### Auto-Reconnect

The streaming client automatically reconnects on disconnect with exponential backoff. All symbols are resubscribed after reconnection.

```python
async def on_disconnect(reason):
    print(f"Disconnected: {reason}")

async def on_reconnect(attempt):
    print(f"Reconnecting (attempt #{attempt})...")

await client.stream(
    ["2222", "1120"],
    on_quote=on_quote,
    on_disconnect=on_disconnect,
    on_reconnect=on_reconnect,
    # max_reconnect_attempts=0 means unlimited (default)
    # Set to -1 to disable reconnect entirely
)
```

Connection URL: `wss://app.sahmk.sa/ws/v1/stocks/?api_key=YOUR_KEY`

## License

MIT — see [LICENSE](LICENSE)
