Metadata-Version: 2.4
Name: wrenai
Version: 0.7.0
Summary: Wren Engine CLI and Python SDK — semantic SQL layer for 20+ data sources
Project-URL: Homepage, https://getwren.ai
Project-URL: Repository, https://github.com/Canner/WrenAI
Project-URL: Issues, https://github.com/Canner/WrenAI/issues
Author-email: Wren AI <contact@getwren.ai>
License: Apache-2.0
Keywords: analytics,cli,data-modeling,database,datafusion,mdl,python,sdk,semantic,semantic-layer,sql,wren,wrenai
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Requires-Python: >=3.11
Requires-Dist: boto3>=1.26
Requires-Dist: duckdb>=1.0
Requires-Dist: ibis-framework>=10
Requires-Dist: loguru>=0.7
Requires-Dist: opendal>=0.45
Requires-Dist: pandas>=2
Requires-Dist: pyarrow-hotfix>=0.6
Requires-Dist: pyarrow>=14
Requires-Dist: pyasn1>=0.6.3
Requires-Dist: pydantic>=2
Requires-Dist: pyopenssl>=26.0.0
Requires-Dist: python-dotenv>=1.0
Requires-Dist: pyyaml>=6.0
Requires-Dist: requests>=2.33.0
Requires-Dist: sqlglot>=27
Requires-Dist: typer>=0.12
Requires-Dist: wren-core-py>=0.1
Provides-Extra: all
Requires-Dist: clickhouse-connect>=0.8; extra == 'all'
Requires-Dist: databricks-sdk; extra == 'all'
Requires-Dist: databricks-sql-connector; extra == 'all'
Requires-Dist: google-auth; extra == 'all'
Requires-Dist: ibis-framework[bigquery]; extra == 'all'
Requires-Dist: inquirerpy>=0.3.4; extra == 'all'
Requires-Dist: jinja2>=3.1; extra == 'all'
Requires-Dist: lancedb>=0.6; extra == 'all'
Requires-Dist: mysqlclient>=2.2; extra == 'all'
Requires-Dist: oracledb>=2; extra == 'all'
Requires-Dist: psycopg[binary]>=3; extra == 'all'
Requires-Dist: pyathena[pandas]>=3; extra == 'all'
Requires-Dist: pyodbc<6,>=5; extra == 'all'
Requires-Dist: pyspark>=3.5; extra == 'all'
Requires-Dist: python-multipart>=0.0.9; extra == 'all'
Requires-Dist: redshift-connector; extra == 'all'
Requires-Dist: sentence-transformers>=2.2; extra == 'all'
Requires-Dist: snowflake-connector-python[pandas]>=3.10; extra == 'all'
Requires-Dist: starlette>=0.37; extra == 'all'
Requires-Dist: trino<1,>=0.333; extra == 'all'
Requires-Dist: uvicorn>=0.29; extra == 'all'
Provides-Extra: athena
Requires-Dist: pyathena[pandas]>=3; extra == 'athena'
Provides-Extra: bigquery
Requires-Dist: google-auth; extra == 'bigquery'
Requires-Dist: ibis-framework[bigquery]; extra == 'bigquery'
Provides-Extra: clickhouse
Requires-Dist: clickhouse-connect>=0.8; extra == 'clickhouse'
Provides-Extra: databricks
Requires-Dist: databricks-sdk; extra == 'databricks'
Requires-Dist: databricks-sql-connector; extra == 'databricks'
Provides-Extra: dev
Requires-Dist: httpx>=0.27; extra == 'dev'
Requires-Dist: orjson>=3; extra == 'dev'
Requires-Dist: pytest>=8; extra == 'dev'
Requires-Dist: ruff>=0.4; extra == 'dev'
Requires-Dist: testcontainers[mysql,postgres]>=4; extra == 'dev'
Provides-Extra: interactive
Requires-Dist: inquirerpy>=0.3.4; extra == 'interactive'
Provides-Extra: main
Requires-Dist: inquirerpy>=0.3.4; extra == 'main'
Requires-Dist: jinja2>=3.1; extra == 'main'
Requires-Dist: python-multipart>=0.0.9; extra == 'main'
Requires-Dist: starlette>=0.37; extra == 'main'
Requires-Dist: uvicorn>=0.29; extra == 'main'
Provides-Extra: memory
Requires-Dist: lancedb>=0.6; extra == 'memory'
Requires-Dist: sentence-transformers>=2.2; extra == 'memory'
Provides-Extra: mssql
Requires-Dist: pyodbc<6,>=5; extra == 'mssql'
Provides-Extra: mysql
Requires-Dist: mysqlclient>=2.2; extra == 'mysql'
Provides-Extra: oracle
Requires-Dist: oracledb>=2; extra == 'oracle'
Provides-Extra: postgres
Requires-Dist: psycopg[binary]>=3; extra == 'postgres'
Provides-Extra: redshift
Requires-Dist: redshift-connector; extra == 'redshift'
Provides-Extra: snowflake
Requires-Dist: snowflake-connector-python[pandas]>=3.10; extra == 'snowflake'
Provides-Extra: spark
Requires-Dist: pyspark>=3.5; extra == 'spark'
Provides-Extra: trino
Requires-Dist: trino<1,>=0.333; extra == 'trino'
Provides-Extra: ui
Requires-Dist: jinja2>=3.1; extra == 'ui'
Requires-Dist: python-multipart>=0.0.9; extra == 'ui'
Requires-Dist: starlette>=0.37; extra == 'ui'
Requires-Dist: uvicorn>=0.29; extra == 'ui'
Description-Content-Type: text/markdown

# wrenai

[![PyPI version](https://img.shields.io/pypi/v/wrenai.svg)](https://pypi.org/project/wrenai/)
[![Python](https://img.shields.io/pypi/pyversions/wrenai.svg)](https://pypi.org/project/wrenai/)
[![License](https://img.shields.io/pypi/l/wrenai.svg)](https://github.com/Canner/WrenAI/blob/main/LICENSE)

Wren AI CLI and Python SDK — semantic SQL layer for 20+ data sources.

Translate natural SQL queries through an [MDL (Modeling Definition Language)](https://docs.getwren.ai/) semantic layer and execute them against your database. Powered by [Apache DataFusion](https://datafusion.apache.org/).

## Installation

```bash
pip install wrenai              # Core (DuckDB included)
pip install wrenai[postgres]    # PostgreSQL
pip install wrenai[mysql]       # MySQL
pip install wrenai[bigquery]    # BigQuery
pip install wrenai[snowflake]   # Snowflake
pip install wrenai[clickhouse]  # ClickHouse
pip install wrenai[trino]       # Trino
pip install wrenai[mssql]       # SQL Server
pip install wrenai[databricks]  # Databricks
pip install wrenai[redshift]    # Redshift
pip install wrenai[spark]       # Spark
pip install wrenai[athena]      # Athena
pip install wrenai[oracle]      # Oracle
pip install 'wrenai[memory]'    # Schema & query memory (LanceDB)
pip install 'wrenai[ui]'        # Browser-based profile form (starlette + uvicorn)
pip install 'wrenai[main]'      # memory + interactive prompts + ui
pip install 'wrenai[all]'       # All connectors + main
```

Requires Python 3.11+.

## Quick start

**1. Initialize a project** — scaffolds a YAML-based MDL project:

```bash
mkdir my-project && cd my-project
wren context init
```

This creates `wren_project.yml`, `models/`, and `views/`. Edit `wren_project.yml` to set your `data_source` and add models under `models/`:

```yaml
# wren_project.yml
schema_version: 2
name: my_project
catalog: wren
schema: public
data_source: postgres
```

```yaml
# models/orders/metadata.yml
name: orders
table_reference:
  schema: mydb
  table: orders
columns:
  - name: order_id
    type: integer
  - name: customer_id
    type: integer
  - name: total
    type: double
  - name: status
    type: varchar
primary_key: order_id
```

> **Already have an MDL JSON?** Import it directly:
> `wren context init --from-mdl path/to/mdl.json`

**2. Configure a connection profile:**

```bash
# Browser form (recommended, requires wrenai[ui])
wren profile add my-db --ui

# Interactive terminal prompts
wren profile add my-db --interactive

# Import from an existing connection file
wren profile add my-db --from-file connection_info.json
```

**3. Build the manifest:**

```bash
wren context build
```

This compiles YAML files into `target/mdl.json`. The CLI auto-discovers this file when you run queries from within the project directory.

**4. Run queries:**

```bash
wren --sql 'SELECT order_id FROM "orders" LIMIT 10'
```

`wren` walks up from the current directory to find `wren_project.yml` and uses `target/mdl.json`. You can also pass `--mdl path/to/mdl.json` explicitly.

For the full CLI reference and per-datasource connection field reference, see [`docs/cli.md`](docs/cli.md) and [`docs/connections.md`](docs/connections.md).

**4a. (Optional) Aggregation queries with cubes** — define cubes under `cubes/`,
then query them with a structured input instead of writing `GROUP BY` SQL by hand:

```bash
wren cube list
wren cube describe order_metrics
wren cube query --cube order_metrics --measures revenue --time-dimension "created_at:month"
```

The translator produces `DATE_TRUNC` / `GROUP BY` / `WHERE` clauses for you and
runs them through the same engine path as `wren --sql`. See the
[Cube guide](../../docs/core/guides/modeling/cube.md) for full YAML structure
and the [CLI reference](../../docs/core/reference/cli.md#wren-cube--pre-aggregation-queries) for all
flags.

**5. (Optional) Configure security policy** — create `~/.wren/config.json`:

```json
{
  "strict_mode": true,
  "denied_functions": ["pg_read_file", "dblink", "lo_import"]
}
```

| Key | Default | Description |
|-----|---------|-------------|
| `strict_mode` | `false` | When `true`, every table in a query must be defined in the MDL. Queries referencing undeclared tables are rejected before execution. |
| `denied_functions` | `[]` | List of function names (case-insensitive) that are forbidden in queries. |

**6. (Optional) Index schema for semantic search** (requires `wrenai[memory]`):

```bash
wren memory index                              # index MDL schema
wren memory fetch -q "customer order price"    # fetch relevant schema context
wren memory store --nl "top customers" --sql "SELECT ..."  # store NL→SQL pair
wren memory recall -q "best customers"         # retrieve similar past queries
```

---

## Connection profiles

Profiles let you store named connection configurations in `~/.wren/profiles.yml` and switch between them easily — useful when working across multiple databases or environments.

```bash
# Add a profile (browser form, interactive prompts, or file import)
wren profile add prod --ui                        # opens http://localhost:<port>
wren profile add staging --interactive            # terminal prompts
wren profile add local --from-file conn.json      # import existing file

# List and switch profiles
wren profile list                                 # * marks the active profile
wren profile switch prod

# Inspect a profile (sensitive fields masked)
wren profile debug prod

# Remove a profile
wren profile rm old-profile --force
```

The `--ui` flag opens a browser-based form that auto-derives fields from each datasource's schema — including file upload for BigQuery credentials, variant selection for Databricks/Redshift, and sensible defaults for all 20+ supported sources. Requires `pip install 'wrenai[ui]'`.

Once a profile is active, `wren` uses it automatically:

```bash
wren profile switch prod
wren --sql 'SELECT COUNT(*) FROM "orders"'        # connects using prod profile
```

---

## Python SDK

```python
import base64, orjson
from wren import WrenEngine, DataSource

manifest = { ... }  # your MDL dict
manifest_str = base64.b64encode(orjson.dumps(manifest)).decode()

with WrenEngine(manifest_str, DataSource.mysql, {"host": "...", ...}) as engine:
    result = engine.query('SELECT * FROM "orders" LIMIT 10')
    print(result.to_pandas())
```

---

## Development

```bash
just install-dev    # Install with dev dependencies
just lint           # Ruff format check + lint
just format         # Auto-fix
```

| Command | What it runs | Docker needed |
|---------|-------------|---------------|
| `just test-unit` | Unit tests (engine, CTE rewriter, field registry, profiles) | No |
| `just test-duckdb` | DuckDB connector tests | No |
| `just test-postgres` | PostgreSQL connector tests | Yes |
| `just test-mysql` | MySQL connector tests | Yes |
| `just test` | All tests | Yes |

Profile web tests (`test_profile_web.py`) require `wrenai[ui]`:

```bash
uv sync --extra dev --extra ui --find-links ../wren-core-py/target/wheels/
uv run pytest tests/test_profile_web.py -v
```

## Publishing

```bash
./scripts/publish.sh            # Build + publish to PyPI
./scripts/publish.sh --test     # Build + publish to TestPyPI
./scripts/publish.sh --build    # Build only
```

## Package rename: `wren-engine` → `wrenai`

Starting with the 0.7.0 release, this PyPI distribution is renamed from
[`wren-engine`](https://pypi.org/project/wren-engine/) to
[`wrenai`](https://pypi.org/project/wrenai/) to align with the **Wren AI**
brand. The legacy `wren-engine` project on PyPI is frozen at 0.6.x and
will not receive further updates.

### What stays the same

- The Python import path: `import wren` (and submodules under `wren.*`)
- The `wren` CLI entrypoint and every subcommand (`wren query`,
  `wren context`, `wren profile`, `wren memory`, …)
- All extras (`postgres`, `mysql`, `bigquery`, …, `memory`, `ui`, `main`,
  `all`)
- Configuration files under `~/.wren/` (profiles, memory, config)

Only the name you type after `pip install` is different.

### Migration

```bash
pip uninstall wren-engine
pip install wrenai                  # or: pip install "wrenai[<extras>]"
wren --version                      # should print: wrenai X.Y.Z
```

If your project pinned `wren-engine` in a `requirements.txt`,
`pyproject.toml`, or lockfile, replace it with `wrenai` and re-lock.

## License

Apache-2.0
