Metadata-Version: 2.4
Name: hqg-algorithms
Version: 1.0.1
Summary: Algorithmic trading research utilities for quant teams
Project-URL: Homepage, https://github.com/Husky-Quantitative-Group/hqg-algorithms/
Requires-Python: >=3.10
Description-Content-Type: text/markdown

# hqg-algorithms

Interfaces and helper types for writing HQG trading strategies.

## Install

```shell
python3 -m pip install --upgrade pip setuptools wheel
pip install hqg-algorithms
```

## Quick start

Subclass `Strategy`, declare your `universe` and `cadence`, and implement `on_data`:

```python
from hqg_algorithms import (
    Strategy, Cadence, Slice, PortfolioView,
    BarSize, ExecutionTiming, Signal, TargetWeights,
)

class BuyAndRebalance(Strategy):
    universe = ["SPY", "IEF"]
    cadence = Cadence(bar_size=BarSize.DAILY, execution=ExecutionTiming.CLOSE_TO_CLOSE)

    def on_data(self, data: Slice, portfolio: PortfolioView) -> Signal:
        return TargetWeights({"SPY": 0.6, "IEF": 0.4})
```

| Declaration | Purpose | Required? |
| -------- | --------- | --------- |
| `universe` | List of ticker strings the platform loads for this strategy | Yes |
| `cadence` | Bar resolution and execution timing (optional - defaults to daily, close-to-close) | No |
| `on_data()` | Return a `Signal`: `TargetWeights(...)`, `Hold()`, or `Liquidate()` | Yes |

### Important constraints

- `universe` **must** be a non-empty list literal of ticker strings (e.g. `["SPY", "IEF"]`). Variables, function calls, and concatenation are not supported. Tickers are normalized to uppercase and deduplicated. Empty strings, whitespace-only tickers, and tickers exceeding 12 characters are rejected. Maximum universe size is 200.
- `cadence` **must** be a direct `Cadence(...)` call with `BarSize.X` and/or `ExecutionTiming.Y` keyword arguments. Positional arguments and unknown keyword arguments are rejected. If omitted, it defaults to `Cadence(bar_size=BarSize.DAILY, execution=ExecutionTiming.CLOSE_TO_CLOSE)`.
- Both `universe` and `cadence` are fixed at class definition and cannot be changed at runtime.

`Slice` maps each symbol to a `Bar` dataclass with typed fields (`open`, `high`, `low`, `close`, `volume`). You can access prices via helpers like `data.close("SPY")`, or grab the full bar with `data.bar("SPY")` for direct attribute access. `PortfolioView` gives read-only access to current equity, cash, positions, and weights.

### Execution timing

`ExecutionTiming` controls when your strategy receives data and when the resulting trades fill. Pick the option that matches your signal logic:

| `ExecutionTiming` | `on_data` fires at | Trades fill at |
| --- | --- | --- |
| `CLOSE_TO_CLOSE` | Bar close | Same bar's close (DEFAULT) |
| `CLOSE_TO_NEXT_OPEN` | Bar close | Next bar's open |

`CLOSE_TO_NEXT_OPEN` is the most realistic for intraday strategies since it avoids look-ahead bias - your signal only uses data that was already available before the trade executes.

### Signal types

`on_data()` returns a `Signal` that tells the backtester what to do:

| Signal | Meaning |
| -------- | --------- |
| `TargetWeights({"SPY": 0.6, "IEF": 0.4})` | Rebalance to these weights. Omitted symbols are sold to zero. Weights summing to less than 1.0 leave the remainder in cash. |
| `Hold()` | Keep the current allocation unchanged. |
| `Liquidate()` | Sell all positions and move fully to cash. |

## Validating and parsing strategies

The `parsing` module provides two functions for working with strategy source code without executing it. Both use AST parsing - no user code is ever run.

### `validate_strategy`

Checks strategy source code and returns a list of error strings. An empty list means the strategy is valid.

```python
from hqg_algorithms import validate_strategy

source = open("my_strategy.py").read()
errors = validate_strategy(source)

if errors:
    for e in errors:
        print(f"❌ {e}")
else:
    print("✅ Strategy is valid")
```

| Check | Rule |
| --- | --- |
| **Syntax** | Source must be parseable Python |
| **Strategy class** | At least one class inheriting from `Strategy` must be present |
| **`universe`** | Must be a non-empty list literal of valid ticker strings |
| **`cadence`** | If present, must be a `Cadence(...)` call with valid keyword args only |
| **`on_data`** | Must be defined as a method on the strategy class |

All errors are blocking - if `validate_strategy` returns anything, the strategy cannot run.

### `get_strategy_metadata`

Extracts `universe` and `cadence` as typed objects. Raises `ValueError` if the source is invalid.

```python
from hqg_algorithms import get_strategy_metadata

meta = get_strategy_metadata(source)
print(meta.universe)   # ["SPY", "IEF"] (normalized, deduplicated)
print(meta.cadence)    # Cadence(bar_size=BarSize.DAILY, execution=ExecutionTiming.CLOSE_TO_CLOSE)
```

Returns a frozen `StrategyMetadata` dataclass:

```python
@dataclass(frozen=True)
class StrategyMetadata:
    universe: list[str]
    cadence: Cadence
```

This is intended for services that need to know what a strategy requires (data subscriptions, scheduling) without loading it.

## Example - SMA crossover

```python
from hqg_algorithms import (
    Strategy, Cadence, Slice, PortfolioView,
    BarSize, ExecutionTiming, Signal, TargetWeights, Hold,
)
from collections import deque


class SimpleSMA(Strategy):
    """Go risk-on when SPY is above its 21-day mean, otherwise hold bonds."""

    universe = ["SPY", "BND"]
    cadence = Cadence(bar_size=BarSize.DAILY, execution=ExecutionTiming.CLOSE_TO_CLOSE)

    def __init__(self):
        self._window = 21
        self._q: deque[float] = deque(maxlen=self._window)

    def on_data(self, data: Slice, portfolio: PortfolioView) -> Signal:
        spy_close = data.close("SPY")
        if spy_close is None:
            return Hold()

        self._q.append(spy_close)

        if len(self._q) < self._window:
            return TargetWeights({"BND": 1.0})  # hold bonds while warming up

        sma = sum(self._q) / len(self._q)

        if spy_close > sma:
            return TargetWeights({"SPY": 0.5, "BND": 0.5})  # uptrend
        return TargetWeights({"BND": 1.0})                   # downtrend
```

## Additional docs

- Publishing workflow and release checklist: [`docs/publishing.md`](docs/publishing.md)
