Metadata-Version: 2.4
Name: sortai
Version: 0.1.4
Summary: LLM-powered directory organizer using Google Gemini
Author: sortai
License: MIT
Keywords: cli,organize,files,gemini,llm
Classifier: Development Status :: 4 - Beta
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
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
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: click>=8.0
Requires-Dist: google-genai>=0.2.0
Requires-Dist: pdfplumber>=0.10.0
Requires-Dist: python-docx>=1.0
Dynamic: license-file

# sortai

[![GitHub release (latest by date)](https://img.shields.io/github/v/release/ajs2583/sortai?label=release)](https://github.com/ajs2583/sortai/releases/latest)

LLM-powered directory organizer. Uses **Google Gemini** to suggest a folder structure from filenames and (for text-based files) the first ~500 characters of content, then moves files into the suggested subfolders.

- **Dry-run by default** – see exactly what would move where before touching anything.
- **Confirm before apply** – with `--apply`, you are prompted to confirm before any files are moved.

📦 **PyPI Package:** [https://pypi.org/project/sortai/0.1.4/](https://pypi.org/project/sortai/0.1.4/)  
🔄 **Releases:** [Latest](https://github.com/ajs2583/sortai/releases/latest) · [All releases](https://github.com/ajs2583/sortai/releases)

## Install

Install from PyPI:

```bash
pip install sortai
```

Or view the package on [PyPI](https://pypi.org/project/sortai/0.1.0/).

Development install from source:

```bash
git clone https://github.com/ajs2583/sortai.git
cd sortai
pip install -e .
```

## Setup

Set your Google Gemini API key (required):

```bash
export GEMINI_API_KEY=your_key_here
```

Get a key at: **https://aistudio.google.com/app/apikey**

You can copy `.env.example` to `.env` and set `GEMINI_API_KEY` there; load it with your shell or a tool like `python-dotenv` if you use one (sortai does not load `.env` automatically).

## Demo

![Demo](https://raw.githubusercontent.com/ajs2583/sortai/main/docs/demo.gif)

*Demo showing `sortai test-demo` dry-run preview, then `--apply` with confirmation.*

## Usage

| Command | Description |
|--------|-------------|
| `sortai <path>` | Dry-run: show what would be moved where (default). |
| `sortai <path> --apply` | After dry-run, prompt and then actually move files. |
| `sortai <path> --depth 2` | Organize up to 2 levels of subfolders (e.g. `documents/work`). |
| `sortai <path> --model gemini-2.5-flash` | Override Gemini model (default: gemini-2.5-flash). |
| `sortai --version` | Print version. |
| `sortai --help` | Show help. |

### Example output

**Before (flat directory):**

```
my-folder/
├── report.pdf
├── notes.txt
├── budget.csv
├── vacation.jpg
└── readme.md
```

**Dry-run:**

```
$ sortai ./my-folder
Dry run – would move:
  report.pdf  ->  documents/
  notes.txt   ->  documents/
  budget.csv  ->  finance/
  vacation.jpg ->  images/
  readme.md   ->  (keep at root)
Run with --apply to perform moves.
```

**After applying:**

```
my-folder/
├── readme.md
├── documents/
│   ├── report.pdf
│   └── notes.txt
├── finance/
│   └── budget.csv
└── images/
    └── vacation.jpg
```

## Supported file types for content reading

sortai reads the **first ~500 characters** of content for:

- `.pdf` (first page via pdfplumber)
- `.txt`, `.md`, `.csv` (plain text)
- `.docx` (paragraph text via python-docx)

All other files are categorized by **filename and extension only**.

## Releasing

### GitHub Releases (Automated)

The release workflow (`.github/workflows/release.yml`) runs when you **push a version tag**. It builds the package and creates a GitHub Release with artifacts. Follow these steps exactly.

1. **Bump version in both places** (must match exactly):
   - `pyproject.toml` → `version = "0.1.4"` (example)
   - `sortai/__init__.py` → `__version__ = "0.1.4"`  
   If they don’t match the tag, the workflow will fail at “Verify version consistency”.

2. **Commit and push**:
   ```bash
   git add pyproject.toml sortai/__init__.py
   git commit -m "Bump version to 0.1.4"
   git push origin main
   ```
   (Use your default branch name if different.)

3. **Create the tag and push it** (this triggers the workflow):
   ```bash
   git tag v0.1.4
   git push origin v0.1.4
   ```
   The tag must match the version: `v` + version (e.g. `v0.1.4` for `0.1.4`).

4. **Check the run**:
   - Open the repo on GitHub → **Actions** tab.
   - The “Release” workflow should run. When it finishes, **Releases** will have a new release with the wheel and sdist attached.

**If the release doesn’t run or fails:**

- **Workflow didn’t run:** The workflow only runs on **tag** push. Pushing a branch alone does not trigger it. Run `git push origin v0.1.4` (or your tag) after creating the tag.
- **“Verify version consistency” failed:** The tag (e.g. `v0.1.4`), `version` in `pyproject.toml`, and `__version__` in `sortai/__init__.py` must all be the same (aside from the leading `v`). Fix the files, commit, delete the tag locally (`git tag -d v0.1.4`), recreate it, and force-push the tag (`git push origin :refs/tags/v0.1.4` then `git push origin v0.1.4`).
- **Manual run from Actions:** You can “Run workflow” and enter a version (e.g. `v0.1.4`). That run will check out **that tag**, so the tag must already exist and be pushed. Use this to re-run a release, not to create the first time.

### Publishing to PyPI

1. **Create a PyPI account** (and optionally [Test PyPI](https://test.pypi.org/) for testing):
   - https://pypi.org/account/register/

2. **Install build tools** (one-time):
   ```bash
   pip install build twine
   ```

3. **Bump version** in `pyproject.toml` and `sortai/__init__.py` when releasing a new version.

4. **Build the package** (from the project root):
   ```bash
   python -m build
   ```
   This creates `dist/sortai-0.1.0.tar.gz` and a wheel.

5. **Upload to PyPI** (manual):
   ```bash
   twine upload dist/*
   ```
   Twine will prompt for your PyPI username and password. Prefer an [API token](https://pypi.org/manage/account/token/) (username: `__token__`, password: your token) over your account password.

   **Or enable automated PyPI upload**: Add your PyPI API token as a GitHub secret named `PYPI_API_TOKEN`, then edit `.github/workflows/release.yml` and change `if: false` to `if: true` in the "Upload to PyPI" step. Releases will then automatically publish to PyPI.

   To try Test PyPI first:
   ```bash
   twine upload --repository testpypi dist/*
   ```
   Then install with: `pip install -i https://test.pypi.org/simple/ sortai`


## License

MIT



