Compare commits
1 Commits
dynamic-ad
...
feature/en
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d2299d2679 |
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
paper_plot/data/big_graph_degree_data.npz filter=lfs diff=lfs merge=lfs -text
|
||||||
50
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
50
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
@@ -1,50 +0,0 @@
|
|||||||
name: Bug Report
|
|
||||||
description: Report a bug in LEANN
|
|
||||||
labels: ["bug"]
|
|
||||||
|
|
||||||
body:
|
|
||||||
- type: textarea
|
|
||||||
id: description
|
|
||||||
attributes:
|
|
||||||
label: What happened?
|
|
||||||
description: A clear description of the bug
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
id: reproduce
|
|
||||||
attributes:
|
|
||||||
label: How to reproduce
|
|
||||||
placeholder: |
|
|
||||||
1. Install with...
|
|
||||||
2. Run command...
|
|
||||||
3. See error
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
id: error
|
|
||||||
attributes:
|
|
||||||
label: Error message
|
|
||||||
description: Paste any error messages
|
|
||||||
render: shell
|
|
||||||
|
|
||||||
- type: input
|
|
||||||
id: version
|
|
||||||
attributes:
|
|
||||||
label: LEANN Version
|
|
||||||
placeholder: "0.1.0"
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: dropdown
|
|
||||||
id: os
|
|
||||||
attributes:
|
|
||||||
label: Operating System
|
|
||||||
options:
|
|
||||||
- macOS
|
|
||||||
- Linux
|
|
||||||
- Windows
|
|
||||||
- Docker
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
8
.github/ISSUE_TEMPLATE/config.yml
vendored
8
.github/ISSUE_TEMPLATE/config.yml
vendored
@@ -1,8 +0,0 @@
|
|||||||
blank_issues_enabled: true
|
|
||||||
contact_links:
|
|
||||||
- name: Documentation
|
|
||||||
url: https://github.com/LEANN-RAG/LEANN-RAG/tree/main/docs
|
|
||||||
about: Read the docs first
|
|
||||||
- name: Discussions
|
|
||||||
url: https://github.com/LEANN-RAG/LEANN-RAG/discussions
|
|
||||||
about: Ask questions and share ideas
|
|
||||||
27
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
27
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
@@ -1,27 +0,0 @@
|
|||||||
name: Feature Request
|
|
||||||
description: Suggest a new feature for LEANN
|
|
||||||
labels: ["enhancement"]
|
|
||||||
|
|
||||||
body:
|
|
||||||
- type: textarea
|
|
||||||
id: problem
|
|
||||||
attributes:
|
|
||||||
label: What problem does this solve?
|
|
||||||
description: Describe the problem or need
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
id: solution
|
|
||||||
attributes:
|
|
||||||
label: Proposed solution
|
|
||||||
description: How would you like this to work?
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
id: example
|
|
||||||
attributes:
|
|
||||||
label: Example usage
|
|
||||||
description: Show how the API might look
|
|
||||||
render: python
|
|
||||||
13
.github/pull_request_template.md
vendored
13
.github/pull_request_template.md
vendored
@@ -1,13 +0,0 @@
|
|||||||
## What does this PR do?
|
|
||||||
|
|
||||||
<!-- Brief description of your changes -->
|
|
||||||
|
|
||||||
## Related Issues
|
|
||||||
|
|
||||||
Fixes #
|
|
||||||
|
|
||||||
## Checklist
|
|
||||||
|
|
||||||
- [ ] Tests pass (`uv run pytest`)
|
|
||||||
- [ ] Code formatted (`ruff format` and `ruff check`)
|
|
||||||
- [ ] Pre-commit hooks pass (`pre-commit run --all-files`)
|
|
||||||
114
.github/workflows/build-reusable.yml
vendored
114
.github/workflows/build-reusable.yml
vendored
@@ -54,17 +54,6 @@ jobs:
|
|||||||
python: '3.12'
|
python: '3.12'
|
||||||
- os: ubuntu-22.04
|
- os: ubuntu-22.04
|
||||||
python: '3.13'
|
python: '3.13'
|
||||||
# ARM64 Linux builds
|
|
||||||
- os: ubuntu-24.04-arm
|
|
||||||
python: '3.9'
|
|
||||||
- os: ubuntu-24.04-arm
|
|
||||||
python: '3.10'
|
|
||||||
- os: ubuntu-24.04-arm
|
|
||||||
python: '3.11'
|
|
||||||
- os: ubuntu-24.04-arm
|
|
||||||
python: '3.12'
|
|
||||||
- os: ubuntu-24.04-arm
|
|
||||||
python: '3.13'
|
|
||||||
- os: macos-14
|
- os: macos-14
|
||||||
python: '3.9'
|
python: '3.9'
|
||||||
- os: macos-14
|
- os: macos-14
|
||||||
@@ -98,7 +87,7 @@ jobs:
|
|||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v5
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: ${{ inputs.ref }}
|
ref: ${{ inputs.ref }}
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
@@ -109,56 +98,21 @@ jobs:
|
|||||||
python-version: ${{ matrix.python }}
|
python-version: ${{ matrix.python }}
|
||||||
|
|
||||||
- name: Install uv
|
- name: Install uv
|
||||||
uses: astral-sh/setup-uv@v6
|
uses: astral-sh/setup-uv@v4
|
||||||
|
|
||||||
- name: Install system dependencies (Ubuntu)
|
- name: Install system dependencies (Ubuntu)
|
||||||
if: runner.os == 'Linux'
|
if: runner.os == 'Linux'
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install -y libomp-dev libboost-all-dev protobuf-compiler libzmq3-dev \
|
sudo apt-get install -y libomp-dev libboost-all-dev protobuf-compiler libzmq3-dev \
|
||||||
pkg-config libabsl-dev libaio-dev libprotobuf-dev \
|
pkg-config libopenblas-dev patchelf libabsl-dev libaio-dev libprotobuf-dev
|
||||||
patchelf
|
|
||||||
|
|
||||||
# Debug: Show system information
|
# Install Intel MKL for DiskANN
|
||||||
echo "🔍 System Information:"
|
wget -q https://registrationcenter-download.intel.com/akdlm/IRC_NAS/79153e0f-74d7-45af-b8c2-258941adf58a/intel-onemkl-2025.0.0.940.sh
|
||||||
echo "Architecture: $(uname -m)"
|
sudo sh intel-onemkl-2025.0.0.940.sh -a --components intel.oneapi.lin.mkl.devel --action install --eula accept -s
|
||||||
echo "OS: $(uname -a)"
|
source /opt/intel/oneapi/setvars.sh
|
||||||
echo "CPU info: $(lscpu | head -5)"
|
echo "MKLROOT=/opt/intel/oneapi/mkl/latest" >> $GITHUB_ENV
|
||||||
|
echo "LD_LIBRARY_PATH=/opt/intel/oneapi/mkl/latest/lib/intel64:$LD_LIBRARY_PATH" >> $GITHUB_ENV
|
||||||
# Install math library based on architecture
|
|
||||||
ARCH=$(uname -m)
|
|
||||||
echo "🔍 Setting up math library for architecture: $ARCH"
|
|
||||||
|
|
||||||
if [[ "$ARCH" == "x86_64" ]]; then
|
|
||||||
# Install Intel MKL for DiskANN on x86_64
|
|
||||||
echo "📦 Installing Intel MKL for x86_64..."
|
|
||||||
wget -q https://registrationcenter-download.intel.com/akdlm/IRC_NAS/79153e0f-74d7-45af-b8c2-258941adf58a/intel-onemkl-2025.0.0.940.sh
|
|
||||||
sudo sh intel-onemkl-2025.0.0.940.sh -a --components intel.oneapi.lin.mkl.devel --action install --eula accept -s
|
|
||||||
source /opt/intel/oneapi/setvars.sh
|
|
||||||
echo "MKLROOT=/opt/intel/oneapi/mkl/latest" >> $GITHUB_ENV
|
|
||||||
echo "LD_LIBRARY_PATH=/opt/intel/oneapi/compiler/latest/linux/compiler/lib/intel64_lin" >> $GITHUB_ENV
|
|
||||||
echo "LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/opt/intel/oneapi/mkl/latest/lib/intel64" >> $GITHUB_ENV
|
|
||||||
echo "✅ Intel MKL installed for x86_64"
|
|
||||||
|
|
||||||
# Debug: Check MKL installation
|
|
||||||
echo "🔍 MKL Installation Check:"
|
|
||||||
ls -la /opt/intel/oneapi/mkl/latest/ || echo "MKL directory not found"
|
|
||||||
ls -la /opt/intel/oneapi/mkl/latest/lib/ || echo "MKL lib directory not found"
|
|
||||||
|
|
||||||
elif [[ "$ARCH" == "aarch64" ]]; then
|
|
||||||
# Use OpenBLAS for ARM64 (MKL installer not compatible with ARM64)
|
|
||||||
echo "📦 Installing OpenBLAS for ARM64..."
|
|
||||||
sudo apt-get install -y libopenblas-dev liblapack-dev liblapacke-dev
|
|
||||||
echo "✅ OpenBLAS installed for ARM64"
|
|
||||||
|
|
||||||
# Debug: Check OpenBLAS installation
|
|
||||||
echo "🔍 OpenBLAS Installation Check:"
|
|
||||||
dpkg -l | grep openblas || echo "OpenBLAS package not found"
|
|
||||||
ls -la /usr/lib/aarch64-linux-gnu/openblas/ || echo "OpenBLAS directory not found"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Debug: Show final library paths
|
|
||||||
echo "🔍 Final LD_LIBRARY_PATH: $LD_LIBRARY_PATH"
|
|
||||||
|
|
||||||
- name: Install system dependencies (macOS)
|
- name: Install system dependencies (macOS)
|
||||||
if: runner.os == 'macOS'
|
if: runner.os == 'macOS'
|
||||||
@@ -350,53 +304,3 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
name: packages-${{ matrix.os }}-py${{ matrix.python }}
|
name: packages-${{ matrix.os }}-py${{ matrix.python }}
|
||||||
path: packages/*/dist/
|
path: packages/*/dist/
|
||||||
|
|
||||||
|
|
||||||
arch-smoke:
|
|
||||||
name: Arch Linux smoke test (install & import)
|
|
||||||
needs: build
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
container:
|
|
||||||
image: archlinux:latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Prepare system
|
|
||||||
run: |
|
|
||||||
pacman -Syu --noconfirm
|
|
||||||
pacman -S --noconfirm python python-pip gcc git zlib openssl
|
|
||||||
|
|
||||||
- name: Download ALL wheel artifacts from this run
|
|
||||||
uses: actions/download-artifact@v5
|
|
||||||
with:
|
|
||||||
# Don't specify name, download all artifacts
|
|
||||||
path: ./wheels
|
|
||||||
|
|
||||||
- name: Install uv
|
|
||||||
uses: astral-sh/setup-uv@v6
|
|
||||||
|
|
||||||
- name: Create virtual environment and install wheels
|
|
||||||
run: |
|
|
||||||
uv venv
|
|
||||||
source .venv/bin/activate || source .venv/Scripts/activate
|
|
||||||
uv pip install --find-links wheels leann-core
|
|
||||||
uv pip install --find-links wheels leann-backend-hnsw
|
|
||||||
uv pip install --find-links wheels leann-backend-diskann
|
|
||||||
uv pip install --find-links wheels leann
|
|
||||||
|
|
||||||
- name: Import & tiny runtime check
|
|
||||||
env:
|
|
||||||
OMP_NUM_THREADS: 1
|
|
||||||
MKL_NUM_THREADS: 1
|
|
||||||
run: |
|
|
||||||
source .venv/bin/activate || source .venv/Scripts/activate
|
|
||||||
python - <<'PY'
|
|
||||||
import leann
|
|
||||||
import leann_backend_hnsw as h
|
|
||||||
import leann_backend_diskann as d
|
|
||||||
from leann import LeannBuilder, LeannSearcher
|
|
||||||
b = LeannBuilder(backend_name="hnsw")
|
|
||||||
b.add_text("hello arch")
|
|
||||||
b.build_index("arch_demo.leann")
|
|
||||||
s = LeannSearcher("arch_demo.leann")
|
|
||||||
print("search:", s.search("hello", top_k=1))
|
|
||||||
PY
|
|
||||||
|
|||||||
2
.github/workflows/link-check.yml
vendored
2
.github/workflows/link-check.yml
vendored
@@ -14,6 +14,6 @@ jobs:
|
|||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: lycheeverse/lychee-action@v2
|
- uses: lycheeverse/lychee-action@v2
|
||||||
with:
|
with:
|
||||||
args: --no-progress --insecure --user-agent 'curl/7.68.0' README.md docs/ apps/ examples/ benchmarks/
|
args: --no-progress --insecure README.md docs/ apps/ examples/ benchmarks/
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|||||||
10
.gitignore
vendored
10
.gitignore
vendored
@@ -18,11 +18,9 @@ demo/experiment_results/**/*.json
|
|||||||
*.eml
|
*.eml
|
||||||
*.emlx
|
*.emlx
|
||||||
*.json
|
*.json
|
||||||
!.vscode/*.json
|
|
||||||
*.sh
|
*.sh
|
||||||
*.txt
|
*.txt
|
||||||
!CMakeLists.txt
|
!CMakeLists.txt
|
||||||
!llms.txt
|
|
||||||
latency_breakdown*.json
|
latency_breakdown*.json
|
||||||
experiment_results/eval_results/diskann/*.json
|
experiment_results/eval_results/diskann/*.json
|
||||||
aws/
|
aws/
|
||||||
@@ -94,11 +92,3 @@ packages/leann-backend-diskann/third_party/DiskANN/_deps/
|
|||||||
batchtest.py
|
batchtest.py
|
||||||
tests/__pytest_cache__/
|
tests/__pytest_cache__/
|
||||||
tests/__pycache__/
|
tests/__pycache__/
|
||||||
paru-bin/
|
|
||||||
|
|
||||||
CLAUDE.md
|
|
||||||
CLAUDE.local.md
|
|
||||||
.claude/*.local.*
|
|
||||||
.claude/local/*
|
|
||||||
benchmarks/data/
|
|
||||||
test_add/*
|
|
||||||
|
|||||||
3
.gitmodules
vendored
3
.gitmodules
vendored
@@ -14,6 +14,3 @@
|
|||||||
[submodule "packages/leann-backend-hnsw/third_party/libzmq"]
|
[submodule "packages/leann-backend-hnsw/third_party/libzmq"]
|
||||||
path = packages/leann-backend-hnsw/third_party/libzmq
|
path = packages/leann-backend-hnsw/third_party/libzmq
|
||||||
url = https://github.com/zeromq/libzmq.git
|
url = https://github.com/zeromq/libzmq.git
|
||||||
[submodule "packages/astchunk-leann"]
|
|
||||||
path = packages/astchunk-leann
|
|
||||||
url = https://github.com/yichuan-w/astchunk-leann.git
|
|
||||||
|
|||||||
@@ -13,5 +13,4 @@ repos:
|
|||||||
rev: v0.12.7 # Fixed version to match pyproject.toml
|
rev: v0.12.7 # Fixed version to match pyproject.toml
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff
|
- id: ruff
|
||||||
args: [--fix, --exit-non-zero-on-fix]
|
|
||||||
- id: ruff-format
|
- id: ruff-format
|
||||||
|
|||||||
5
.vscode/extensions.json
vendored
5
.vscode/extensions.json
vendored
@@ -1,5 +0,0 @@
|
|||||||
{
|
|
||||||
"recommendations": [
|
|
||||||
"charliermarsh.ruff",
|
|
||||||
]
|
|
||||||
}
|
|
||||||
22
.vscode/settings.json
vendored
22
.vscode/settings.json
vendored
@@ -1,22 +0,0 @@
|
|||||||
{
|
|
||||||
"python.defaultInterpreterPath": ".venv/bin/python",
|
|
||||||
"python.terminal.activateEnvironment": true,
|
|
||||||
"[python]": {
|
|
||||||
"editor.defaultFormatter": "charliermarsh.ruff",
|
|
||||||
"editor.formatOnSave": true,
|
|
||||||
"editor.codeActionsOnSave": {
|
|
||||||
"source.organizeImports": "explicit",
|
|
||||||
"source.fixAll": "explicit"
|
|
||||||
},
|
|
||||||
"editor.insertSpaces": true,
|
|
||||||
"editor.tabSize": 4
|
|
||||||
},
|
|
||||||
"ruff.enable": true,
|
|
||||||
"files.watcherExclude": {
|
|
||||||
"**/.venv/**": true,
|
|
||||||
"**/__pycache__/**": true,
|
|
||||||
"**/*.egg-info/**": true,
|
|
||||||
"**/build/**": true,
|
|
||||||
"**/dist/**": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
132
README.md
132
README.md
@@ -8,8 +8,6 @@
|
|||||||
<img src="https://img.shields.io/badge/Platform-Ubuntu%20%26%20Arch%20%26%20WSL%20%7C%20macOS%20(ARM64%2FIntel)-lightgrey" alt="Platform">
|
<img src="https://img.shields.io/badge/Platform-Ubuntu%20%26%20Arch%20%26%20WSL%20%7C%20macOS%20(ARM64%2FIntel)-lightgrey" alt="Platform">
|
||||||
<img src="https://img.shields.io/badge/License-MIT-green.svg" alt="MIT License">
|
<img src="https://img.shields.io/badge/License-MIT-green.svg" alt="MIT License">
|
||||||
<img src="https://img.shields.io/badge/MCP-Native%20Integration-blue" alt="MCP Integration">
|
<img src="https://img.shields.io/badge/MCP-Native%20Integration-blue" alt="MCP Integration">
|
||||||
<a href="https://join.slack.com/t/leann-e2u9779/shared_invite/zt-3ckd2f6w1-OX08~NN4gkWhh10PRVBj1Q"><img src="https://img.shields.io/badge/Slack-Join-4A154B?logo=slack&logoColor=white" alt="Join Slack">
|
|
||||||
<a href="assets/wechat_user_group.JPG" title="Join WeChat group"><img src="https://img.shields.io/badge/WeChat-Join-2DC100?logo=wechat&logoColor=white" alt="Join WeChat group"></a>
|
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<h2 align="center" tabindex="-1" class="heading-element" dir="auto">
|
<h2 align="center" tabindex="-1" class="heading-element" dir="auto">
|
||||||
@@ -89,60 +87,17 @@ git submodule update --init --recursive
|
|||||||
```
|
```
|
||||||
|
|
||||||
**macOS:**
|
**macOS:**
|
||||||
|
|
||||||
Note: DiskANN requires MacOS 13.3 or later.
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
brew install libomp boost protobuf zeromq pkgconf
|
brew install llvm libomp boost protobuf zeromq pkgconf
|
||||||
uv sync --extra diskann
|
CC=$(brew --prefix llvm)/bin/clang CXX=$(brew --prefix llvm)/bin/clang++ uv sync
|
||||||
```
|
```
|
||||||
|
|
||||||
**Linux (Ubuntu/Debian):**
|
**Linux:**
|
||||||
|
|
||||||
Note: On Ubuntu 20.04, you may need to build a newer Abseil and pin Protobuf (e.g., v3.20.x) for building DiskANN. See [Issue #30](https://github.com/yichuan-w/LEANN/issues/30) for a step-by-step note.
|
|
||||||
|
|
||||||
You can manually install [Intel oneAPI MKL](https://www.intel.com/content/www/us/en/developer/tools/oneapi/onemkl.html) instead of `libmkl-full-dev` for DiskANN. You can also use `libopenblas-dev` for building HNSW only, by removing `--extra diskann` in the command below.
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sudo apt-get update && sudo apt-get install -y \
|
# Ubuntu/Debian (For Arch Linux: sudo pacman -S blas lapack openblas libaio boost protobuf abseil-cpp zeromq)
|
||||||
libomp-dev libboost-all-dev protobuf-compiler libzmq3-dev \
|
sudo apt-get update && sudo apt-get install -y libomp-dev libboost-all-dev protobuf-compiler libabsl-dev libmkl-full-dev libaio-dev libzmq3-dev
|
||||||
pkg-config libabsl-dev libaio-dev libprotobuf-dev \
|
|
||||||
libmkl-full-dev
|
|
||||||
|
|
||||||
uv sync --extra diskann
|
uv sync
|
||||||
```
|
|
||||||
|
|
||||||
**Linux (Arch Linux):**
|
|
||||||
|
|
||||||
```bash
|
|
||||||
sudo pacman -Syu && sudo pacman -S --needed base-devel cmake pkgconf git gcc \
|
|
||||||
boost boost-libs protobuf abseil-cpp libaio zeromq
|
|
||||||
|
|
||||||
# For MKL in DiskANN
|
|
||||||
sudo pacman -S --needed base-devel git
|
|
||||||
git clone https://aur.archlinux.org/paru-bin.git
|
|
||||||
cd paru-bin && makepkg -si
|
|
||||||
paru -S intel-oneapi-mkl intel-oneapi-compiler
|
|
||||||
source /opt/intel/oneapi/setvars.sh
|
|
||||||
|
|
||||||
uv sync --extra diskann
|
|
||||||
```
|
|
||||||
|
|
||||||
**Linux (RHEL / CentOS Stream / Oracle / Rocky / AlmaLinux):**
|
|
||||||
|
|
||||||
See [Issue #50](https://github.com/yichuan-w/LEANN/issues/50) for more details.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
sudo dnf groupinstall -y "Development Tools"
|
|
||||||
sudo dnf install -y libomp-devel boost-devel protobuf-compiler protobuf-devel \
|
|
||||||
abseil-cpp-devel libaio-devel zeromq-devel pkgconf-pkg-config
|
|
||||||
|
|
||||||
# For MKL in DiskANN
|
|
||||||
sudo dnf install -y intel-oneapi-mkl intel-oneapi-mkl-devel \
|
|
||||||
intel-oneapi-openmp || sudo dnf install -y intel-oneapi-compiler
|
|
||||||
source /opt/intel/oneapi/setvars.sh
|
|
||||||
|
|
||||||
uv sync --extra diskann
|
|
||||||
```
|
```
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
@@ -178,8 +133,6 @@ response = chat.ask("How much storage does LEANN save?", top_k=1)
|
|||||||
|
|
||||||
LEANN supports RAG on various data sources including documents (`.pdf`, `.txt`, `.md`), Apple Mail, Google Search History, WeChat, and more.
|
LEANN supports RAG on various data sources including documents (`.pdf`, `.txt`, `.md`), Apple Mail, Google Search History, WeChat, and more.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Generation Model Setup
|
### Generation Model Setup
|
||||||
|
|
||||||
LEANN supports multiple LLM providers for text generation (OpenAI API, HuggingFace, Ollama).
|
LEANN supports multiple LLM providers for text generation (OpenAI API, HuggingFace, Ollama).
|
||||||
@@ -222,8 +175,7 @@ ollama pull llama3.2:1b
|
|||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
### ⭐ Flexible Configuration
|
||||||
## ⭐ Flexible Configuration
|
|
||||||
|
|
||||||
LEANN provides flexible parameters for embedding models, search strategies, and data processing to fit your specific needs.
|
LEANN provides flexible parameters for embedding models, search strategies, and data processing to fit your specific needs.
|
||||||
|
|
||||||
@@ -299,12 +251,6 @@ python -m apps.document_rag --data-dir "~/Documents/Papers" --chunk-size 1024
|
|||||||
|
|
||||||
# Filter only markdown and Python files with smaller chunks
|
# Filter only markdown and Python files with smaller chunks
|
||||||
python -m apps.document_rag --data-dir "./docs" --chunk-size 256 --file-types .md .py
|
python -m apps.document_rag --data-dir "./docs" --chunk-size 256 --file-types .md .py
|
||||||
|
|
||||||
# Enable AST-aware chunking for code files
|
|
||||||
python -m apps.document_rag --enable-code-chunking --data-dir "./my_project"
|
|
||||||
|
|
||||||
# Or use the specialized code RAG for better code understanding
|
|
||||||
python -m apps.code_rag --repo-dir "./my_codebase" --query "How does authentication work?"
|
|
||||||
```
|
```
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
@@ -479,20 +425,10 @@ Once the index is built, you can ask questions like:
|
|||||||
|
|
||||||
### 🚀 Claude Code Integration: Transform Your Development Workflow!
|
### 🚀 Claude Code Integration: Transform Your Development Workflow!
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary><strong>NEW!! AST‑Aware Code Chunking</strong></summary>
|
|
||||||
|
|
||||||
LEANN features intelligent code chunking that preserves semantic boundaries (functions, classes, methods) for Python, Java, C#, and TypeScript, improving code understanding compared to text-based chunking.
|
|
||||||
|
|
||||||
📖 Read the [AST Chunking Guide →](docs/ast_chunking_guide.md)
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
**The future of code assistance is here.** Transform your development workflow with LEANN's native MCP integration for Claude Code. Index your entire codebase and get intelligent code assistance directly in your IDE.
|
**The future of code assistance is here.** Transform your development workflow with LEANN's native MCP integration for Claude Code. Index your entire codebase and get intelligent code assistance directly in your IDE.
|
||||||
|
|
||||||
**Key features:**
|
**Key features:**
|
||||||
- 🔍 **Semantic code search** across your entire project, fully local index and lightweight
|
- 🔍 **Semantic code search** across your entire project, fully local index and lightweight
|
||||||
- 🧠 **AST-aware chunking** preserves code structure (functions, classes)
|
|
||||||
- 📚 **Context-aware assistance** for debugging and development
|
- 📚 **Context-aware assistance** for debugging and development
|
||||||
- 🚀 **Zero-config setup** with automatic language detection
|
- 🚀 **Zero-config setup** with automatic language detection
|
||||||
|
|
||||||
@@ -555,8 +491,7 @@ leann remove my-docs
|
|||||||
|
|
||||||
**Key CLI features:**
|
**Key CLI features:**
|
||||||
- Auto-detects document formats (PDF, TXT, MD, DOCX, PPTX + code files)
|
- Auto-detects document formats (PDF, TXT, MD, DOCX, PPTX + code files)
|
||||||
- **🧠 AST-aware chunking** for Python, Java, C#, TypeScript files
|
- Smart text chunking with overlap
|
||||||
- Smart text chunking with overlap for all other content
|
|
||||||
- Multiple LLM providers (Ollama, OpenAI, HuggingFace)
|
- Multiple LLM providers (Ollama, OpenAI, HuggingFace)
|
||||||
- Organized index storage in `.leann/indexes/` (project-local)
|
- Organized index storage in `.leann/indexes/` (project-local)
|
||||||
- Support for advanced search parameters
|
- Support for advanced search parameters
|
||||||
@@ -607,10 +542,8 @@ Options:
|
|||||||
leann list
|
leann list
|
||||||
|
|
||||||
# Lists all indexes across all projects with status indicators:
|
# Lists all indexes across all projects with status indicators:
|
||||||
# ✅ - Index is complete and ready to use
|
# ✓ - Index is complete and ready to use
|
||||||
# ❌ - Index is incomplete or corrupted
|
# ✗ - Index is incomplete or corrupted
|
||||||
# 📁 - CLI-created index (in .leann/indexes/)
|
|
||||||
# 📄 - App-created index (*.leann.meta.json files)
|
|
||||||
```
|
```
|
||||||
|
|
||||||
**Remove Command:**
|
**Remove Command:**
|
||||||
@@ -624,51 +557,10 @@ Options:
|
|||||||
# - Shows all matching indexes across projects
|
# - Shows all matching indexes across projects
|
||||||
# - Requires confirmation for cross-project removal
|
# - Requires confirmation for cross-project removal
|
||||||
# - Interactive selection when multiple matches found
|
# - Interactive selection when multiple matches found
|
||||||
# - Supports both CLI and app-created indexes
|
|
||||||
```
|
```
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
## 🚀 Advanced Features
|
|
||||||
|
|
||||||
### 🎯 Metadata Filtering
|
|
||||||
|
|
||||||
LEANN supports a simple metadata filtering system to enable sophisticated use cases like document filtering by date/type, code search by file extension, and content management based on custom criteria.
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Add metadata during indexing
|
|
||||||
builder.add_text(
|
|
||||||
"def authenticate_user(token): ...",
|
|
||||||
metadata={"file_extension": ".py", "lines_of_code": 25}
|
|
||||||
)
|
|
||||||
|
|
||||||
# Search with filters
|
|
||||||
results = searcher.search(
|
|
||||||
query="authentication function",
|
|
||||||
metadata_filters={
|
|
||||||
"file_extension": {"==": ".py"},
|
|
||||||
"lines_of_code": {"<": 100}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
**Supported operators**: `==`, `!=`, `<`, `<=`, `>`, `>=`, `in`, `not_in`, `contains`, `starts_with`, `ends_with`, `is_true`, `is_false`
|
|
||||||
|
|
||||||
📖 **[Complete Metadata filtering guide →](docs/metadata_filtering.md)**
|
|
||||||
|
|
||||||
### 🔍 Grep Search
|
|
||||||
|
|
||||||
For exact text matching instead of semantic search, use the `use_grep` parameter:
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Exact text search
|
|
||||||
results = searcher.search("banana‑crocodile", use_grep=True, top_k=1)
|
|
||||||
```
|
|
||||||
|
|
||||||
**Use cases**: Finding specific code patterns, error messages, function names, or exact phrases where semantic similarity isn't needed.
|
|
||||||
|
|
||||||
📖 **[Complete grep search guide →](docs/grep_search.md)**
|
|
||||||
|
|
||||||
## 🏗️ Architecture & How It Works
|
## 🏗️ Architecture & How It Works
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
@@ -708,7 +600,6 @@ results = searcher.search("banana‑crocodile", use_grep=True, top_k=1)
|
|||||||
```bash
|
```bash
|
||||||
uv pip install -e ".[dev]" # Install dev dependencies
|
uv pip install -e ".[dev]" # Install dev dependencies
|
||||||
python benchmarks/run_evaluation.py # Will auto-download evaluation data and run benchmarks
|
python benchmarks/run_evaluation.py # Will auto-download evaluation data and run benchmarks
|
||||||
python benchmarks/run_evaluation.py benchmarks/data/indices/rpj_wiki/rpj_wiki --num-queries 2000 # After downloading data, you can run the benchmark with our biggest index
|
|
||||||
```
|
```
|
||||||
|
|
||||||
The evaluation script downloads data automatically on first run. The last three results were tested with partial personal data, and you can reproduce them with your own data!
|
The evaluation script downloads data automatically on first run. The last three results were tested with partial personal data, and you can reproduce them with your own data!
|
||||||
@@ -748,9 +639,6 @@ MIT License - see [LICENSE](LICENSE) for details.
|
|||||||
|
|
||||||
Core Contributors: [Yichuan Wang](https://yichuan-w.github.io/) & [Zhifei Li](https://github.com/andylizf).
|
Core Contributors: [Yichuan Wang](https://yichuan-w.github.io/) & [Zhifei Li](https://github.com/andylizf).
|
||||||
|
|
||||||
Active Contributors: [Gabriel Dehan](https://github.com/gabriel-dehan)
|
|
||||||
|
|
||||||
|
|
||||||
We welcome more contributors! Feel free to open issues or submit PRs.
|
We welcome more contributors! Feel free to open issues or submit PRs.
|
||||||
|
|
||||||
This work is done at [**Berkeley Sky Computing Lab**](https://sky.cs.berkeley.edu/).
|
This work is done at [**Berkeley Sky Computing Lab**](https://sky.cs.berkeley.edu/).
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from typing import Any
|
|||||||
|
|
||||||
import dotenv
|
import dotenv
|
||||||
from leann.api import LeannBuilder, LeannChat
|
from leann.api import LeannBuilder, LeannChat
|
||||||
from leann.registry import register_project_directory
|
from llama_index.core.node_parser import SentenceSplitter
|
||||||
|
|
||||||
dotenv.load_dotenv()
|
dotenv.load_dotenv()
|
||||||
|
|
||||||
@@ -108,38 +108,6 @@ class BaseRAGExample(ABC):
|
|||||||
help="Thinking budget for reasoning models (low/medium/high). Supported by GPT-Oss:20b and other reasoning models.",
|
help="Thinking budget for reasoning models (low/medium/high). Supported by GPT-Oss:20b and other reasoning models.",
|
||||||
)
|
)
|
||||||
|
|
||||||
# AST Chunking parameters
|
|
||||||
ast_group = parser.add_argument_group("AST Chunking Parameters")
|
|
||||||
ast_group.add_argument(
|
|
||||||
"--use-ast-chunking",
|
|
||||||
action="store_true",
|
|
||||||
help="Enable AST-aware chunking for code files (requires astchunk)",
|
|
||||||
)
|
|
||||||
ast_group.add_argument(
|
|
||||||
"--ast-chunk-size",
|
|
||||||
type=int,
|
|
||||||
default=512,
|
|
||||||
help="Maximum characters per AST chunk (default: 512)",
|
|
||||||
)
|
|
||||||
ast_group.add_argument(
|
|
||||||
"--ast-chunk-overlap",
|
|
||||||
type=int,
|
|
||||||
default=64,
|
|
||||||
help="Overlap between AST chunks (default: 64)",
|
|
||||||
)
|
|
||||||
ast_group.add_argument(
|
|
||||||
"--code-file-extensions",
|
|
||||||
nargs="+",
|
|
||||||
default=None,
|
|
||||||
help="Additional code file extensions to process with AST chunking (e.g., .py .java .cs .ts)",
|
|
||||||
)
|
|
||||||
ast_group.add_argument(
|
|
||||||
"--ast-fallback-traditional",
|
|
||||||
action="store_true",
|
|
||||||
default=True,
|
|
||||||
help="Fall back to traditional chunking if AST chunking fails (default: True)",
|
|
||||||
)
|
|
||||||
|
|
||||||
# Search parameters
|
# Search parameters
|
||||||
search_group = parser.add_argument_group("Search Parameters")
|
search_group = parser.add_argument_group("Search Parameters")
|
||||||
search_group.add_argument(
|
search_group.add_argument(
|
||||||
@@ -246,11 +214,6 @@ class BaseRAGExample(ABC):
|
|||||||
builder.build_index(index_path)
|
builder.build_index(index_path)
|
||||||
print(f"Index saved to: {index_path}")
|
print(f"Index saved to: {index_path}")
|
||||||
|
|
||||||
# Register project directory so leann list can discover this index
|
|
||||||
# The index is saved as args.index_dir/index_name.leann
|
|
||||||
# We want to register the current working directory where the app is run
|
|
||||||
register_project_directory(Path.cwd())
|
|
||||||
|
|
||||||
return index_path
|
return index_path
|
||||||
|
|
||||||
async def run_interactive_chat(self, args, index_path: str):
|
async def run_interactive_chat(self, args, index_path: str):
|
||||||
@@ -299,6 +262,7 @@ class BaseRAGExample(ABC):
|
|||||||
chat = LeannChat(
|
chat = LeannChat(
|
||||||
index_path,
|
index_path,
|
||||||
llm_config=self.get_llm_config(args),
|
llm_config=self.get_llm_config(args),
|
||||||
|
system_prompt=f"You are a helpful assistant that answers questions about {self.name} data.",
|
||||||
complexity=args.search_complexity,
|
complexity=args.search_complexity,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -340,3 +304,21 @@ class BaseRAGExample(ABC):
|
|||||||
await self.run_single_query(args, index_path, args.query)
|
await self.run_single_query(args, index_path, args.query)
|
||||||
else:
|
else:
|
||||||
await self.run_interactive_chat(args, index_path)
|
await self.run_interactive_chat(args, index_path)
|
||||||
|
|
||||||
|
|
||||||
|
def create_text_chunks(documents, chunk_size=256, chunk_overlap=25) -> list[str]:
|
||||||
|
"""Helper function to create text chunks from documents."""
|
||||||
|
node_parser = SentenceSplitter(
|
||||||
|
chunk_size=chunk_size,
|
||||||
|
chunk_overlap=chunk_overlap,
|
||||||
|
separator=" ",
|
||||||
|
paragraph_separator="\n\n",
|
||||||
|
)
|
||||||
|
|
||||||
|
all_texts = []
|
||||||
|
for doc in documents:
|
||||||
|
nodes = node_parser.get_nodes_from_documents([doc])
|
||||||
|
if nodes:
|
||||||
|
all_texts.extend(node.get_content() for node in nodes)
|
||||||
|
|
||||||
|
return all_texts
|
||||||
|
|||||||
@@ -10,8 +10,7 @@ from pathlib import Path
|
|||||||
# Add parent directory to path for imports
|
# Add parent directory to path for imports
|
||||||
sys.path.insert(0, str(Path(__file__).parent))
|
sys.path.insert(0, str(Path(__file__).parent))
|
||||||
|
|
||||||
from base_rag_example import BaseRAGExample
|
from base_rag_example import BaseRAGExample, create_text_chunks
|
||||||
from chunking import create_text_chunks
|
|
||||||
|
|
||||||
from .history_data.history import ChromeHistoryReader
|
from .history_data.history import ChromeHistoryReader
|
||||||
|
|
||||||
|
|||||||
@@ -1,44 +0,0 @@
|
|||||||
"""Unified chunking utilities facade.
|
|
||||||
|
|
||||||
This module re-exports the packaged utilities from `leann.chunking_utils` so
|
|
||||||
that both repo apps (importing `chunking`) and installed wheels share one
|
|
||||||
single implementation. When running from the repo without installation, it
|
|
||||||
adds the `packages/leann-core/src` directory to `sys.path` as a fallback.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import sys
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
try:
|
|
||||||
from leann.chunking_utils import (
|
|
||||||
CODE_EXTENSIONS,
|
|
||||||
create_ast_chunks,
|
|
||||||
create_text_chunks,
|
|
||||||
create_traditional_chunks,
|
|
||||||
detect_code_files,
|
|
||||||
get_language_from_extension,
|
|
||||||
)
|
|
||||||
except Exception: # pragma: no cover - best-effort fallback for dev environment
|
|
||||||
repo_root = Path(__file__).resolve().parents[2]
|
|
||||||
leann_src = repo_root / "packages" / "leann-core" / "src"
|
|
||||||
if leann_src.exists():
|
|
||||||
sys.path.insert(0, str(leann_src))
|
|
||||||
from leann.chunking_utils import (
|
|
||||||
CODE_EXTENSIONS,
|
|
||||||
create_ast_chunks,
|
|
||||||
create_text_chunks,
|
|
||||||
create_traditional_chunks,
|
|
||||||
detect_code_files,
|
|
||||||
get_language_from_extension,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
raise
|
|
||||||
|
|
||||||
__all__ = [
|
|
||||||
"CODE_EXTENSIONS",
|
|
||||||
"create_ast_chunks",
|
|
||||||
"create_text_chunks",
|
|
||||||
"create_traditional_chunks",
|
|
||||||
"detect_code_files",
|
|
||||||
"get_language_from_extension",
|
|
||||||
]
|
|
||||||
211
apps/code_rag.py
211
apps/code_rag.py
@@ -1,211 +0,0 @@
|
|||||||
"""
|
|
||||||
Code RAG example using AST-aware chunking for optimal code understanding.
|
|
||||||
Specialized for code repositories with automatic language detection and
|
|
||||||
optimized chunking parameters.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import sys
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
# Add parent directory to path for imports
|
|
||||||
sys.path.insert(0, str(Path(__file__).parent))
|
|
||||||
|
|
||||||
from base_rag_example import BaseRAGExample
|
|
||||||
from chunking import CODE_EXTENSIONS, create_text_chunks
|
|
||||||
from llama_index.core import SimpleDirectoryReader
|
|
||||||
|
|
||||||
|
|
||||||
class CodeRAG(BaseRAGExample):
|
|
||||||
"""Specialized RAG example for code repositories with AST-aware chunking."""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__(
|
|
||||||
name="Code",
|
|
||||||
description="Process and query code repositories with AST-aware chunking",
|
|
||||||
default_index_name="code_index",
|
|
||||||
)
|
|
||||||
# Override defaults for code-specific usage
|
|
||||||
self.embedding_model_default = "facebook/contriever" # Good for code
|
|
||||||
self.max_items_default = -1 # Process all code files by default
|
|
||||||
|
|
||||||
def _add_specific_arguments(self, parser):
|
|
||||||
"""Add code-specific arguments."""
|
|
||||||
code_group = parser.add_argument_group("Code Repository Parameters")
|
|
||||||
|
|
||||||
code_group.add_argument(
|
|
||||||
"--repo-dir",
|
|
||||||
type=str,
|
|
||||||
default=".",
|
|
||||||
help="Code repository directory to index (default: current directory)",
|
|
||||||
)
|
|
||||||
code_group.add_argument(
|
|
||||||
"--include-extensions",
|
|
||||||
nargs="+",
|
|
||||||
default=list(CODE_EXTENSIONS.keys()),
|
|
||||||
help="File extensions to include (default: supported code extensions)",
|
|
||||||
)
|
|
||||||
code_group.add_argument(
|
|
||||||
"--exclude-dirs",
|
|
||||||
nargs="+",
|
|
||||||
default=[
|
|
||||||
".git",
|
|
||||||
"__pycache__",
|
|
||||||
"node_modules",
|
|
||||||
"venv",
|
|
||||||
".venv",
|
|
||||||
"build",
|
|
||||||
"dist",
|
|
||||||
"target",
|
|
||||||
],
|
|
||||||
help="Directories to exclude from indexing",
|
|
||||||
)
|
|
||||||
code_group.add_argument(
|
|
||||||
"--max-file-size",
|
|
||||||
type=int,
|
|
||||||
default=1000000, # 1MB
|
|
||||||
help="Maximum file size in bytes to process (default: 1MB)",
|
|
||||||
)
|
|
||||||
code_group.add_argument(
|
|
||||||
"--include-comments",
|
|
||||||
action="store_true",
|
|
||||||
help="Include comments in chunking (useful for documentation)",
|
|
||||||
)
|
|
||||||
code_group.add_argument(
|
|
||||||
"--preserve-imports",
|
|
||||||
action="store_true",
|
|
||||||
default=True,
|
|
||||||
help="Try to preserve import statements in chunks (default: True)",
|
|
||||||
)
|
|
||||||
|
|
||||||
async def load_data(self, args) -> list[str]:
|
|
||||||
"""Load code files and convert to AST-aware chunks."""
|
|
||||||
print(f"🔍 Scanning code repository: {args.repo_dir}")
|
|
||||||
print(f"📁 Including extensions: {args.include_extensions}")
|
|
||||||
print(f"🚫 Excluding directories: {args.exclude_dirs}")
|
|
||||||
|
|
||||||
# Check if repository directory exists
|
|
||||||
repo_path = Path(args.repo_dir)
|
|
||||||
if not repo_path.exists():
|
|
||||||
raise ValueError(f"Repository directory not found: {args.repo_dir}")
|
|
||||||
|
|
||||||
# Load code files with filtering
|
|
||||||
reader_kwargs = {
|
|
||||||
"recursive": True,
|
|
||||||
"encoding": "utf-8",
|
|
||||||
"required_exts": args.include_extensions,
|
|
||||||
"exclude_hidden": True,
|
|
||||||
}
|
|
||||||
|
|
||||||
# Create exclusion filter
|
|
||||||
def file_filter(file_path: str) -> bool:
|
|
||||||
"""Filter out unwanted files and directories."""
|
|
||||||
path = Path(file_path)
|
|
||||||
|
|
||||||
# Check file size
|
|
||||||
try:
|
|
||||||
if path.stat().st_size > args.max_file_size:
|
|
||||||
print(f"⚠️ Skipping large file: {path.name} ({path.stat().st_size} bytes)")
|
|
||||||
return False
|
|
||||||
except Exception:
|
|
||||||
return False
|
|
||||||
|
|
||||||
# Check if in excluded directory
|
|
||||||
for exclude_dir in args.exclude_dirs:
|
|
||||||
if exclude_dir in path.parts:
|
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
try:
|
|
||||||
# Load documents with file filtering
|
|
||||||
documents = SimpleDirectoryReader(
|
|
||||||
args.repo_dir,
|
|
||||||
file_extractor=None, # Use default extractors
|
|
||||||
**reader_kwargs,
|
|
||||||
).load_data(show_progress=True)
|
|
||||||
|
|
||||||
# Apply custom filtering
|
|
||||||
filtered_docs = []
|
|
||||||
for doc in documents:
|
|
||||||
file_path = doc.metadata.get("file_path", "")
|
|
||||||
if file_filter(file_path):
|
|
||||||
filtered_docs.append(doc)
|
|
||||||
|
|
||||||
documents = filtered_docs
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
print(f"❌ Error loading code files: {e}")
|
|
||||||
return []
|
|
||||||
|
|
||||||
if not documents:
|
|
||||||
print(
|
|
||||||
f"❌ No code files found in {args.repo_dir} with extensions {args.include_extensions}"
|
|
||||||
)
|
|
||||||
return []
|
|
||||||
|
|
||||||
print(f"✅ Loaded {len(documents)} code files")
|
|
||||||
|
|
||||||
# Show breakdown by language/extension
|
|
||||||
ext_counts = {}
|
|
||||||
for doc in documents:
|
|
||||||
file_path = doc.metadata.get("file_path", "")
|
|
||||||
if file_path:
|
|
||||||
ext = Path(file_path).suffix.lower()
|
|
||||||
ext_counts[ext] = ext_counts.get(ext, 0) + 1
|
|
||||||
|
|
||||||
print("📊 Files by extension:")
|
|
||||||
for ext, count in sorted(ext_counts.items()):
|
|
||||||
print(f" {ext}: {count} files")
|
|
||||||
|
|
||||||
# Use AST-aware chunking by default for code
|
|
||||||
print(
|
|
||||||
f"🧠 Using AST-aware chunking (chunk_size: {args.ast_chunk_size}, overlap: {args.ast_chunk_overlap})"
|
|
||||||
)
|
|
||||||
|
|
||||||
all_texts = create_text_chunks(
|
|
||||||
documents,
|
|
||||||
chunk_size=256, # Fallback for non-code files
|
|
||||||
chunk_overlap=64,
|
|
||||||
use_ast_chunking=True, # Always use AST for code RAG
|
|
||||||
ast_chunk_size=args.ast_chunk_size,
|
|
||||||
ast_chunk_overlap=args.ast_chunk_overlap,
|
|
||||||
code_file_extensions=args.include_extensions,
|
|
||||||
ast_fallback_traditional=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Apply max_items limit if specified
|
|
||||||
if args.max_items > 0 and len(all_texts) > args.max_items:
|
|
||||||
print(f"⏳ Limiting to {args.max_items} chunks (from {len(all_texts)})")
|
|
||||||
all_texts = all_texts[: args.max_items]
|
|
||||||
|
|
||||||
print(f"✅ Generated {len(all_texts)} code chunks")
|
|
||||||
return all_texts
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
import asyncio
|
|
||||||
|
|
||||||
# Example queries for code RAG
|
|
||||||
print("\n💻 Code RAG Example")
|
|
||||||
print("=" * 50)
|
|
||||||
print("\nExample queries you can try:")
|
|
||||||
print("- 'How does the embedding computation work?'")
|
|
||||||
print("- 'What are the main classes in this codebase?'")
|
|
||||||
print("- 'Show me the search implementation'")
|
|
||||||
print("- 'How is error handling implemented?'")
|
|
||||||
print("- 'What design patterns are used?'")
|
|
||||||
print("- 'Explain the chunking logic'")
|
|
||||||
print("\n🚀 Features:")
|
|
||||||
print("- ✅ AST-aware chunking preserves code structure")
|
|
||||||
print("- ✅ Automatic language detection")
|
|
||||||
print("- ✅ Smart filtering of large files and common excludes")
|
|
||||||
print("- ✅ Optimized for code understanding")
|
|
||||||
print("\nUsage examples:")
|
|
||||||
print(" python -m apps.code_rag --repo-dir ./my_project")
|
|
||||||
print(
|
|
||||||
" python -m apps.code_rag --include-extensions .py .js --query 'How does authentication work?'"
|
|
||||||
)
|
|
||||||
print("\nOr run without --query for interactive mode\n")
|
|
||||||
|
|
||||||
rag = CodeRAG()
|
|
||||||
asyncio.run(rag.run())
|
|
||||||
@@ -9,8 +9,7 @@ from pathlib import Path
|
|||||||
# Add parent directory to path for imports
|
# Add parent directory to path for imports
|
||||||
sys.path.insert(0, str(Path(__file__).parent))
|
sys.path.insert(0, str(Path(__file__).parent))
|
||||||
|
|
||||||
from base_rag_example import BaseRAGExample
|
from base_rag_example import BaseRAGExample, create_text_chunks
|
||||||
from chunking import create_text_chunks
|
|
||||||
from llama_index.core import SimpleDirectoryReader
|
from llama_index.core import SimpleDirectoryReader
|
||||||
|
|
||||||
|
|
||||||
@@ -45,11 +44,6 @@ class DocumentRAG(BaseRAGExample):
|
|||||||
doc_group.add_argument(
|
doc_group.add_argument(
|
||||||
"--chunk-overlap", type=int, default=128, help="Text chunk overlap (default: 128)"
|
"--chunk-overlap", type=int, default=128, help="Text chunk overlap (default: 128)"
|
||||||
)
|
)
|
||||||
doc_group.add_argument(
|
|
||||||
"--enable-code-chunking",
|
|
||||||
action="store_true",
|
|
||||||
help="Enable AST-aware chunking for code files in the data directory",
|
|
||||||
)
|
|
||||||
|
|
||||||
async def load_data(self, args) -> list[str]:
|
async def load_data(self, args) -> list[str]:
|
||||||
"""Load documents and convert to text chunks."""
|
"""Load documents and convert to text chunks."""
|
||||||
@@ -82,22 +76,9 @@ class DocumentRAG(BaseRAGExample):
|
|||||||
|
|
||||||
print(f"Loaded {len(documents)} documents")
|
print(f"Loaded {len(documents)} documents")
|
||||||
|
|
||||||
# Determine chunking strategy
|
# Convert to text chunks
|
||||||
use_ast = args.enable_code_chunking or getattr(args, "use_ast_chunking", False)
|
|
||||||
|
|
||||||
if use_ast:
|
|
||||||
print("Using AST-aware chunking for code files")
|
|
||||||
|
|
||||||
# Convert to text chunks with optional AST support
|
|
||||||
all_texts = create_text_chunks(
|
all_texts = create_text_chunks(
|
||||||
documents,
|
documents, chunk_size=args.chunk_size, chunk_overlap=args.chunk_overlap
|
||||||
chunk_size=args.chunk_size,
|
|
||||||
chunk_overlap=args.chunk_overlap,
|
|
||||||
use_ast_chunking=use_ast,
|
|
||||||
ast_chunk_size=getattr(args, "ast_chunk_size", 512),
|
|
||||||
ast_chunk_overlap=getattr(args, "ast_chunk_overlap", 64),
|
|
||||||
code_file_extensions=getattr(args, "code_file_extensions", None),
|
|
||||||
ast_fallback_traditional=getattr(args, "ast_fallback_traditional", True),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Apply max_items limit if specified
|
# Apply max_items limit if specified
|
||||||
@@ -121,10 +102,6 @@ if __name__ == "__main__":
|
|||||||
print(
|
print(
|
||||||
"- 'What is the problem of developing pan gu model Huawei meets? (盘古大模型开发中遇到什么问题?)'"
|
"- 'What is the problem of developing pan gu model Huawei meets? (盘古大模型开发中遇到什么问题?)'"
|
||||||
)
|
)
|
||||||
print("\n🚀 NEW: Code-aware chunking available!")
|
|
||||||
print("- Use --enable-code-chunking to enable AST-aware chunking for code files")
|
|
||||||
print("- Supports Python, Java, C#, TypeScript files")
|
|
||||||
print("- Better semantic understanding of code structure")
|
|
||||||
print("\nOr run without --query for interactive mode\n")
|
print("\nOr run without --query for interactive mode\n")
|
||||||
|
|
||||||
rag = DocumentRAG()
|
rag = DocumentRAG()
|
||||||
|
|||||||
@@ -9,8 +9,7 @@ from pathlib import Path
|
|||||||
# Add parent directory to path for imports
|
# Add parent directory to path for imports
|
||||||
sys.path.insert(0, str(Path(__file__).parent))
|
sys.path.insert(0, str(Path(__file__).parent))
|
||||||
|
|
||||||
from base_rag_example import BaseRAGExample
|
from base_rag_example import BaseRAGExample, create_text_chunks
|
||||||
from chunking import create_text_chunks
|
|
||||||
|
|
||||||
from .email_data.LEANN_email_reader import EmlxReader
|
from .email_data.LEANN_email_reader import EmlxReader
|
||||||
|
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ class ChromeHistoryReader(BaseReader):
|
|||||||
if count >= max_count and max_count > 0:
|
if count >= max_count and max_count > 0:
|
||||||
break
|
break
|
||||||
|
|
||||||
last_visit, url, title, visit_count, typed_count, _hidden = row
|
last_visit, url, title, visit_count, typed_count, hidden = row
|
||||||
|
|
||||||
# Create document content with metadata embedded in text
|
# Create document content with metadata embedded in text
|
||||||
doc_content = f"""
|
doc_content = f"""
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 152 KiB |
82
benchmarks/data/.gitattributes
vendored
Normal file
82
benchmarks/data/.gitattributes
vendored
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
*.7z filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.arrow filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.bin filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.bz2 filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.ckpt filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.ftz filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.gz filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.h5 filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.joblib filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.lfs.* filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.lz4 filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.mds filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.mlmodel filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.model filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.msgpack filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.npy filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.npz filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.onnx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.ot filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.parquet filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.pb filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.pickle filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.pkl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.pt filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.pth filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.rar filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.safetensors filter=lfs diff=lfs merge=lfs -text
|
||||||
|
saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.tar.* filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.tar filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.tflite filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.tgz filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.wasm filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.xz filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.zip filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.zst filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
||||||
|
# Audio files - uncompressed
|
||||||
|
*.pcm filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.sam filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.raw filter=lfs diff=lfs merge=lfs -text
|
||||||
|
# Audio files - compressed
|
||||||
|
*.aac filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.flac filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.mp3 filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.ogg filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.wav filter=lfs diff=lfs merge=lfs -text
|
||||||
|
# Image files - uncompressed
|
||||||
|
*.bmp filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.gif filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.png filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.tiff filter=lfs diff=lfs merge=lfs -text
|
||||||
|
# Image files - compressed
|
||||||
|
*.jpg filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.jpeg filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.webp filter=lfs diff=lfs merge=lfs -text
|
||||||
|
# Video files - compressed
|
||||||
|
*.mp4 filter=lfs diff=lfs merge=lfs -text
|
||||||
|
*.webm filter=lfs diff=lfs merge=lfs -text
|
||||||
|
ground_truth/dpr/id_map.json filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/dpr/dpr_diskann.passages.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/dpr/dpr_diskann.passages.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/dpr/dpr_diskann_disk.index filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/dpr/leann.labels.map filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/leann.labels.map filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.index filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.0.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.0.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.1.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.1.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.2.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.2.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.3.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.3.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.4.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.4.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.5.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.5.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.6.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.6.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.7.idx filter=lfs diff=lfs merge=lfs -text
|
||||||
|
indices/rpj_wiki/rpj_wiki.passages.7.jsonl filter=lfs diff=lfs merge=lfs -text
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
---
|
|
||||||
license: mit
|
|
||||||
---
|
|
||||||
|
|
||||||
# LEANN-RAG Evaluation Data
|
|
||||||
|
|
||||||
This repository contains the necessary data to run the recall evaluation scripts for the [LEANN-RAG](https://huggingface.co/LEANN-RAG) project.
|
|
||||||
|
|
||||||
## Dataset Components
|
|
||||||
|
|
||||||
This dataset is structured into three main parts:
|
|
||||||
|
|
||||||
1. **Pre-built LEANN Indices**:
|
|
||||||
* `dpr/`: A pre-built index for the DPR dataset.
|
|
||||||
* `rpj_wiki/`: A pre-built index for the RPJ-Wiki dataset.
|
|
||||||
These indices were created using the `leann-core` library and are required by the `LeannSearcher`.
|
|
||||||
|
|
||||||
2. **Ground Truth Data**:
|
|
||||||
* `ground_truth/`: Contains the ground truth files (`flat_results_nq_k3.json`) for both the DPR and RPJ-Wiki datasets. These files map queries to the original passage IDs from the Natural Questions benchmark, evaluated using the Contriever model.
|
|
||||||
|
|
||||||
3. **Queries**:
|
|
||||||
* `queries/`: Contains the `nq_open.jsonl` file with the Natural Questions queries used for the evaluation.
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
To use this data, you can download it locally using the `huggingface-hub` library. First, install the library:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
pip install huggingface-hub
|
|
||||||
```
|
|
||||||
|
|
||||||
Then, you can download the entire dataset to a local directory (e.g., `data/`) with the following Python script:
|
|
||||||
|
|
||||||
```python
|
|
||||||
from huggingface_hub import snapshot_download
|
|
||||||
|
|
||||||
snapshot_download(
|
|
||||||
repo_id="LEANN-RAG/leann-rag-evaluation-data",
|
|
||||||
repo_type="dataset",
|
|
||||||
local_dir="data"
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
This will download all the necessary files into a local `data` folder, preserving the repository structure. The evaluation scripts in the main [LEANN-RAG Space](https://huggingface.co/LEANN-RAG) are configured to work with this data structure.
|
|
||||||
@@ -12,7 +12,7 @@ import time
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
from leann.api import LeannBuilder, LeannChat, LeannSearcher
|
from leann.api import LeannBuilder, LeannSearcher
|
||||||
|
|
||||||
|
|
||||||
def download_data_if_needed(data_root: Path, download_embeddings: bool = False):
|
def download_data_if_needed(data_root: Path, download_embeddings: bool = False):
|
||||||
@@ -197,25 +197,6 @@ def main():
|
|||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--ef-search", type=int, default=120, help="The 'efSearch' parameter for HNSW."
|
"--ef-search", type=int, default=120, help="The 'efSearch' parameter for HNSW."
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
|
||||||
"--batch-size",
|
|
||||||
type=int,
|
|
||||||
default=0,
|
|
||||||
help="Batch size for HNSW batched search (0 disables batching)",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--llm-type",
|
|
||||||
type=str,
|
|
||||||
choices=["ollama", "hf", "openai", "gemini", "simulated"],
|
|
||||||
default="ollama",
|
|
||||||
help="LLM backend type to optionally query during evaluation (default: ollama)",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--llm-model",
|
|
||||||
type=str,
|
|
||||||
default="qwen3:1.7b",
|
|
||||||
help="LLM model identifier for the chosen backend (default: qwen3:1.7b)",
|
|
||||||
)
|
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
# --- Path Configuration ---
|
# --- Path Configuration ---
|
||||||
@@ -337,24 +318,9 @@ def main():
|
|||||||
|
|
||||||
for i in range(num_eval_queries):
|
for i in range(num_eval_queries):
|
||||||
start_time = time.time()
|
start_time = time.time()
|
||||||
new_results = searcher.search(
|
new_results = searcher.search(queries[i], top_k=args.top_k, ef=args.ef_search)
|
||||||
queries[i],
|
|
||||||
top_k=args.top_k,
|
|
||||||
complexity=args.ef_search,
|
|
||||||
batch_size=args.batch_size,
|
|
||||||
)
|
|
||||||
search_times.append(time.time() - start_time)
|
search_times.append(time.time() - start_time)
|
||||||
|
|
||||||
# Optional: also call the LLM with configurable backend/model (does not affect recall)
|
|
||||||
llm_config = {"type": args.llm_type, "model": args.llm_model}
|
|
||||||
chat = LeannChat(args.index_path, llm_config=llm_config, searcher=searcher)
|
|
||||||
answer = chat.ask(
|
|
||||||
queries[i],
|
|
||||||
top_k=args.top_k,
|
|
||||||
complexity=args.ef_search,
|
|
||||||
batch_size=args.batch_size,
|
|
||||||
)
|
|
||||||
print(f"Answer: {answer}")
|
|
||||||
# Correct Recall Calculation: Based on TEXT content
|
# Correct Recall Calculation: Based on TEXT content
|
||||||
new_texts = {result.text for result in new_results}
|
new_texts = {result.text for result in new_results}
|
||||||
|
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ except ImportError:
|
|||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class BenchmarkConfig:
|
class BenchmarkConfig:
|
||||||
model_path: str = "facebook/contriever-msmarco"
|
model_path: str = "facebook/contriever"
|
||||||
batch_sizes: list[int] = None
|
batch_sizes: list[int] = None
|
||||||
seq_length: int = 256
|
seq_length: int = 256
|
||||||
num_runs: int = 5
|
num_runs: int = 5
|
||||||
@@ -34,7 +34,7 @@ class BenchmarkConfig:
|
|||||||
|
|
||||||
def __post_init__(self):
|
def __post_init__(self):
|
||||||
if self.batch_sizes is None:
|
if self.batch_sizes is None:
|
||||||
self.batch_sizes = [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024]
|
self.batch_sizes = [1, 2, 4, 8, 16, 32, 64]
|
||||||
|
|
||||||
|
|
||||||
class MLXBenchmark:
|
class MLXBenchmark:
|
||||||
@@ -179,16 +179,10 @@ class Benchmark:
|
|||||||
|
|
||||||
def _run_inference(self, input_ids: torch.Tensor) -> float:
|
def _run_inference(self, input_ids: torch.Tensor) -> float:
|
||||||
attention_mask = torch.ones_like(input_ids)
|
attention_mask = torch.ones_like(input_ids)
|
||||||
# print shape of input_ids and attention_mask
|
|
||||||
print(f"input_ids shape: {input_ids.shape}")
|
|
||||||
print(f"attention_mask shape: {attention_mask.shape}")
|
|
||||||
start_time = time.time()
|
start_time = time.time()
|
||||||
with torch.no_grad():
|
with torch.no_grad():
|
||||||
self.model(input_ids=input_ids, attention_mask=attention_mask)
|
self.model(input_ids=input_ids, attention_mask=attention_mask)
|
||||||
if torch.cuda.is_available():
|
|
||||||
torch.cuda.synchronize()
|
|
||||||
if torch.backends.mps.is_available():
|
|
||||||
torch.mps.synchronize()
|
|
||||||
end_time = time.time()
|
end_time = time.time()
|
||||||
|
|
||||||
return end_time - start_time
|
return end_time - start_time
|
||||||
|
|||||||
@@ -1,143 +0,0 @@
|
|||||||
# AST-Aware Code chunking guide
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
This guide covers best practices for using AST-aware code chunking in LEANN. AST chunking provides better semantic understanding of code structure compared to traditional text-based chunking.
|
|
||||||
|
|
||||||
## Quick Start
|
|
||||||
|
|
||||||
### Basic Usage
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Enable AST chunking for mixed content (code + docs)
|
|
||||||
python -m apps.document_rag --enable-code-chunking --data-dir ./my_project
|
|
||||||
|
|
||||||
# Specialized code repository indexing
|
|
||||||
python -m apps.code_rag --repo-dir ./my_codebase
|
|
||||||
|
|
||||||
# Global CLI with AST support
|
|
||||||
leann build my-code-index --docs ./src --use-ast-chunking
|
|
||||||
```
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Install LEANN with AST chunking support
|
|
||||||
uv pip install -e "."
|
|
||||||
```
|
|
||||||
|
|
||||||
#### For normal users (PyPI install)
|
|
||||||
- Use `pip install leann` or `uv pip install leann`.
|
|
||||||
- `astchunk` is pulled automatically from PyPI as a dependency; no extra steps.
|
|
||||||
|
|
||||||
#### For developers (from source, editable)
|
|
||||||
```bash
|
|
||||||
git clone https://github.com/yichuan-w/LEANN.git leann
|
|
||||||
cd leann
|
|
||||||
git submodule update --init --recursive
|
|
||||||
uv sync
|
|
||||||
```
|
|
||||||
- This repo vendors `astchunk` as a git submodule at `packages/astchunk-leann` (our fork).
|
|
||||||
- `[tool.uv.sources]` maps the `astchunk` package to that path in editable mode.
|
|
||||||
- You can edit code under `packages/astchunk-leann` and Python will use your changes immediately (no separate `pip install astchunk` needed).
|
|
||||||
|
|
||||||
## Best Practices
|
|
||||||
|
|
||||||
### When to Use AST Chunking
|
|
||||||
|
|
||||||
✅ **Recommended for:**
|
|
||||||
- Code repositories with multiple languages
|
|
||||||
- Mixed documentation and code content
|
|
||||||
- Complex codebases with deep function/class hierarchies
|
|
||||||
- When working with Claude Code for code assistance
|
|
||||||
|
|
||||||
❌ **Not recommended for:**
|
|
||||||
- Pure text documents
|
|
||||||
- Very large files (>1MB)
|
|
||||||
- Languages not supported by tree-sitter
|
|
||||||
|
|
||||||
### Optimal Configuration
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Recommended settings for most codebases
|
|
||||||
python -m apps.code_rag \
|
|
||||||
--repo-dir ./src \
|
|
||||||
--ast-chunk-size 768 \
|
|
||||||
--ast-chunk-overlap 96 \
|
|
||||||
--exclude-dirs .git __pycache__ node_modules build dist
|
|
||||||
```
|
|
||||||
|
|
||||||
### Supported Languages
|
|
||||||
|
|
||||||
| Extension | Language | Status |
|
|
||||||
|-----------|----------|--------|
|
|
||||||
| `.py` | Python | ✅ Full support |
|
|
||||||
| `.java` | Java | ✅ Full support |
|
|
||||||
| `.cs` | C# | ✅ Full support |
|
|
||||||
| `.ts`, `.tsx` | TypeScript | ✅ Full support |
|
|
||||||
| `.js`, `.jsx` | JavaScript | ✅ Via TypeScript parser |
|
|
||||||
|
|
||||||
## Integration Examples
|
|
||||||
|
|
||||||
### Document RAG with Code Support
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Enable code chunking in document RAG
|
|
||||||
python -m apps.document_rag \
|
|
||||||
--enable-code-chunking \
|
|
||||||
--data-dir ./project \
|
|
||||||
--query "How does authentication work in the codebase?"
|
|
||||||
```
|
|
||||||
|
|
||||||
### Claude Code Integration
|
|
||||||
|
|
||||||
When using with Claude Code MCP server, AST chunking provides better context for:
|
|
||||||
- Code completion and suggestions
|
|
||||||
- Bug analysis and debugging
|
|
||||||
- Architecture understanding
|
|
||||||
- Refactoring assistance
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
### Common Issues
|
|
||||||
|
|
||||||
1. **Fallback to Traditional Chunking**
|
|
||||||
- Normal behavior for unsupported languages
|
|
||||||
- Check logs for specific language support
|
|
||||||
|
|
||||||
2. **Performance with Large Files**
|
|
||||||
- Adjust `--max-file-size` parameter
|
|
||||||
- Use `--exclude-dirs` to skip unnecessary directories
|
|
||||||
|
|
||||||
3. **Quality Issues**
|
|
||||||
- Try different `--ast-chunk-size` values (512, 768, 1024)
|
|
||||||
- Adjust overlap for better context preservation
|
|
||||||
|
|
||||||
### Debug Mode
|
|
||||||
|
|
||||||
```bash
|
|
||||||
export LEANN_LOG_LEVEL=DEBUG
|
|
||||||
python -m apps.code_rag --repo-dir ./my_code
|
|
||||||
```
|
|
||||||
|
|
||||||
## Migration from Traditional Chunking
|
|
||||||
|
|
||||||
Existing workflows continue to work without changes. To enable AST chunking:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Before
|
|
||||||
python -m apps.document_rag --chunk-size 256
|
|
||||||
|
|
||||||
# After (maintains traditional chunking for non-code files)
|
|
||||||
python -m apps.document_rag --enable-code-chunking --chunk-size 256 --ast-chunk-size 768
|
|
||||||
```
|
|
||||||
|
|
||||||
## References
|
|
||||||
|
|
||||||
- [astchunk GitHub Repository](https://github.com/yilinjz/astchunk)
|
|
||||||
- [LEANN MCP Integration](../packages/leann-mcp/README.md)
|
|
||||||
- [Research Paper](https://arxiv.org/html/2506.15655v1)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Note**: AST chunking maintains full backward compatibility while enhancing code understanding capabilities.
|
|
||||||
@@ -3,7 +3,6 @@
|
|||||||
## 🔥 Core Features
|
## 🔥 Core Features
|
||||||
|
|
||||||
- **🔄 Real-time Embeddings** - Eliminate heavy embedding storage with dynamic computation using optimized ZMQ servers and highly optimized search paradigm (overlapping and batching) with highly optimized embedding engine
|
- **🔄 Real-time Embeddings** - Eliminate heavy embedding storage with dynamic computation using optimized ZMQ servers and highly optimized search paradigm (overlapping and batching) with highly optimized embedding engine
|
||||||
- **🧠 AST-Aware Code Chunking** - Intelligent code chunking that preserves semantic boundaries (functions, classes, methods) for Python, Java, C#, and TypeScript files
|
|
||||||
- **📈 Scalable Architecture** - Handles millions of documents on consumer hardware; the larger your dataset, the more LEANN can save
|
- **📈 Scalable Architecture** - Handles millions of documents on consumer hardware; the larger your dataset, the more LEANN can save
|
||||||
- **🎯 Graph Pruning** - Advanced techniques to minimize the storage overhead of vector search to a limited footprint
|
- **🎯 Graph Pruning** - Advanced techniques to minimize the storage overhead of vector search to a limited footprint
|
||||||
- **🏗️ Pluggable Backends** - HNSW/FAISS (default), with optional DiskANN for large-scale deployments
|
- **🏗️ Pluggable Backends** - HNSW/FAISS (default), with optional DiskANN for large-scale deployments
|
||||||
|
|||||||
@@ -1,149 +0,0 @@
|
|||||||
# LEANN Grep Search Usage Guide
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
LEANN's grep search functionality provides exact text matching for finding specific code patterns, error messages, function names, or exact phrases in your indexed documents.
|
|
||||||
|
|
||||||
## Basic Usage
|
|
||||||
|
|
||||||
### Simple Grep Search
|
|
||||||
|
|
||||||
```python
|
|
||||||
from leann.api import LeannSearcher
|
|
||||||
|
|
||||||
searcher = LeannSearcher("your_index_path")
|
|
||||||
|
|
||||||
# Exact text search
|
|
||||||
results = searcher.search("def authenticate_user", use_grep=True, top_k=5)
|
|
||||||
|
|
||||||
for result in results:
|
|
||||||
print(f"Score: {result.score}")
|
|
||||||
print(f"Text: {result.text[:100]}...")
|
|
||||||
print("-" * 40)
|
|
||||||
```
|
|
||||||
|
|
||||||
### Comparison: Semantic vs Grep Search
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Semantic search - finds conceptually similar content
|
|
||||||
semantic_results = searcher.search("machine learning algorithms", top_k=3)
|
|
||||||
|
|
||||||
# Grep search - finds exact text matches
|
|
||||||
grep_results = searcher.search("def train_model", use_grep=True, top_k=3)
|
|
||||||
```
|
|
||||||
|
|
||||||
## When to Use Grep Search
|
|
||||||
|
|
||||||
### Use Cases
|
|
||||||
|
|
||||||
- **Code Search**: Finding specific function definitions, class names, or variable references
|
|
||||||
- **Error Debugging**: Locating exact error messages or stack traces
|
|
||||||
- **Documentation**: Finding specific API endpoints or exact terminology
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Find function definitions
|
|
||||||
functions = searcher.search("def __init__", use_grep=True)
|
|
||||||
|
|
||||||
# Find import statements
|
|
||||||
imports = searcher.search("from sklearn import", use_grep=True)
|
|
||||||
|
|
||||||
# Find specific error types
|
|
||||||
errors = searcher.search("FileNotFoundError", use_grep=True)
|
|
||||||
|
|
||||||
# Find TODO comments
|
|
||||||
todos = searcher.search("TODO:", use_grep=True)
|
|
||||||
|
|
||||||
# Find configuration entries
|
|
||||||
configs = searcher.search("server_port=", use_grep=True)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Technical Details
|
|
||||||
|
|
||||||
### How It Works
|
|
||||||
|
|
||||||
1. **File Location**: Grep search operates on the raw text stored in `.jsonl` files
|
|
||||||
2. **Command Execution**: Uses the system `grep` command with case-insensitive search
|
|
||||||
3. **Result Processing**: Parses JSON lines and extracts text and metadata
|
|
||||||
4. **Scoring**: Simple frequency-based scoring based on query term occurrences
|
|
||||||
|
|
||||||
### Search Process
|
|
||||||
|
|
||||||
```
|
|
||||||
Query: "def train_model"
|
|
||||||
↓
|
|
||||||
grep -i -n "def train_model" documents.leann.passages.jsonl
|
|
||||||
↓
|
|
||||||
Parse matching JSON lines
|
|
||||||
↓
|
|
||||||
Calculate scores based on term frequency
|
|
||||||
↓
|
|
||||||
Return top_k results
|
|
||||||
```
|
|
||||||
|
|
||||||
### Scoring Algorithm
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Term frequency in document
|
|
||||||
score = text.lower().count(query.lower())
|
|
||||||
```
|
|
||||||
|
|
||||||
Results are ranked by score (highest first), with higher scores indicating more occurrences of the search term.
|
|
||||||
|
|
||||||
## Error Handling
|
|
||||||
|
|
||||||
### Common Issues
|
|
||||||
|
|
||||||
#### Grep Command Not Found
|
|
||||||
```
|
|
||||||
RuntimeError: grep command not found. Please install grep or use semantic search.
|
|
||||||
```
|
|
||||||
|
|
||||||
**Solution**: Install grep on your system:
|
|
||||||
- **Ubuntu/Debian**: `sudo apt-get install grep`
|
|
||||||
- **macOS**: grep is pre-installed
|
|
||||||
- **Windows**: Use WSL or install grep via Git Bash/MSYS2
|
|
||||||
|
|
||||||
#### No Results Found
|
|
||||||
```python
|
|
||||||
# Check if your query exists in the raw data
|
|
||||||
results = searcher.search("your_query", use_grep=True)
|
|
||||||
if not results:
|
|
||||||
print("No exact matches found. Try:")
|
|
||||||
print("1. Check spelling and case")
|
|
||||||
print("2. Use partial terms")
|
|
||||||
print("3. Switch to semantic search")
|
|
||||||
```
|
|
||||||
|
|
||||||
## Complete Example
|
|
||||||
|
|
||||||
```python
|
|
||||||
#!/usr/bin/env python3
|
|
||||||
"""
|
|
||||||
Grep Search Example
|
|
||||||
Demonstrates grep search for exact text matching.
|
|
||||||
"""
|
|
||||||
|
|
||||||
from leann.api import LeannSearcher
|
|
||||||
|
|
||||||
def demonstrate_grep_search():
|
|
||||||
# Initialize searcher
|
|
||||||
searcher = LeannSearcher("my_index")
|
|
||||||
|
|
||||||
print("=== Function Search ===")
|
|
||||||
functions = searcher.search("def __init__", use_grep=True, top_k=5)
|
|
||||||
for i, result in enumerate(functions, 1):
|
|
||||||
print(f"{i}. Score: {result.score}")
|
|
||||||
print(f" Preview: {result.text[:60]}...")
|
|
||||||
print()
|
|
||||||
|
|
||||||
print("=== Error Search ===")
|
|
||||||
errors = searcher.search("FileNotFoundError", use_grep=True, top_k=3)
|
|
||||||
for result in errors:
|
|
||||||
print(f"Content: {result.text.strip()}")
|
|
||||||
print("-" * 40)
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
demonstrate_grep_search()
|
|
||||||
```
|
|
||||||
@@ -1,300 +0,0 @@
|
|||||||
# LEANN Metadata Filtering Usage Guide
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
Leann possesses metadata filtering capabilities that allow you to filter search results based on arbitrary metadata fields set during chunking. This feature enables use cases like spoiler-free book search, document filtering by date/type, code search by file type, and potentially much more.
|
|
||||||
|
|
||||||
## Basic Usage
|
|
||||||
|
|
||||||
### Adding Metadata to Your Documents
|
|
||||||
|
|
||||||
When building your index, add metadata to each text chunk:
|
|
||||||
|
|
||||||
```python
|
|
||||||
from leann.api import LeannBuilder
|
|
||||||
|
|
||||||
builder = LeannBuilder("hnsw")
|
|
||||||
|
|
||||||
# Add text with metadata
|
|
||||||
builder.add_text(
|
|
||||||
text="Chapter 1: Alice falls down the rabbit hole",
|
|
||||||
metadata={
|
|
||||||
"chapter": 1,
|
|
||||||
"character": "Alice",
|
|
||||||
"themes": ["adventure", "curiosity"],
|
|
||||||
"word_count": 150
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
builder.build_index("alice_in_wonderland_index")
|
|
||||||
```
|
|
||||||
|
|
||||||
### Searching with Metadata Filters
|
|
||||||
|
|
||||||
Use the `metadata_filters` parameter in search calls:
|
|
||||||
|
|
||||||
```python
|
|
||||||
from leann.api import LeannSearcher
|
|
||||||
|
|
||||||
searcher = LeannSearcher("alice_in_wonderland_index")
|
|
||||||
|
|
||||||
# Search with filters
|
|
||||||
results = searcher.search(
|
|
||||||
query="What happens to Alice?",
|
|
||||||
top_k=10,
|
|
||||||
metadata_filters={
|
|
||||||
"chapter": {"<=": 5}, # Only chapters 1-5
|
|
||||||
"spoiler_level": {"!=": "high"} # No high spoilers
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Filter Syntax
|
|
||||||
|
|
||||||
### Basic Structure
|
|
||||||
|
|
||||||
```python
|
|
||||||
metadata_filters = {
|
|
||||||
"field_name": {"operator": value},
|
|
||||||
"another_field": {"operator": value}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Supported Operators
|
|
||||||
|
|
||||||
#### Comparison Operators
|
|
||||||
- `"=="`: Equal to
|
|
||||||
- `"!="`: Not equal to
|
|
||||||
- `"<"`: Less than
|
|
||||||
- `"<="`: Less than or equal
|
|
||||||
- `">"`: Greater than
|
|
||||||
- `">="`: Greater than or equal
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Examples
|
|
||||||
{"chapter": {"==": 1}} # Exactly chapter 1
|
|
||||||
{"page": {">": 100}} # Pages after 100
|
|
||||||
{"rating": {">=": 4.0}} # Rating 4.0 or higher
|
|
||||||
{"word_count": {"<": 500}} # Short passages
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Membership Operators
|
|
||||||
- `"in"`: Value is in list
|
|
||||||
- `"not_in"`: Value is not in list
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Examples
|
|
||||||
{"character": {"in": ["Alice", "Bob"]}} # Alice OR Bob
|
|
||||||
{"genre": {"not_in": ["horror", "thriller"]}} # Exclude genres
|
|
||||||
{"tags": {"in": ["fiction", "adventure"]}} # Any of these tags
|
|
||||||
```
|
|
||||||
|
|
||||||
#### String Operators
|
|
||||||
- `"contains"`: String contains substring
|
|
||||||
- `"starts_with"`: String starts with prefix
|
|
||||||
- `"ends_with"`: String ends with suffix
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Examples
|
|
||||||
{"title": {"contains": "alice"}} # Title contains "alice"
|
|
||||||
{"filename": {"ends_with": ".py"}} # Python files
|
|
||||||
{"author": {"starts_with": "Dr."}} # Authors with "Dr." prefix
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Boolean Operators
|
|
||||||
- `"is_true"`: Field is truthy
|
|
||||||
- `"is_false"`: Field is falsy
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Examples
|
|
||||||
{"is_published": {"is_true": True}} # Published content
|
|
||||||
{"is_draft": {"is_false": False}} # Not drafts
|
|
||||||
```
|
|
||||||
|
|
||||||
### Multiple Operators on Same Field
|
|
||||||
|
|
||||||
You can apply multiple operators to the same field (AND logic):
|
|
||||||
|
|
||||||
```python
|
|
||||||
metadata_filters = {
|
|
||||||
"word_count": {
|
|
||||||
">=": 100, # At least 100 words
|
|
||||||
"<=": 500 # At most 500 words
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Compound Filters
|
|
||||||
|
|
||||||
Multiple fields are combined with AND logic:
|
|
||||||
|
|
||||||
```python
|
|
||||||
metadata_filters = {
|
|
||||||
"chapter": {"<=": 10}, # Up to chapter 10
|
|
||||||
"character": {"==": "Alice"}, # About Alice
|
|
||||||
"spoiler_level": {"!=": "high"} # No major spoilers
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Use Case Examples
|
|
||||||
|
|
||||||
### 1. Spoiler-Free Book Search
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Reader has only read up to chapter 5
|
|
||||||
def search_spoiler_free(query, max_chapter):
|
|
||||||
return searcher.search(
|
|
||||||
query=query,
|
|
||||||
metadata_filters={
|
|
||||||
"chapter": {"<=": max_chapter},
|
|
||||||
"spoiler_level": {"in": ["none", "low"]}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
results = search_spoiler_free("What happens to Alice?", max_chapter=5)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Document Management by Date
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Find recent documents
|
|
||||||
recent_docs = searcher.search(
|
|
||||||
query="project updates",
|
|
||||||
metadata_filters={
|
|
||||||
"date": {">=": "2024-01-01"},
|
|
||||||
"document_type": {"==": "report"}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Code Search by File Type
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Search only Python files
|
|
||||||
python_code = searcher.search(
|
|
||||||
query="authentication function",
|
|
||||||
metadata_filters={
|
|
||||||
"file_extension": {"==": ".py"},
|
|
||||||
"lines_of_code": {"<": 100}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. Content Filtering by Audience
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Age-appropriate content
|
|
||||||
family_content = searcher.search(
|
|
||||||
query="adventure stories",
|
|
||||||
metadata_filters={
|
|
||||||
"age_rating": {"in": ["G", "PG"]},
|
|
||||||
"content_warnings": {"not_in": ["violence", "adult_themes"]}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 5. Multi-Book Series Management
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Search across first 3 books only
|
|
||||||
early_series = searcher.search(
|
|
||||||
query="character development",
|
|
||||||
metadata_filters={
|
|
||||||
"series": {"==": "Harry Potter"},
|
|
||||||
"book_number": {"<=": 3}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Running the Example
|
|
||||||
|
|
||||||
You can see metadata filtering in action with our spoiler-free book RAG example:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Don't forget to set up the environment
|
|
||||||
uv venv
|
|
||||||
source .venv/bin/activate
|
|
||||||
|
|
||||||
# Set your OpenAI API key (required for embeddings, but you can update the example locally and use ollama instead)
|
|
||||||
export OPENAI_API_KEY="your-api-key-here"
|
|
||||||
|
|
||||||
# Run the spoiler-free book RAG example
|
|
||||||
uv run examples/spoiler_free_book_rag.py
|
|
||||||
```
|
|
||||||
|
|
||||||
This example demonstrates:
|
|
||||||
- Building an index with metadata (chapter numbers, characters, themes, locations)
|
|
||||||
- Searching with filters to avoid spoilers (e.g., only show results up to chapter 5)
|
|
||||||
- Different scenarios for readers at various points in the book
|
|
||||||
|
|
||||||
The example uses Alice's Adventures in Wonderland as sample data and shows how you can search for information without revealing plot points from later chapters.
|
|
||||||
|
|
||||||
## Advanced Patterns
|
|
||||||
|
|
||||||
### Custom Chunking with metadata
|
|
||||||
|
|
||||||
```python
|
|
||||||
def chunk_book_with_metadata(book_text, book_info):
|
|
||||||
chunks = []
|
|
||||||
|
|
||||||
for chapter_num, chapter_text in parse_chapters(book_text):
|
|
||||||
# Extract entities, themes, etc.
|
|
||||||
characters = extract_characters(chapter_text)
|
|
||||||
themes = classify_themes(chapter_text)
|
|
||||||
spoiler_level = assess_spoiler_level(chapter_text, chapter_num)
|
|
||||||
|
|
||||||
# Create chunks with rich metadata
|
|
||||||
for paragraph in split_paragraphs(chapter_text):
|
|
||||||
chunks.append({
|
|
||||||
"text": paragraph,
|
|
||||||
"metadata": {
|
|
||||||
"book_title": book_info["title"],
|
|
||||||
"chapter": chapter_num,
|
|
||||||
"characters": characters,
|
|
||||||
"themes": themes,
|
|
||||||
"spoiler_level": spoiler_level,
|
|
||||||
"word_count": len(paragraph.split()),
|
|
||||||
"reading_level": calculate_reading_level(paragraph)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return chunks
|
|
||||||
```
|
|
||||||
|
|
||||||
## Performance Considerations
|
|
||||||
|
|
||||||
### Efficient Filtering Strategies
|
|
||||||
|
|
||||||
1. **Post-search filtering**: Applies filters after vector search, which should be efficient for typical result sets (10-100 results).
|
|
||||||
|
|
||||||
2. **Metadata design**: Keep metadata fields simple and avoid deeply nested structures.
|
|
||||||
|
|
||||||
### Best Practices
|
|
||||||
|
|
||||||
1. **Consistent metadata schema**: Use consistent field names and value types across your documents.
|
|
||||||
|
|
||||||
2. **Reasonable metadata size**: Keep metadata reasonably sized to avoid storage overhead.
|
|
||||||
|
|
||||||
3. **Type consistency**: Use consistent data types for the same fields (e.g., always integers for chapter numbers).
|
|
||||||
|
|
||||||
4. **Index multiple granularities**: Consider chunking at different levels (paragraph, section, chapter) with appropriate metadata.
|
|
||||||
|
|
||||||
### Adding Metadata to Existing Indices
|
|
||||||
|
|
||||||
To add metadata filtering to existing indices, you'll need to rebuild them with metadata:
|
|
||||||
|
|
||||||
```python
|
|
||||||
# Read existing passages and add metadata
|
|
||||||
def add_metadata_to_existing_chunks(chunks):
|
|
||||||
for chunk in chunks:
|
|
||||||
# Extract or assign metadata based on content
|
|
||||||
chunk["metadata"] = extract_metadata(chunk["text"])
|
|
||||||
return chunks
|
|
||||||
|
|
||||||
# Rebuild index with metadata
|
|
||||||
enhanced_chunks = add_metadata_to_existing_chunks(existing_chunks)
|
|
||||||
builder = LeannBuilder("hnsw")
|
|
||||||
for chunk in enhanced_chunks:
|
|
||||||
builder.add_text(chunk["text"], chunk["metadata"])
|
|
||||||
builder.build_index("enhanced_index")
|
|
||||||
```
|
|
||||||
@@ -1,380 +0,0 @@
|
|||||||
"""
|
|
||||||
Dynamic add example for LEANN using HNSW backend without recompute.
|
|
||||||
|
|
||||||
- Builds a base index from a directory of documents
|
|
||||||
- Incrementally adds new documents without recomputing stored embeddings
|
|
||||||
|
|
||||||
Defaults:
|
|
||||||
- Base data: /Users/yichuan/Desktop/code/LEANN/leann/data
|
|
||||||
- Incremental data: /Users/yichuan/Desktop/code/LEANN/leann/test_add
|
|
||||||
- Index path: <index_dir>/documents.leann
|
|
||||||
|
|
||||||
Usage examples:
|
|
||||||
uv run python examples/dynamic_add_leann_no_recompute.py --build-base \
|
|
||||||
--base-dir /Users/yichuan/Desktop/code/LEANN/leann/data \
|
|
||||||
--index-dir ./test_doc_files
|
|
||||||
|
|
||||||
uv run python examples/dynamic_add_leann_no_recompute.py --add-incremental \
|
|
||||||
--add-dir /Users/yichuan/Desktop/code/LEANN/leann/test_add \
|
|
||||||
--index-dir ./test_doc_files
|
|
||||||
|
|
||||||
Quick recompute test (both true):
|
|
||||||
# Recompute build
|
|
||||||
uv run python examples/dynamic_add_leann_no_recompute.py --build-base \
|
|
||||||
--recompute-build --ef-construction 200 \
|
|
||||||
--base-dir /Users/yichuan/Desktop/code/LEANN/leann/data \
|
|
||||||
--index-dir ./test_doc_files --index-name documents.leann
|
|
||||||
|
|
||||||
# Recompute add
|
|
||||||
uv run python examples/dynamic_add_leann_no_recompute.py --add-incremental \
|
|
||||||
--recompute-add --ef-construction 32 \
|
|
||||||
--add-dir /Users/yichuan/Desktop/code/LEANN/leann/test_add \
|
|
||||||
--index-dir ./test_doc_files --index-name documents.leann
|
|
||||||
"""
|
|
||||||
|
|
||||||
import argparse
|
|
||||||
import json
|
|
||||||
import pickle
|
|
||||||
import sys
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Any, Optional
|
|
||||||
|
|
||||||
# Ensure we can import from the local packages and apps folders
|
|
||||||
ROOT = Path(__file__).resolve().parents[1]
|
|
||||||
CORE_SRC = ROOT / "packages" / "leann-core" / "src"
|
|
||||||
HNSW_PKG_DIR = ROOT / "packages" / "leann-backend-hnsw"
|
|
||||||
APPS_DIR = ROOT / "apps"
|
|
||||||
|
|
||||||
|
|
||||||
# Prefer the installed backend if available (it contains the compiled extension)
|
|
||||||
def _prefer_installed(pkg_name: str) -> bool:
|
|
||||||
try:
|
|
||||||
import importlib
|
|
||||||
import importlib.util
|
|
||||||
|
|
||||||
spec = importlib.util.find_spec(pkg_name)
|
|
||||||
if spec and spec.origin and "site-packages" in spec.origin:
|
|
||||||
# ensure the faiss shim/extension is importable from the installed package
|
|
||||||
importlib.import_module(f"{pkg_name}.faiss")
|
|
||||||
return True
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
# Prepend paths, but only add the repo backend if the installed one is not present
|
|
||||||
paths_to_prepend = [CORE_SRC, APPS_DIR]
|
|
||||||
if not _prefer_installed("leann_backend_hnsw"):
|
|
||||||
paths_to_prepend.insert(1, HNSW_PKG_DIR)
|
|
||||||
|
|
||||||
for p in paths_to_prepend:
|
|
||||||
p_str = str(p)
|
|
||||||
if p_str not in sys.path:
|
|
||||||
sys.path.insert(0, p_str)
|
|
||||||
|
|
||||||
# Defer non-stdlib imports until after sys.path setup within functions (avoid E402)
|
|
||||||
|
|
||||||
|
|
||||||
def _load_documents(data_dir: str, required_exts: Optional[list[str]] = None) -> list[Any]:
|
|
||||||
from llama_index.core import SimpleDirectoryReader # type: ignore
|
|
||||||
|
|
||||||
reader_kwargs: dict[str, Any] = {"recursive": True, "encoding": "utf-8"}
|
|
||||||
if required_exts:
|
|
||||||
reader_kwargs["required_exts"] = required_exts
|
|
||||||
documents = SimpleDirectoryReader(data_dir, **reader_kwargs).load_data(show_progress=True)
|
|
||||||
return documents
|
|
||||||
|
|
||||||
|
|
||||||
def _ensure_index_dir(index_dir: Path) -> None:
|
|
||||||
index_dir.mkdir(parents=True, exist_ok=True)
|
|
||||||
|
|
||||||
|
|
||||||
def _index_files(index_path: Path) -> tuple[Path, Path, Path]:
|
|
||||||
"""Return (passages.jsonl, passages.idx, index.index) paths for a given index base path.
|
|
||||||
|
|
||||||
Note: HNSWBackend writes the FAISS index using the stem (without .leann),
|
|
||||||
i.e., for base 'documents.leann' the file is 'documents.index'. We prefer the
|
|
||||||
existing file among candidates.
|
|
||||||
"""
|
|
||||||
passages_file = index_path.parent / f"{index_path.name}.passages.jsonl"
|
|
||||||
offsets_file = index_path.parent / f"{index_path.name}.passages.idx"
|
|
||||||
candidate_name_index = index_path.parent / f"{index_path.name}.index"
|
|
||||||
candidate_stem_index = index_path.parent / f"{index_path.stem}.index"
|
|
||||||
index_file = candidate_stem_index if candidate_stem_index.exists() else candidate_name_index
|
|
||||||
return passages_file, offsets_file, index_file
|
|
||||||
|
|
||||||
|
|
||||||
def _read_meta(index_path: Path) -> dict[str, Any]:
|
|
||||||
meta_path = index_path.parent / f"{index_path.name}.meta.json"
|
|
||||||
if not meta_path.exists():
|
|
||||||
raise FileNotFoundError(f"Metadata file not found: {meta_path}")
|
|
||||||
with open(meta_path, encoding="utf-8") as f:
|
|
||||||
return json.load(f)
|
|
||||||
|
|
||||||
|
|
||||||
def _autodetect_index_base(index_dir: Path) -> Optional[Path]:
|
|
||||||
"""If exactly one *.leann.meta.json exists, return its base path (without .meta.json)."""
|
|
||||||
candidates = list(index_dir.glob("*.leann.meta.json"))
|
|
||||||
if len(candidates) == 1:
|
|
||||||
meta = candidates[0]
|
|
||||||
base = meta.with_suffix("") # remove .json
|
|
||||||
base = base.with_suffix("") # remove .meta
|
|
||||||
return base
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
def _load_offset_map(offsets_file: Path) -> dict[str, int]:
|
|
||||||
if not offsets_file.exists():
|
|
||||||
return {}
|
|
||||||
with open(offsets_file, "rb") as f:
|
|
||||||
return pickle.load(f)
|
|
||||||
|
|
||||||
|
|
||||||
def _next_numeric_id(existing_ids: list[str]) -> int:
|
|
||||||
numeric_ids = [int(x) for x in existing_ids if x.isdigit()]
|
|
||||||
if not numeric_ids:
|
|
||||||
return 0
|
|
||||||
return max(numeric_ids) + 1
|
|
||||||
|
|
||||||
|
|
||||||
def build_base_index(
|
|
||||||
base_dir: str,
|
|
||||||
index_dir: str,
|
|
||||||
index_name: str,
|
|
||||||
embedding_model: str,
|
|
||||||
embedding_mode: str,
|
|
||||||
chunk_size: int,
|
|
||||||
chunk_overlap: int,
|
|
||||||
file_types: Optional[list[str]] = None,
|
|
||||||
max_items: int = -1,
|
|
||||||
ef_construction: Optional[int] = None,
|
|
||||||
recompute_build: bool = False,
|
|
||||||
) -> str:
|
|
||||||
print(f"Building base index from: {base_dir}")
|
|
||||||
documents = _load_documents(base_dir, required_exts=file_types)
|
|
||||||
if not documents:
|
|
||||||
raise ValueError(f"No documents found in base_dir: {base_dir}")
|
|
||||||
|
|
||||||
from chunking import create_text_chunks
|
|
||||||
|
|
||||||
texts = create_text_chunks(
|
|
||||||
documents,
|
|
||||||
chunk_size=chunk_size,
|
|
||||||
chunk_overlap=chunk_overlap,
|
|
||||||
use_ast_chunking=False,
|
|
||||||
)
|
|
||||||
if max_items > 0 and len(texts) > max_items:
|
|
||||||
texts = texts[:max_items]
|
|
||||||
print(f"Limiting to {max_items} chunks")
|
|
||||||
|
|
||||||
index_dir_path = Path(index_dir)
|
|
||||||
_ensure_index_dir(index_dir_path)
|
|
||||||
index_path = index_dir_path / index_name
|
|
||||||
|
|
||||||
print("Creating HNSW index (non-compact)...")
|
|
||||||
from leann.api import LeannBuilder
|
|
||||||
from leann.registry import register_project_directory
|
|
||||||
|
|
||||||
builder = LeannBuilder(
|
|
||||||
backend_name="hnsw",
|
|
||||||
embedding_model=embedding_model,
|
|
||||||
embedding_mode=embedding_mode,
|
|
||||||
is_recompute=recompute_build,
|
|
||||||
is_compact=False,
|
|
||||||
efConstruction=(ef_construction if ef_construction is not None else 200),
|
|
||||||
)
|
|
||||||
for t in texts:
|
|
||||||
builder.add_text(t)
|
|
||||||
builder.build_index(str(index_path))
|
|
||||||
|
|
||||||
# Register for discovery
|
|
||||||
register_project_directory(Path.cwd())
|
|
||||||
|
|
||||||
print(f"Base index built at: {index_path}")
|
|
||||||
return str(index_path)
|
|
||||||
|
|
||||||
|
|
||||||
def add_incremental(
|
|
||||||
add_dir: str,
|
|
||||||
index_dir: str,
|
|
||||||
index_name: Optional[str] = None,
|
|
||||||
embedding_model: Optional[str] = None,
|
|
||||||
embedding_mode: Optional[str] = None,
|
|
||||||
chunk_size: int = 256,
|
|
||||||
chunk_overlap: int = 128,
|
|
||||||
file_types: Optional[list[str]] = None,
|
|
||||||
max_items: int = -1,
|
|
||||||
ef_construction: Optional[int] = None,
|
|
||||||
recompute_add: bool = False,
|
|
||||||
) -> str:
|
|
||||||
print(f"Adding incremental data from: {add_dir}")
|
|
||||||
index_dir_path = Path(index_dir)
|
|
||||||
index_path = index_dir_path / (index_name or "documents.leann")
|
|
||||||
|
|
||||||
# If specified base doesn't exist, try to auto-detect an existing base
|
|
||||||
try:
|
|
||||||
_read_meta(index_path)
|
|
||||||
except FileNotFoundError:
|
|
||||||
auto_base = _autodetect_index_base(index_dir_path)
|
|
||||||
if auto_base is not None:
|
|
||||||
print(f"Auto-detected index base: {auto_base.name}")
|
|
||||||
index_path = auto_base
|
|
||||||
_read_meta(index_path)
|
|
||||||
else:
|
|
||||||
raise FileNotFoundError(
|
|
||||||
f"No index metadata found for base '{index_path.name}'. Build base first with --build-base "
|
|
||||||
f"or provide --index-name to match an existing index (e.g., 'test_doc_files.leann')."
|
|
||||||
)
|
|
||||||
|
|
||||||
# Prepare validated context from core (checks backend/no-recompute and resolves embedding defaults)
|
|
||||||
from leann.api import create_incremental_add_context, incremental_add_texts_with_context
|
|
||||||
|
|
||||||
ctx = create_incremental_add_context(
|
|
||||||
str(index_path),
|
|
||||||
embedding_model=embedding_model,
|
|
||||||
embedding_mode=embedding_mode,
|
|
||||||
data_dir=add_dir,
|
|
||||||
required_exts=file_types,
|
|
||||||
chunk_size=chunk_size,
|
|
||||||
chunk_overlap=chunk_overlap,
|
|
||||||
max_items=max_items,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Use prepared texts from context to perform the add
|
|
||||||
prepared_texts = ctx.prepared_texts or []
|
|
||||||
if not prepared_texts:
|
|
||||||
print("No new chunks to add.")
|
|
||||||
return str(index_path)
|
|
||||||
|
|
||||||
added = incremental_add_texts_with_context(
|
|
||||||
ctx,
|
|
||||||
prepared_texts,
|
|
||||||
ef_construction=ef_construction,
|
|
||||||
recompute=recompute_add,
|
|
||||||
)
|
|
||||||
|
|
||||||
print(f"Incremental add completed. Added {added} chunks. Index: {index_path}")
|
|
||||||
return str(index_path)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
parser = argparse.ArgumentParser(
|
|
||||||
description="Dynamic add to LEANN HNSW index without recompute",
|
|
||||||
formatter_class=argparse.RawDescriptionHelpFormatter,
|
|
||||||
)
|
|
||||||
|
|
||||||
parser.add_argument("--build-base", action="store_true", help="Build base index")
|
|
||||||
parser.add_argument("--add-incremental", action="store_true", help="Add incremental data")
|
|
||||||
|
|
||||||
parser.add_argument(
|
|
||||||
"--base-dir",
|
|
||||||
type=str,
|
|
||||||
default="/Users/yichuan/Desktop/code/LEANN/leann/data",
|
|
||||||
help="Base data directory",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--add-dir",
|
|
||||||
type=str,
|
|
||||||
default="/Users/yichuan/Desktop/code/LEANN/leann/test_add",
|
|
||||||
help="Incremental data directory",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--index-dir",
|
|
||||||
type=str,
|
|
||||||
default="./test_doc_files",
|
|
||||||
help="Directory containing the index",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--index-name",
|
|
||||||
type=str,
|
|
||||||
default="documents.leann",
|
|
||||||
help=(
|
|
||||||
"Index base file name. If you built via document_rag.py, use 'test_doc_files.leann'. "
|
|
||||||
"Default: documents.leann"
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
parser.add_argument(
|
|
||||||
"--embedding-model",
|
|
||||||
type=str,
|
|
||||||
default="facebook/contriever",
|
|
||||||
help="Embedding model name",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--embedding-mode",
|
|
||||||
type=str,
|
|
||||||
default="sentence-transformers",
|
|
||||||
choices=["sentence-transformers", "openai", "mlx", "ollama"],
|
|
||||||
help="Embedding backend mode",
|
|
||||||
)
|
|
||||||
|
|
||||||
parser.add_argument("--chunk-size", type=int, default=256)
|
|
||||||
parser.add_argument("--chunk-overlap", type=int, default=128)
|
|
||||||
parser.add_argument("--file-types", nargs="+", default=None)
|
|
||||||
parser.add_argument("--max-items", type=int, default=-1)
|
|
||||||
parser.add_argument("--ef-construction", type=int, default=32)
|
|
||||||
parser.add_argument(
|
|
||||||
"--recompute-add", action="store_true", help="Enable recompute-mode add (non-compact only)"
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--recompute-build",
|
|
||||||
action="store_true",
|
|
||||||
help="Enable recompute-mode base build (non-compact only)",
|
|
||||||
)
|
|
||||||
|
|
||||||
args = parser.parse_args()
|
|
||||||
|
|
||||||
if not args.build_base and not args.add_incremental:
|
|
||||||
print("Nothing to do. Use --build-base and/or --add-incremental.")
|
|
||||||
return
|
|
||||||
|
|
||||||
index_path_str: Optional[str] = None
|
|
||||||
|
|
||||||
if args.build_base:
|
|
||||||
index_path_str = build_base_index(
|
|
||||||
base_dir=args.base_dir,
|
|
||||||
index_dir=args.index_dir,
|
|
||||||
index_name=args.index_name,
|
|
||||||
embedding_model=args.embedding_model,
|
|
||||||
embedding_mode=args.embedding_mode,
|
|
||||||
chunk_size=args.chunk_size,
|
|
||||||
chunk_overlap=args.chunk_overlap,
|
|
||||||
file_types=args.file_types,
|
|
||||||
max_items=args.max_items,
|
|
||||||
ef_construction=args.ef_construction,
|
|
||||||
recompute_build=args.recompute_build,
|
|
||||||
)
|
|
||||||
|
|
||||||
if args.add_incremental:
|
|
||||||
index_path_str = add_incremental(
|
|
||||||
add_dir=args.add_dir,
|
|
||||||
index_dir=args.index_dir,
|
|
||||||
index_name=args.index_name,
|
|
||||||
embedding_model=args.embedding_model,
|
|
||||||
embedding_mode=args.embedding_mode,
|
|
||||||
chunk_size=args.chunk_size,
|
|
||||||
chunk_overlap=args.chunk_overlap,
|
|
||||||
file_types=args.file_types,
|
|
||||||
max_items=args.max_items,
|
|
||||||
ef_construction=args.ef_construction,
|
|
||||||
recompute_add=args.recompute_add,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Optional: quick test query using searcher
|
|
||||||
if index_path_str:
|
|
||||||
try:
|
|
||||||
from leann.api import LeannSearcher
|
|
||||||
|
|
||||||
searcher = LeannSearcher(index_path_str)
|
|
||||||
query = "what is LEANN?"
|
|
||||||
if args.add_incremental:
|
|
||||||
query = "what is the multi vector search and how it works?"
|
|
||||||
results = searcher.search(query, top_k=5)
|
|
||||||
if results:
|
|
||||||
print(f"Sample result: {results[0].text[:80]}...")
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
"""
|
|
||||||
Grep Search Example
|
|
||||||
|
|
||||||
Shows how to use grep-based text search instead of semantic search.
|
|
||||||
Useful when you need exact text matches rather than meaning-based results.
|
|
||||||
"""
|
|
||||||
|
|
||||||
from leann import LeannSearcher
|
|
||||||
|
|
||||||
# Load your index
|
|
||||||
searcher = LeannSearcher("my-documents.leann")
|
|
||||||
|
|
||||||
# Regular semantic search
|
|
||||||
print("=== Semantic Search ===")
|
|
||||||
results = searcher.search("machine learning algorithms", top_k=3)
|
|
||||||
for result in results:
|
|
||||||
print(f"Score: {result.score:.3f}")
|
|
||||||
print(f"Text: {result.text[:80]}...")
|
|
||||||
print()
|
|
||||||
|
|
||||||
# Grep-based search for exact text matches
|
|
||||||
print("=== Grep Search ===")
|
|
||||||
results = searcher.search("def train_model", top_k=3, use_grep=True)
|
|
||||||
for result in results:
|
|
||||||
print(f"Score: {result.score}")
|
|
||||||
print(f"Text: {result.text[:80]}...")
|
|
||||||
print()
|
|
||||||
|
|
||||||
# Find specific error messages
|
|
||||||
error_results = searcher.search("FileNotFoundError", use_grep=True)
|
|
||||||
print(f"Found {len(error_results)} files mentioning FileNotFoundError")
|
|
||||||
|
|
||||||
# Search for function definitions
|
|
||||||
func_results = searcher.search("class SearchResult", use_grep=True, top_k=5)
|
|
||||||
print(f"Found {len(func_results)} class definitions")
|
|
||||||
@@ -1,250 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
"""
|
|
||||||
Spoiler-Free Book RAG Example using LEANN Metadata Filtering
|
|
||||||
|
|
||||||
This example demonstrates how to use LEANN's metadata filtering to create
|
|
||||||
a spoiler-free book RAG system where users can search for information
|
|
||||||
up to a specific chapter they've read.
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
python spoiler_free_book_rag.py
|
|
||||||
"""
|
|
||||||
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
from typing import Any, Optional
|
|
||||||
|
|
||||||
# Add LEANN to path (adjust path as needed)
|
|
||||||
sys.path.insert(0, os.path.join(os.path.dirname(__file__), "../packages/leann-core/src"))
|
|
||||||
|
|
||||||
from leann.api import LeannBuilder, LeannSearcher
|
|
||||||
|
|
||||||
|
|
||||||
def chunk_book_with_metadata(book_title: str = "Sample Book") -> list[dict[str, Any]]:
|
|
||||||
"""
|
|
||||||
Create sample book chunks with metadata for demonstration.
|
|
||||||
|
|
||||||
In a real implementation, this would parse actual book files (epub, txt, etc.)
|
|
||||||
and extract chapter boundaries, character mentions, etc.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
book_title: Title of the book
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List of chunk dictionaries with text and metadata
|
|
||||||
"""
|
|
||||||
# Sample book chunks with metadata
|
|
||||||
# In practice, you'd use proper text processing libraries
|
|
||||||
|
|
||||||
sample_chunks = [
|
|
||||||
{
|
|
||||||
"text": "Alice was beginning to get very tired of sitting by her sister on the bank, and of having nothing to do.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 1,
|
|
||||||
"page": 1,
|
|
||||||
"characters": ["Alice", "Sister"],
|
|
||||||
"themes": ["boredom", "curiosity"],
|
|
||||||
"location": "riverbank",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"text": "So she was considering in her own mind (as well as she could, for the hot day made her feel very sleepy and stupid), whether the pleasure of making a daisy-chain would be worth the trouble of getting up and picking the daisies, when suddenly a White Rabbit with pink eyes ran close by her.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 1,
|
|
||||||
"page": 2,
|
|
||||||
"characters": ["Alice", "White Rabbit"],
|
|
||||||
"themes": ["decision", "surprise", "magic"],
|
|
||||||
"location": "riverbank",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"text": "Alice found herself falling down a very deep well. Either the well was very deep, or she fell very slowly, for she had plenty of time as she fell to look about her and to wonder what was going to happen next.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 2,
|
|
||||||
"page": 15,
|
|
||||||
"characters": ["Alice"],
|
|
||||||
"themes": ["falling", "wonder", "transformation"],
|
|
||||||
"location": "rabbit hole",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"text": "Alice meets the Cheshire Cat, who tells her that everyone in Wonderland is mad, including Alice herself.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 6,
|
|
||||||
"page": 85,
|
|
||||||
"characters": ["Alice", "Cheshire Cat"],
|
|
||||||
"themes": ["madness", "philosophy", "identity"],
|
|
||||||
"location": "Duchess's house",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"text": "At the Queen's croquet ground, Alice witnesses the absurd trial that reveals the arbitrary nature of Wonderland's justice system.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 8,
|
|
||||||
"page": 120,
|
|
||||||
"characters": ["Alice", "Queen of Hearts", "King of Hearts"],
|
|
||||||
"themes": ["justice", "absurdity", "authority"],
|
|
||||||
"location": "Queen's court",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"text": "Alice realizes that Wonderland was all a dream, even the Rabbit, as she wakes up on the riverbank next to her sister.",
|
|
||||||
"metadata": {
|
|
||||||
"book": book_title,
|
|
||||||
"chapter": 12,
|
|
||||||
"page": 180,
|
|
||||||
"characters": ["Alice", "Sister", "Rabbit"],
|
|
||||||
"themes": ["revelation", "reality", "growth"],
|
|
||||||
"location": "riverbank",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
return sample_chunks
|
|
||||||
|
|
||||||
|
|
||||||
def build_spoiler_free_index(book_chunks: list[dict[str, Any]], index_name: str) -> str:
|
|
||||||
"""
|
|
||||||
Build a LEANN index with book chunks that include spoiler metadata.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
book_chunks: List of book chunks with metadata
|
|
||||||
index_name: Name for the index
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Path to the built index
|
|
||||||
"""
|
|
||||||
print(f"📚 Building spoiler-free book index: {index_name}")
|
|
||||||
|
|
||||||
# Initialize LEANN builder
|
|
||||||
builder = LeannBuilder(
|
|
||||||
backend_name="hnsw", embedding_model="text-embedding-3-small", embedding_mode="openai"
|
|
||||||
)
|
|
||||||
|
|
||||||
# Add each chunk with its metadata
|
|
||||||
for chunk in book_chunks:
|
|
||||||
builder.add_text(text=chunk["text"], metadata=chunk["metadata"])
|
|
||||||
|
|
||||||
# Build the index
|
|
||||||
index_path = f"{index_name}_book_index"
|
|
||||||
builder.build_index(index_path)
|
|
||||||
|
|
||||||
print(f"✅ Index built successfully: {index_path}")
|
|
||||||
return index_path
|
|
||||||
|
|
||||||
|
|
||||||
def spoiler_free_search(
|
|
||||||
index_path: str,
|
|
||||||
query: str,
|
|
||||||
max_chapter: int,
|
|
||||||
character_filter: Optional[list[str]] = None,
|
|
||||||
) -> list[dict[str, Any]]:
|
|
||||||
"""
|
|
||||||
Perform a spoiler-free search on the book index.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
index_path: Path to the LEANN index
|
|
||||||
query: Search query
|
|
||||||
max_chapter: Maximum chapter number to include
|
|
||||||
character_filter: Optional list of characters to focus on
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List of search results safe for the reader
|
|
||||||
"""
|
|
||||||
print(f"🔍 Searching: '{query}' (up to chapter {max_chapter})")
|
|
||||||
|
|
||||||
searcher = LeannSearcher(index_path)
|
|
||||||
|
|
||||||
metadata_filters = {"chapter": {"<=": max_chapter}}
|
|
||||||
|
|
||||||
if character_filter:
|
|
||||||
metadata_filters["characters"] = {"contains": character_filter[0]}
|
|
||||||
|
|
||||||
results = searcher.search(query=query, top_k=10, metadata_filters=metadata_filters)
|
|
||||||
|
|
||||||
return results
|
|
||||||
|
|
||||||
|
|
||||||
def demo_spoiler_free_rag():
|
|
||||||
"""
|
|
||||||
Demonstrate the spoiler-free book RAG system.
|
|
||||||
"""
|
|
||||||
print("🎭 Spoiler-Free Book RAG Demo")
|
|
||||||
print("=" * 40)
|
|
||||||
|
|
||||||
# Step 1: Prepare book data
|
|
||||||
book_title = "Alice's Adventures in Wonderland"
|
|
||||||
book_chunks = chunk_book_with_metadata(book_title)
|
|
||||||
|
|
||||||
print(f"📖 Loaded {len(book_chunks)} chunks from '{book_title}'")
|
|
||||||
|
|
||||||
# Step 2: Build the index (in practice, this would be done once)
|
|
||||||
try:
|
|
||||||
index_path = build_spoiler_free_index(book_chunks, "alice_wonderland")
|
|
||||||
except Exception as e:
|
|
||||||
print(f"❌ Failed to build index (likely missing dependencies): {e}")
|
|
||||||
print(
|
|
||||||
"💡 This demo shows the filtering logic - actual indexing requires LEANN dependencies"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# Step 3: Demonstrate various spoiler-free searches
|
|
||||||
search_scenarios = [
|
|
||||||
{
|
|
||||||
"description": "Reader who has only read Chapter 1",
|
|
||||||
"query": "What can you tell me about the rabbit?",
|
|
||||||
"max_chapter": 1,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Reader who has read up to Chapter 5",
|
|
||||||
"query": "Tell me about Alice's adventures",
|
|
||||||
"max_chapter": 5,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Reader who has read most of the book",
|
|
||||||
"query": "What does the Cheshire Cat represent?",
|
|
||||||
"max_chapter": 10,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"description": "Reader who has read the whole book",
|
|
||||||
"query": "What can you tell me about the rabbit?",
|
|
||||||
"max_chapter": 12,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
for scenario in search_scenarios:
|
|
||||||
print(f"\n📚 Scenario: {scenario['description']}")
|
|
||||||
print(f" Query: {scenario['query']}")
|
|
||||||
|
|
||||||
try:
|
|
||||||
results = spoiler_free_search(
|
|
||||||
index_path=index_path,
|
|
||||||
query=scenario["query"],
|
|
||||||
max_chapter=scenario["max_chapter"],
|
|
||||||
)
|
|
||||||
|
|
||||||
print(f" 📄 Found {len(results)} results:")
|
|
||||||
for i, result in enumerate(results[:3], 1): # Show top 3
|
|
||||||
chapter = result.metadata.get("chapter", "?")
|
|
||||||
location = result.metadata.get("location", "?")
|
|
||||||
print(f" {i}. Chapter {chapter} ({location}): {result.text[:80]}...")
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
print(f" ❌ Search failed: {e}")
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
print("📚 LEANN Spoiler-Free Book RAG Example")
|
|
||||||
print("=====================================")
|
|
||||||
|
|
||||||
try:
|
|
||||||
demo_spoiler_free_rag()
|
|
||||||
except ImportError as e:
|
|
||||||
print(f"❌ Cannot run demo due to missing dependencies: {e}")
|
|
||||||
except Exception as e:
|
|
||||||
print(f"❌ Error running demo: {e}")
|
|
||||||
28
llms.txt
28
llms.txt
@@ -1,28 +0,0 @@
|
|||||||
# llms.txt — LEANN MCP and Agent Integration
|
|
||||||
product: LEANN
|
|
||||||
homepage: https://github.com/yichuan-w/LEANN
|
|
||||||
contact: https://github.com/yichuan-w/LEANN/issues
|
|
||||||
|
|
||||||
# Installation
|
|
||||||
install: uv tool install leann-core --with leann
|
|
||||||
|
|
||||||
# MCP Server Entry Point
|
|
||||||
mcp.server: leann_mcp
|
|
||||||
mcp.protocol_version: 2024-11-05
|
|
||||||
|
|
||||||
# Tools
|
|
||||||
mcp.tools: leann_list, leann_search
|
|
||||||
|
|
||||||
mcp.tool.leann_list.description: List available LEANN indexes
|
|
||||||
mcp.tool.leann_list.input: {}
|
|
||||||
|
|
||||||
mcp.tool.leann_search.description: Semantic search across a named LEANN index
|
|
||||||
mcp.tool.leann_search.input.index_name: string, required
|
|
||||||
mcp.tool.leann_search.input.query: string, required
|
|
||||||
mcp.tool.leann_search.input.top_k: integer, optional, default=5, min=1, max=20
|
|
||||||
mcp.tool.leann_search.input.complexity: integer, optional, default=32, min=16, max=128
|
|
||||||
|
|
||||||
# Notes
|
|
||||||
note: Build indexes with `leann build <name> --docs <files...>` before searching.
|
|
||||||
example.add: claude mcp add --scope user leann-server -- leann_mcp
|
|
||||||
example.verify: claude mcp list | cat
|
|
||||||
Submodule packages/astchunk-leann deleted from ad9afa07b9
@@ -83,7 +83,9 @@ def create_diskann_embedding_server(
|
|||||||
|
|
||||||
logger.info(f"Loading PassageManager with metadata_file_path: {passages_file}")
|
logger.info(f"Loading PassageManager with metadata_file_path: {passages_file}")
|
||||||
passages = PassageManager(meta["passage_sources"], metadata_file_path=passages_file)
|
passages = PassageManager(meta["passage_sources"], metadata_file_path=passages_file)
|
||||||
logger.info(f"Loaded PassageManager with {len(passages)} passages from metadata")
|
logger.info(
|
||||||
|
f"Loaded PassageManager with {len(passages.global_offset_map)} passages from metadata"
|
||||||
|
)
|
||||||
|
|
||||||
# Import protobuf after ensuring the path is correct
|
# Import protobuf after ensuring the path is correct
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ build-backend = "scikit_build_core.build"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "leann-backend-diskann"
|
name = "leann-backend-diskann"
|
||||||
version = "0.3.4"
|
version = "0.3.0"
|
||||||
dependencies = ["leann-core==0.3.4", "numpy", "protobuf>=3.19.0"]
|
dependencies = ["leann-core==0.3.0", "numpy", "protobuf>=3.19.0"]
|
||||||
|
|
||||||
[tool.scikit-build]
|
[tool.scikit-build]
|
||||||
# Key: simplified CMake path
|
# Key: simplified CMake path
|
||||||
|
|||||||
Submodule packages/leann-backend-diskann/third_party/DiskANN updated: 19f9603c72...04048bb302
@@ -49,28 +49,9 @@ set(BUILD_TESTING OFF CACHE BOOL "" FORCE)
|
|||||||
set(FAISS_ENABLE_C_API OFF CACHE BOOL "" FORCE)
|
set(FAISS_ENABLE_C_API OFF CACHE BOOL "" FORCE)
|
||||||
set(FAISS_OPT_LEVEL "generic" CACHE STRING "" FORCE)
|
set(FAISS_OPT_LEVEL "generic" CACHE STRING "" FORCE)
|
||||||
|
|
||||||
# Disable x86-specific SIMD optimizations (important for ARM64 compatibility)
|
# Disable additional SIMD versions to speed up compilation
|
||||||
set(FAISS_ENABLE_AVX2 OFF CACHE BOOL "" FORCE)
|
set(FAISS_ENABLE_AVX2 OFF CACHE BOOL "" FORCE)
|
||||||
set(FAISS_ENABLE_AVX512 OFF CACHE BOOL "" FORCE)
|
set(FAISS_ENABLE_AVX512 OFF CACHE BOOL "" FORCE)
|
||||||
set(FAISS_ENABLE_SSE4_1 OFF CACHE BOOL "" FORCE)
|
|
||||||
|
|
||||||
# ARM64-specific configuration
|
|
||||||
if(CMAKE_SYSTEM_PROCESSOR MATCHES "aarch64|arm64")
|
|
||||||
message(STATUS "Configuring Faiss for ARM64 architecture")
|
|
||||||
|
|
||||||
if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
|
||||||
# Use SVE optimization level for ARM64 Linux (as seen in Faiss conda build)
|
|
||||||
set(FAISS_OPT_LEVEL "sve" CACHE STRING "" FORCE)
|
|
||||||
message(STATUS "Setting FAISS_OPT_LEVEL to 'sve' for ARM64 Linux")
|
|
||||||
else()
|
|
||||||
# Use generic optimization for other ARM64 platforms (like macOS)
|
|
||||||
set(FAISS_OPT_LEVEL "generic" CACHE STRING "" FORCE)
|
|
||||||
message(STATUS "Setting FAISS_OPT_LEVEL to 'generic' for ARM64 ${CMAKE_SYSTEM_NAME}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# ARM64 compatibility: Faiss submodule has been modified to fix x86 header inclusion
|
|
||||||
message(STATUS "Using ARM64-compatible Faiss submodule")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Additional optimization options from INSTALL.md
|
# Additional optimization options from INSTALL.md
|
||||||
set(CMAKE_BUILD_TYPE "Release" CACHE STRING "" FORCE)
|
set(CMAKE_BUILD_TYPE "Release" CACHE STRING "" FORCE)
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import time
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Any, Literal, Optional
|
from typing import Any, Literal, Optional
|
||||||
|
|
||||||
@@ -15,7 +14,6 @@ from leann.registry import register_backend
|
|||||||
from leann.searcher_base import BaseSearcher
|
from leann.searcher_base import BaseSearcher
|
||||||
|
|
||||||
from .convert_to_csr import convert_hnsw_graph_to_csr
|
from .convert_to_csr import convert_hnsw_graph_to_csr
|
||||||
from .prune_index import prune_embeddings_preserve_graph_inplace
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -91,16 +89,8 @@ class HNSWBuilder(LeannBackendBuilderInterface):
|
|||||||
index_file = index_dir / f"{index_prefix}.index"
|
index_file = index_dir / f"{index_prefix}.index"
|
||||||
faiss.write_index(index, str(index_file))
|
faiss.write_index(index, str(index_file))
|
||||||
|
|
||||||
if self.is_recompute:
|
if self.is_compact:
|
||||||
if self.is_compact:
|
self._convert_to_csr(index_file)
|
||||||
self._convert_to_csr(index_file)
|
|
||||||
else:
|
|
||||||
# Non-compact format: prune only embeddings, keep original graph
|
|
||||||
ok = prune_embeddings_preserve_graph_inplace(str(index_file))
|
|
||||||
if not ok:
|
|
||||||
raise RuntimeError(
|
|
||||||
"Pruning embeddings while preserving graph failed for non-compact index"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _convert_to_csr(self, index_file: Path):
|
def _convert_to_csr(self, index_file: Path):
|
||||||
"""Convert built index to CSR format"""
|
"""Convert built index to CSR format"""
|
||||||
@@ -157,13 +147,7 @@ class HNSWSearcher(BaseSearcher):
|
|||||||
self.is_pruned
|
self.is_pruned
|
||||||
) # In C++ code, it's called is_recompute, but it's only for loading IIUC.
|
) # In C++ code, it's called is_recompute, but it's only for loading IIUC.
|
||||||
|
|
||||||
# If pruned (recompute mode), explicitly skip storage to avoid reading
|
self._index = faiss.read_index(str(index_file), faiss.IO_FLAG_MMAP, hnsw_config)
|
||||||
# the pruned section. Still allow MMAP for graph.
|
|
||||||
io_flags = faiss.IO_FLAG_MMAP
|
|
||||||
if self.is_pruned:
|
|
||||||
io_flags |= faiss.IO_FLAG_SKIP_STORAGE
|
|
||||||
|
|
||||||
self._index = faiss.read_index(str(index_file), io_flags, hnsw_config)
|
|
||||||
|
|
||||||
def search(
|
def search(
|
||||||
self,
|
self,
|
||||||
@@ -252,7 +236,6 @@ class HNSWSearcher(BaseSearcher):
|
|||||||
distances = np.empty((batch_size_query, top_k), dtype=np.float32)
|
distances = np.empty((batch_size_query, top_k), dtype=np.float32)
|
||||||
labels = np.empty((batch_size_query, top_k), dtype=np.int64)
|
labels = np.empty((batch_size_query, top_k), dtype=np.int64)
|
||||||
|
|
||||||
search_time = time.time()
|
|
||||||
self._index.search(
|
self._index.search(
|
||||||
query.shape[0],
|
query.shape[0],
|
||||||
faiss.swig_ptr(query),
|
faiss.swig_ptr(query),
|
||||||
@@ -261,60 +244,7 @@ class HNSWSearcher(BaseSearcher):
|
|||||||
faiss.swig_ptr(labels),
|
faiss.swig_ptr(labels),
|
||||||
params,
|
params,
|
||||||
)
|
)
|
||||||
search_time = time.time() - search_time
|
|
||||||
logger.info(f" Search time in HNSWSearcher.search() backend: {search_time} seconds")
|
|
||||||
string_labels = [[str(int_label) for int_label in batch_labels] for batch_labels in labels]
|
string_labels = [[str(int_label) for int_label in batch_labels] for batch_labels in labels]
|
||||||
|
|
||||||
return {"labels": string_labels, "distances": distances}
|
return {"labels": string_labels, "distances": distances}
|
||||||
|
|
||||||
|
|
||||||
# ---------- Helper API for incremental add (Python-level) ----------
|
|
||||||
def add_vectors(
|
|
||||||
index_file_path: str,
|
|
||||||
embeddings: np.ndarray,
|
|
||||||
*,
|
|
||||||
ef_construction: Optional[int] = None,
|
|
||||||
recompute: bool = False,
|
|
||||||
) -> None:
|
|
||||||
"""Append vectors to an existing non-compact HNSW index.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
index_file_path: Path to the HNSW .index file
|
|
||||||
embeddings: float32 numpy array (N, D)
|
|
||||||
ef_construction: Optional override for efConstruction during insertion
|
|
||||||
recompute: Reserved for future use to control insertion-time recompute behaviors
|
|
||||||
"""
|
|
||||||
from . import faiss # type: ignore
|
|
||||||
|
|
||||||
if embeddings.dtype != np.float32:
|
|
||||||
embeddings = embeddings.astype(np.float32)
|
|
||||||
if not embeddings.flags.c_contiguous:
|
|
||||||
embeddings = np.ascontiguousarray(embeddings, dtype=np.float32)
|
|
||||||
|
|
||||||
# Load index normally to ensure storage is present; toggle is_recompute on the object
|
|
||||||
index = faiss.read_index(str(index_file_path), faiss.IO_FLAG_MMAP)
|
|
||||||
|
|
||||||
# Best-effort: explicitly set flag on the object if the binding exposes it
|
|
||||||
try:
|
|
||||||
index.is_recompute = bool(recompute)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
try:
|
|
||||||
if ef_construction is not None:
|
|
||||||
index.hnsw.efConstruction = int(ef_construction)
|
|
||||||
except Exception:
|
|
||||||
# Best-effort; ignore if backend doesn't expose setter
|
|
||||||
pass
|
|
||||||
|
|
||||||
# For non-compact HNSW, calling add directly is sufficient. When is_recompute is set
|
|
||||||
# (via config or attribute), FAISS will run the insertion/search path accordingly.
|
|
||||||
# To strictly follow per-point insert semantics in recompute mode, add one-by-one.
|
|
||||||
if recompute:
|
|
||||||
# Insert row by row
|
|
||||||
n = embeddings.shape[0]
|
|
||||||
for i in range(n):
|
|
||||||
row = embeddings[i : i + 1]
|
|
||||||
index.add(1, faiss.swig_ptr(row))
|
|
||||||
else:
|
|
||||||
index.add(embeddings.shape[0], faiss.swig_ptr(embeddings))
|
|
||||||
faiss.write_index(index, str(index_file_path))
|
|
||||||
|
|||||||
@@ -90,7 +90,9 @@ def create_hnsw_embedding_server(
|
|||||||
embedding_dim: int = int(meta.get("dimensions", 0))
|
embedding_dim: int = int(meta.get("dimensions", 0))
|
||||||
except Exception:
|
except Exception:
|
||||||
embedding_dim = 0
|
embedding_dim = 0
|
||||||
logger.info(f"Loaded PassageManager with {len(passages)} passages from metadata")
|
logger.info(
|
||||||
|
f"Loaded PassageManager with {len(passages.global_offset_map)} passages from metadata"
|
||||||
|
)
|
||||||
|
|
||||||
# (legacy ZMQ thread removed; using shutdown-capable server only)
|
# (legacy ZMQ thread removed; using shutdown-capable server only)
|
||||||
|
|
||||||
|
|||||||
@@ -1,149 +0,0 @@
|
|||||||
import os
|
|
||||||
import struct
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from .convert_to_csr import (
|
|
||||||
EXPECTED_HNSW_FOURCCS,
|
|
||||||
NULL_INDEX_FOURCC,
|
|
||||||
read_struct,
|
|
||||||
read_vector_raw,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _write_vector_raw(f_out, count: int, data_bytes: bytes) -> None:
|
|
||||||
"""Write a vector in the same binary layout as read_vector_raw reads: <Q count> + raw bytes."""
|
|
||||||
f_out.write(struct.pack("<Q", count))
|
|
||||||
if count > 0 and data_bytes:
|
|
||||||
f_out.write(data_bytes)
|
|
||||||
|
|
||||||
|
|
||||||
def prune_embeddings_preserve_graph(input_filename: str, output_filename: str) -> bool:
|
|
||||||
"""
|
|
||||||
Copy an original (non-compact) HNSW index file while pruning the trailing embedding storage.
|
|
||||||
Preserves the graph structure and metadata exactly; only writes a NULL storage marker instead of
|
|
||||||
the original storage fourcc and payload.
|
|
||||||
|
|
||||||
Returns True on success.
|
|
||||||
"""
|
|
||||||
print(f"Pruning embeddings from {input_filename} to {output_filename}")
|
|
||||||
print("--------------------------------")
|
|
||||||
# running in mode is-recompute=True and is-compact=False
|
|
||||||
in_path = Path(input_filename)
|
|
||||||
out_path = Path(output_filename)
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(in_path, "rb") as f_in, open(out_path, "wb") as f_out:
|
|
||||||
# Header
|
|
||||||
index_fourcc = read_struct(f_in, "<I")
|
|
||||||
if index_fourcc not in EXPECTED_HNSW_FOURCCS:
|
|
||||||
# Still proceed, but this is unexpected
|
|
||||||
pass
|
|
||||||
f_out.write(struct.pack("<I", index_fourcc))
|
|
||||||
|
|
||||||
d = read_struct(f_in, "<i")
|
|
||||||
ntotal_hdr = read_struct(f_in, "<q")
|
|
||||||
dummy1 = read_struct(f_in, "<q")
|
|
||||||
dummy2 = read_struct(f_in, "<q")
|
|
||||||
is_trained = read_struct(f_in, "?")
|
|
||||||
metric_type = read_struct(f_in, "<i")
|
|
||||||
f_out.write(struct.pack("<i", d))
|
|
||||||
f_out.write(struct.pack("<q", ntotal_hdr))
|
|
||||||
f_out.write(struct.pack("<q", dummy1))
|
|
||||||
f_out.write(struct.pack("<q", dummy2))
|
|
||||||
f_out.write(struct.pack("<?", is_trained))
|
|
||||||
f_out.write(struct.pack("<i", metric_type))
|
|
||||||
|
|
||||||
if metric_type > 1:
|
|
||||||
metric_arg = read_struct(f_in, "<f")
|
|
||||||
f_out.write(struct.pack("<f", metric_arg))
|
|
||||||
|
|
||||||
# Vectors: assign_probas (double), cum_nneighbor_per_level (int32), levels (int32)
|
|
||||||
cnt, data = read_vector_raw(f_in, "d")
|
|
||||||
_write_vector_raw(f_out, cnt, data)
|
|
||||||
|
|
||||||
cnt, data = read_vector_raw(f_in, "i")
|
|
||||||
_write_vector_raw(f_out, cnt, data)
|
|
||||||
|
|
||||||
cnt, data = read_vector_raw(f_in, "i")
|
|
||||||
_write_vector_raw(f_out, cnt, data)
|
|
||||||
|
|
||||||
# Probe potential extra alignment/flag byte present in some original formats
|
|
||||||
probe = f_in.read(1)
|
|
||||||
if probe:
|
|
||||||
if probe == b"\x00":
|
|
||||||
# Preserve this unexpected 0x00 byte
|
|
||||||
f_out.write(probe)
|
|
||||||
else:
|
|
||||||
# Likely part of the next vector; rewind
|
|
||||||
f_in.seek(-1, os.SEEK_CUR)
|
|
||||||
|
|
||||||
# Offsets (uint64) and neighbors (int32)
|
|
||||||
cnt, data = read_vector_raw(f_in, "Q")
|
|
||||||
_write_vector_raw(f_out, cnt, data)
|
|
||||||
|
|
||||||
cnt, data = read_vector_raw(f_in, "i")
|
|
||||||
_write_vector_raw(f_out, cnt, data)
|
|
||||||
|
|
||||||
# Scalar params
|
|
||||||
entry_point = read_struct(f_in, "<i")
|
|
||||||
max_level = read_struct(f_in, "<i")
|
|
||||||
ef_construction = read_struct(f_in, "<i")
|
|
||||||
ef_search = read_struct(f_in, "<i")
|
|
||||||
dummy_upper_beam = read_struct(f_in, "<i")
|
|
||||||
f_out.write(struct.pack("<i", entry_point))
|
|
||||||
f_out.write(struct.pack("<i", max_level))
|
|
||||||
f_out.write(struct.pack("<i", ef_construction))
|
|
||||||
f_out.write(struct.pack("<i", ef_search))
|
|
||||||
f_out.write(struct.pack("<i", dummy_upper_beam))
|
|
||||||
|
|
||||||
# Storage fourcc (if present) — write NULL marker and drop any remaining data
|
|
||||||
try:
|
|
||||||
read_struct(f_in, "<I")
|
|
||||||
# Regardless of original, write NULL
|
|
||||||
f_out.write(struct.pack("<I", NULL_INDEX_FOURCC))
|
|
||||||
# Discard the rest of the file (embedding payload)
|
|
||||||
# (Do not copy anything else)
|
|
||||||
except EOFError:
|
|
||||||
# No storage section; nothing else to write
|
|
||||||
pass
|
|
||||||
|
|
||||||
return True
|
|
||||||
except Exception:
|
|
||||||
# Best-effort cleanup
|
|
||||||
try:
|
|
||||||
if out_path.exists():
|
|
||||||
out_path.unlink()
|
|
||||||
except OSError:
|
|
||||||
pass
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def prune_embeddings_preserve_graph_inplace(index_file_path: str) -> bool:
|
|
||||||
"""
|
|
||||||
Convenience wrapper: write pruned file to a temporary path next to the
|
|
||||||
original, then atomically replace on success.
|
|
||||||
"""
|
|
||||||
print(f"Pruning embeddings from {index_file_path} to {index_file_path}")
|
|
||||||
print("--------------------------------")
|
|
||||||
# running in mode is-recompute=True and is-compact=False
|
|
||||||
src = Path(index_file_path)
|
|
||||||
tmp = src.with_suffix(".pruned.tmp")
|
|
||||||
ok = prune_embeddings_preserve_graph(str(src), str(tmp))
|
|
||||||
if not ok:
|
|
||||||
if tmp.exists():
|
|
||||||
try:
|
|
||||||
tmp.unlink()
|
|
||||||
except OSError:
|
|
||||||
pass
|
|
||||||
return False
|
|
||||||
try:
|
|
||||||
os.replace(str(tmp), str(src))
|
|
||||||
except Exception:
|
|
||||||
# Rollback on failure
|
|
||||||
try:
|
|
||||||
if tmp.exists():
|
|
||||||
tmp.unlink()
|
|
||||||
except OSError:
|
|
||||||
pass
|
|
||||||
return False
|
|
||||||
return True
|
|
||||||
@@ -6,10 +6,10 @@ build-backend = "scikit_build_core.build"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "leann-backend-hnsw"
|
name = "leann-backend-hnsw"
|
||||||
version = "0.3.4"
|
version = "0.3.0"
|
||||||
description = "Custom-built HNSW (Faiss) backend for the Leann toolkit."
|
description = "Custom-built HNSW (Faiss) backend for the Leann toolkit."
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"leann-core==0.3.4",
|
"leann-core==0.3.0",
|
||||||
"numpy",
|
"numpy",
|
||||||
"pyzmq>=23.0.0",
|
"pyzmq>=23.0.0",
|
||||||
"msgpack>=1.0.0",
|
"msgpack>=1.0.0",
|
||||||
|
|||||||
Submodule packages/leann-backend-hnsw/third_party/faiss updated: ea86d06ceb...4a2c0d67d3
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "leann-core"
|
name = "leann-core"
|
||||||
version = "0.3.4"
|
version = "0.3.0"
|
||||||
description = "Core API and plugin system for LEANN"
|
description = "Core API and plugin system for LEANN"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
requires-python = ">=3.9"
|
requires-python = ">=3.9"
|
||||||
|
|||||||
@@ -5,24 +5,19 @@ with the correct, original embedding logic from the user's reference code.
|
|||||||
|
|
||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
import os
|
|
||||||
import pickle
|
import pickle
|
||||||
import re
|
|
||||||
import subprocess
|
|
||||||
import time
|
import time
|
||||||
import warnings
|
import warnings
|
||||||
from dataclasses import dataclass, field
|
from dataclasses import dataclass, field
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Any, Literal, Optional, Union
|
from typing import Any, Literal, Optional
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
|
|
||||||
from leann.interface import LeannBackendSearcherInterface
|
from leann.interface import LeannBackendSearcherInterface
|
||||||
|
|
||||||
from .chat import get_llm
|
from .chat import get_llm
|
||||||
from .embedding_server_manager import EmbeddingServerManager
|
|
||||||
from .interface import LeannBackendFactoryInterface
|
from .interface import LeannBackendFactoryInterface
|
||||||
from .metadata_filter import MetadataFilterEngine
|
|
||||||
from .registry import BACKEND_REGISTRY
|
from .registry import BACKEND_REGISTRY
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@@ -120,31 +115,13 @@ class SearchResult:
|
|||||||
metadata: dict[str, Any] = field(default_factory=dict)
|
metadata: dict[str, Any] = field(default_factory=dict)
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class IncrementalAddContext:
|
|
||||||
"""Prepared context for safe incremental add operations on an index."""
|
|
||||||
|
|
||||||
index_path: str
|
|
||||||
passages_file: Path
|
|
||||||
offsets_file: Path
|
|
||||||
vector_index_file: Path
|
|
||||||
embedding_model: str
|
|
||||||
embedding_mode: str
|
|
||||||
distance_metric: str
|
|
||||||
prepared_texts: Optional[list[str]] = None
|
|
||||||
|
|
||||||
|
|
||||||
class PassageManager:
|
class PassageManager:
|
||||||
def __init__(
|
def __init__(
|
||||||
self, passage_sources: list[dict[str, Any]], metadata_file_path: Optional[str] = None
|
self, passage_sources: list[dict[str, Any]], metadata_file_path: Optional[str] = None
|
||||||
):
|
):
|
||||||
self.offset_maps: dict[str, dict[str, int]] = {}
|
self.offset_maps = {}
|
||||||
self.passage_files: dict[str, str] = {}
|
self.passage_files = {}
|
||||||
# Avoid materializing a single gigantic global map to reduce memory
|
self.global_offset_map = {} # Combined map for fast lookup
|
||||||
# footprint on very large corpora (e.g., 60M+ passages). Instead, keep
|
|
||||||
# per-shard maps and do a lightweight per-shard lookup on demand.
|
|
||||||
self._total_count: int = 0
|
|
||||||
self.filter_engine = MetadataFilterEngine() # Initialize filter engine
|
|
||||||
|
|
||||||
# Derive index base name for standard sibling fallbacks, e.g., <index_name>.passages.*
|
# Derive index base name for standard sibling fallbacks, e.g., <index_name>.passages.*
|
||||||
index_name_base = None
|
index_name_base = None
|
||||||
@@ -165,25 +142,12 @@ class PassageManager:
|
|||||||
default_name: Optional[str],
|
default_name: Optional[str],
|
||||||
source_dict: dict[str, Any],
|
source_dict: dict[str, Any],
|
||||||
) -> list[Path]:
|
) -> list[Path]:
|
||||||
"""
|
|
||||||
Build an ordered list of candidate paths. For relative paths specified in
|
|
||||||
metadata, prefer resolution relative to the metadata file directory first,
|
|
||||||
then fall back to CWD-based resolution, and finally to conventional
|
|
||||||
sibling defaults (e.g., <index_base>.passages.idx / .jsonl).
|
|
||||||
"""
|
|
||||||
candidates: list[Path] = []
|
candidates: list[Path] = []
|
||||||
# 1) Primary path
|
# 1) Primary as-is (absolute or relative)
|
||||||
if primary:
|
if primary:
|
||||||
p = Path(primary)
|
p = Path(primary)
|
||||||
if p.is_absolute():
|
candidates.append(p if p.is_absolute() else (Path.cwd() / p))
|
||||||
candidates.append(p)
|
# 2) metadata-relative explicit relative key
|
||||||
else:
|
|
||||||
# Prefer metadata-relative resolution for relative paths
|
|
||||||
if metadata_file_path:
|
|
||||||
candidates.append(Path(metadata_file_path).parent / p)
|
|
||||||
# Also consider CWD-relative as a fallback for legacy layouts
|
|
||||||
candidates.append(Path.cwd() / p)
|
|
||||||
# 2) metadata-relative explicit relative key (if present)
|
|
||||||
if metadata_file_path and source_dict.get(relative_key):
|
if metadata_file_path and source_dict.get(relative_key):
|
||||||
candidates.append(Path(metadata_file_path).parent / source_dict[relative_key])
|
candidates.append(Path(metadata_file_path).parent / source_dict[relative_key])
|
||||||
# 3) metadata-relative standard sibling filename
|
# 3) metadata-relative standard sibling filename
|
||||||
@@ -213,78 +177,23 @@ class PassageManager:
|
|||||||
raise FileNotFoundError(f"Passage index file not found: {index_file}")
|
raise FileNotFoundError(f"Passage index file not found: {index_file}")
|
||||||
|
|
||||||
with open(index_file, "rb") as f:
|
with open(index_file, "rb") as f:
|
||||||
offset_map: dict[str, int] = pickle.load(f)
|
offset_map = pickle.load(f)
|
||||||
self.offset_maps[passage_file] = offset_map
|
self.offset_maps[passage_file] = offset_map
|
||||||
self.passage_files[passage_file] = passage_file
|
self.passage_files[passage_file] = passage_file
|
||||||
self._total_count += len(offset_map)
|
|
||||||
|
# Build global map for O(1) lookup
|
||||||
|
for passage_id, offset in offset_map.items():
|
||||||
|
self.global_offset_map[passage_id] = (passage_file, offset)
|
||||||
|
|
||||||
def get_passage(self, passage_id: str) -> dict[str, Any]:
|
def get_passage(self, passage_id: str) -> dict[str, Any]:
|
||||||
# Fast path: check each shard map (there are typically few shards).
|
if passage_id in self.global_offset_map:
|
||||||
# This avoids building a massive combined dict while keeping lookups
|
passage_file, offset = self.global_offset_map[passage_id]
|
||||||
# bounded by the number of shards.
|
# Lazy file opening - only open when needed
|
||||||
for passage_file, offset_map in self.offset_maps.items():
|
with open(passage_file, encoding="utf-8") as f:
|
||||||
try:
|
f.seek(offset)
|
||||||
offset = offset_map[passage_id]
|
return json.loads(f.readline())
|
||||||
with open(passage_file, encoding="utf-8") as f:
|
|
||||||
f.seek(offset)
|
|
||||||
return json.loads(f.readline())
|
|
||||||
except KeyError:
|
|
||||||
continue
|
|
||||||
raise KeyError(f"Passage ID not found: {passage_id}")
|
raise KeyError(f"Passage ID not found: {passage_id}")
|
||||||
|
|
||||||
def filter_search_results(
|
|
||||||
self,
|
|
||||||
search_results: list[SearchResult],
|
|
||||||
metadata_filters: Optional[dict[str, dict[str, Union[str, int, float, bool, list]]]],
|
|
||||||
) -> list[SearchResult]:
|
|
||||||
"""
|
|
||||||
Apply metadata filters to search results.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
search_results: List of SearchResult objects
|
|
||||||
metadata_filters: Filter specifications to apply
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Filtered list of SearchResult objects
|
|
||||||
"""
|
|
||||||
if not metadata_filters:
|
|
||||||
return search_results
|
|
||||||
|
|
||||||
logger.debug(f"Applying metadata filters to {len(search_results)} results")
|
|
||||||
|
|
||||||
# Convert SearchResult objects to dictionaries for the filter engine
|
|
||||||
result_dicts = []
|
|
||||||
for result in search_results:
|
|
||||||
result_dicts.append(
|
|
||||||
{
|
|
||||||
"id": result.id,
|
|
||||||
"score": result.score,
|
|
||||||
"text": result.text,
|
|
||||||
"metadata": result.metadata,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
# Apply filters using the filter engine
|
|
||||||
filtered_dicts = self.filter_engine.apply_filters(result_dicts, metadata_filters)
|
|
||||||
|
|
||||||
# Convert back to SearchResult objects
|
|
||||||
filtered_results = []
|
|
||||||
for result_dict in filtered_dicts:
|
|
||||||
filtered_results.append(
|
|
||||||
SearchResult(
|
|
||||||
id=result_dict["id"],
|
|
||||||
score=result_dict["score"],
|
|
||||||
text=result_dict["text"],
|
|
||||||
metadata=result_dict["metadata"],
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
logger.debug(f"Filtered results: {len(filtered_results)} remaining")
|
|
||||||
return filtered_results
|
|
||||||
|
|
||||||
def __len__(self) -> int:
|
|
||||||
return self._total_count
|
|
||||||
|
|
||||||
|
|
||||||
class LeannBuilder:
|
class LeannBuilder:
|
||||||
def __init__(
|
def __init__(
|
||||||
@@ -492,7 +401,9 @@ class LeannBuilder:
|
|||||||
is_compact = self.backend_kwargs.get("is_compact", True)
|
is_compact = self.backend_kwargs.get("is_compact", True)
|
||||||
is_recompute = self.backend_kwargs.get("is_recompute", True)
|
is_recompute = self.backend_kwargs.get("is_recompute", True)
|
||||||
meta_data["is_compact"] = is_compact
|
meta_data["is_compact"] = is_compact
|
||||||
meta_data["is_pruned"] = is_recompute # Pruned only if compact and recompute
|
meta_data["is_pruned"] = (
|
||||||
|
is_compact and is_recompute
|
||||||
|
) # Pruned only if compact and recompute
|
||||||
with open(leann_meta_path, "w", encoding="utf-8") as f:
|
with open(leann_meta_path, "w", encoding="utf-8") as f:
|
||||||
json.dump(meta_data, f, indent=2)
|
json.dump(meta_data, f, indent=2)
|
||||||
|
|
||||||
@@ -646,8 +557,6 @@ class LeannSearcher:
|
|||||||
self.passage_manager = PassageManager(
|
self.passage_manager = PassageManager(
|
||||||
self.meta_data.get("passage_sources", []), metadata_file_path=self.meta_path_str
|
self.meta_data.get("passage_sources", []), metadata_file_path=self.meta_path_str
|
||||||
)
|
)
|
||||||
# Preserve backend name for conditional parameter forwarding
|
|
||||||
self.backend_name = backend_name
|
|
||||||
backend_factory = BACKEND_REGISTRY.get(backend_name)
|
backend_factory = BACKEND_REGISTRY.get(backend_name)
|
||||||
if backend_factory is None:
|
if backend_factory is None:
|
||||||
raise ValueError(f"Backend '{backend_name}' not found.")
|
raise ValueError(f"Backend '{backend_name}' not found.")
|
||||||
@@ -667,49 +576,15 @@ class LeannSearcher:
|
|||||||
recompute_embeddings: bool = True,
|
recompute_embeddings: bool = True,
|
||||||
pruning_strategy: Literal["global", "local", "proportional"] = "global",
|
pruning_strategy: Literal["global", "local", "proportional"] = "global",
|
||||||
expected_zmq_port: int = 5557,
|
expected_zmq_port: int = 5557,
|
||||||
metadata_filters: Optional[dict[str, dict[str, Union[str, int, float, bool, list]]]] = None,
|
|
||||||
batch_size: int = 0,
|
|
||||||
use_grep: bool = False,
|
|
||||||
**kwargs,
|
**kwargs,
|
||||||
) -> list[SearchResult]:
|
) -> list[SearchResult]:
|
||||||
"""
|
|
||||||
Search for nearest neighbors with optional metadata filtering.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
query: Text query to search for
|
|
||||||
top_k: Number of nearest neighbors to return
|
|
||||||
complexity: Search complexity/candidate list size, higher = more accurate but slower
|
|
||||||
beam_width: Number of parallel search paths/IO requests per iteration
|
|
||||||
prune_ratio: Ratio of neighbors to prune via approximate distance (0.0-1.0)
|
|
||||||
recompute_embeddings: Whether to fetch fresh embeddings from server vs use stored codes
|
|
||||||
pruning_strategy: Candidate selection strategy - "global" (default), "local", or "proportional"
|
|
||||||
expected_zmq_port: ZMQ port for embedding server communication
|
|
||||||
metadata_filters: Optional filters to apply to search results based on metadata.
|
|
||||||
Format: {"field_name": {"operator": value}}
|
|
||||||
Supported operators:
|
|
||||||
- Comparison: "==", "!=", "<", "<=", ">", ">="
|
|
||||||
- Membership: "in", "not_in"
|
|
||||||
- String: "contains", "starts_with", "ends_with"
|
|
||||||
Example: {"chapter": {"<=": 5}, "tags": {"in": ["fiction", "drama"]}}
|
|
||||||
**kwargs: Backend-specific parameters
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List of SearchResult objects with text, metadata, and similarity scores
|
|
||||||
"""
|
|
||||||
# Handle grep search
|
|
||||||
if use_grep:
|
|
||||||
return self._grep_search(query, top_k)
|
|
||||||
|
|
||||||
logger.info("🔍 LeannSearcher.search() called:")
|
logger.info("🔍 LeannSearcher.search() called:")
|
||||||
logger.info(f" Query: '{query}'")
|
logger.info(f" Query: '{query}'")
|
||||||
logger.info(f" Top_k: {top_k}")
|
logger.info(f" Top_k: {top_k}")
|
||||||
logger.info(f" Metadata filters: {metadata_filters}")
|
|
||||||
logger.info(f" Additional kwargs: {kwargs}")
|
logger.info(f" Additional kwargs: {kwargs}")
|
||||||
|
|
||||||
# Smart top_k detection and adjustment
|
# Smart top_k detection and adjustment
|
||||||
# Use PassageManager length (sum of shard sizes) to avoid
|
total_docs = len(self.passage_manager.global_offset_map)
|
||||||
# depending on a massive combined map
|
|
||||||
total_docs = len(self.passage_manager)
|
|
||||||
original_top_k = top_k
|
original_top_k = top_k
|
||||||
if top_k > total_docs:
|
if top_k > total_docs:
|
||||||
top_k = total_docs
|
top_k = total_docs
|
||||||
@@ -738,33 +613,23 @@ class LeannSearcher:
|
|||||||
use_server_if_available=recompute_embeddings,
|
use_server_if_available=recompute_embeddings,
|
||||||
zmq_port=zmq_port,
|
zmq_port=zmq_port,
|
||||||
)
|
)
|
||||||
logger.info(f" Generated embedding shape: {query_embedding.shape}")
|
# logger.info(f" Generated embedding shape: {query_embedding.shape}")
|
||||||
embedding_time = time.time() - start_time
|
time.time() - start_time
|
||||||
logger.info(f" Embedding time: {embedding_time} seconds")
|
# logger.info(f" Embedding time: {embedding_time} seconds")
|
||||||
|
|
||||||
start_time = time.time()
|
start_time = time.time()
|
||||||
backend_search_kwargs: dict[str, Any] = {
|
|
||||||
"complexity": complexity,
|
|
||||||
"beam_width": beam_width,
|
|
||||||
"prune_ratio": prune_ratio,
|
|
||||||
"recompute_embeddings": recompute_embeddings,
|
|
||||||
"pruning_strategy": pruning_strategy,
|
|
||||||
"zmq_port": zmq_port,
|
|
||||||
}
|
|
||||||
# Only HNSW supports batching; forward conditionally
|
|
||||||
if self.backend_name == "hnsw":
|
|
||||||
backend_search_kwargs["batch_size"] = batch_size
|
|
||||||
|
|
||||||
# Merge any extra kwargs last
|
|
||||||
backend_search_kwargs.update(kwargs)
|
|
||||||
|
|
||||||
results = self.backend_impl.search(
|
results = self.backend_impl.search(
|
||||||
query_embedding,
|
query_embedding,
|
||||||
top_k,
|
top_k,
|
||||||
**backend_search_kwargs,
|
complexity=complexity,
|
||||||
|
beam_width=beam_width,
|
||||||
|
prune_ratio=prune_ratio,
|
||||||
|
recompute_embeddings=recompute_embeddings,
|
||||||
|
pruning_strategy=pruning_strategy,
|
||||||
|
zmq_port=zmq_port,
|
||||||
|
**kwargs,
|
||||||
)
|
)
|
||||||
search_time = time.time() - start_time
|
# logger.info(f" Search time: {search_time} seconds")
|
||||||
logger.info(f" Search time in search() LEANN searcher: {search_time} seconds")
|
|
||||||
logger.info(f" Backend returned: labels={len(results.get('labels', [[]])[0])} results")
|
logger.info(f" Backend returned: labels={len(results.get('labels', [[]])[0])} results")
|
||||||
|
|
||||||
enriched_results = []
|
enriched_results = []
|
||||||
@@ -803,115 +668,20 @@ class LeannSearcher:
|
|||||||
f" {RED}✗{RESET} [{i + 1:2d}] ID: '{string_id}' -> {RED}ERROR: Passage not found!{RESET}"
|
f" {RED}✗{RESET} [{i + 1:2d}] ID: '{string_id}' -> {RED}ERROR: Passage not found!{RESET}"
|
||||||
)
|
)
|
||||||
|
|
||||||
# Apply metadata filters if specified
|
|
||||||
if metadata_filters:
|
|
||||||
logger.info(f" 🔍 Applying metadata filters: {metadata_filters}")
|
|
||||||
enriched_results = self.passage_manager.filter_search_results(
|
|
||||||
enriched_results, metadata_filters
|
|
||||||
)
|
|
||||||
|
|
||||||
# Define color codes outside the loop for final message
|
# Define color codes outside the loop for final message
|
||||||
GREEN = "\033[92m"
|
GREEN = "\033[92m"
|
||||||
RESET = "\033[0m"
|
RESET = "\033[0m"
|
||||||
logger.info(f" {GREEN}✓ Final enriched results: {len(enriched_results)} passages{RESET}")
|
logger.info(f" {GREEN}✓ Final enriched results: {len(enriched_results)} passages{RESET}")
|
||||||
return enriched_results
|
return enriched_results
|
||||||
|
|
||||||
def _find_jsonl_file(self) -> Optional[str]:
|
|
||||||
"""Find the .jsonl file containing raw passages for grep search"""
|
|
||||||
index_path = Path(self.meta_path_str).parent
|
|
||||||
potential_files = [
|
|
||||||
index_path / "documents.leann.passages.jsonl",
|
|
||||||
index_path.parent / "documents.leann.passages.jsonl",
|
|
||||||
]
|
|
||||||
|
|
||||||
for file_path in potential_files:
|
|
||||||
if file_path.exists():
|
|
||||||
return str(file_path)
|
|
||||||
return None
|
|
||||||
|
|
||||||
def _grep_search(self, query: str, top_k: int = 5) -> list[SearchResult]:
|
|
||||||
"""Perform grep-based search on raw passages"""
|
|
||||||
jsonl_file = self._find_jsonl_file()
|
|
||||||
if not jsonl_file:
|
|
||||||
raise FileNotFoundError("No .jsonl passages file found for grep search")
|
|
||||||
|
|
||||||
try:
|
|
||||||
cmd = ["grep", "-i", "-n", query, jsonl_file]
|
|
||||||
result = subprocess.run(cmd, capture_output=True, text=True, check=False)
|
|
||||||
|
|
||||||
if result.returncode == 1:
|
|
||||||
return []
|
|
||||||
elif result.returncode != 0:
|
|
||||||
raise RuntimeError(f"Grep failed: {result.stderr}")
|
|
||||||
|
|
||||||
matches = []
|
|
||||||
for line in result.stdout.strip().split("\n"):
|
|
||||||
if not line:
|
|
||||||
continue
|
|
||||||
parts = line.split(":", 1)
|
|
||||||
if len(parts) != 2:
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
|
||||||
data = json.loads(parts[1])
|
|
||||||
text = data.get("text", "")
|
|
||||||
score = text.lower().count(query.lower())
|
|
||||||
|
|
||||||
matches.append(
|
|
||||||
SearchResult(
|
|
||||||
id=data.get("id", parts[0]),
|
|
||||||
text=text,
|
|
||||||
metadata=data.get("metadata", {}),
|
|
||||||
score=float(score),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
except json.JSONDecodeError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
matches.sort(key=lambda x: x.score, reverse=True)
|
|
||||||
return matches[:top_k]
|
|
||||||
|
|
||||||
except FileNotFoundError:
|
|
||||||
raise RuntimeError(
|
|
||||||
"grep command not found. Please install grep or use semantic search."
|
|
||||||
)
|
|
||||||
|
|
||||||
def _python_regex_search(self, query: str, top_k: int = 5) -> list[SearchResult]:
|
|
||||||
"""Fallback regex search"""
|
|
||||||
jsonl_file = self._find_jsonl_file()
|
|
||||||
if not jsonl_file:
|
|
||||||
raise FileNotFoundError("No .jsonl file found")
|
|
||||||
|
|
||||||
pattern = re.compile(re.escape(query), re.IGNORECASE)
|
|
||||||
matches = []
|
|
||||||
|
|
||||||
with open(jsonl_file, encoding="utf-8") as f:
|
|
||||||
for line_num, line in enumerate(f, 1):
|
|
||||||
if pattern.search(line):
|
|
||||||
try:
|
|
||||||
data = json.loads(line.strip())
|
|
||||||
matches.append(
|
|
||||||
SearchResult(
|
|
||||||
id=data.get("id", str(line_num)),
|
|
||||||
text=data.get("text", ""),
|
|
||||||
metadata=data.get("metadata", {}),
|
|
||||||
score=float(len(pattern.findall(data.get("text", "")))),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
except json.JSONDecodeError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
matches.sort(key=lambda x: x.score, reverse=True)
|
|
||||||
return matches[:top_k]
|
|
||||||
|
|
||||||
def cleanup(self):
|
def cleanup(self):
|
||||||
"""Explicitly cleanup embedding server resources.
|
"""Explicitly cleanup embedding server resources.
|
||||||
|
|
||||||
This method should be called after you're done using the searcher,
|
This method should be called after you're done using the searcher,
|
||||||
especially in test environments or batch processing scenarios.
|
especially in test environments or batch processing scenarios.
|
||||||
"""
|
"""
|
||||||
backend = getattr(self.backend_impl, "embedding_server_manager", None)
|
if hasattr(self.backend_impl, "embedding_server_manager"):
|
||||||
if backend is not None:
|
self.backend_impl.embedding_server_manager.stop_server()
|
||||||
backend.stop_server()
|
|
||||||
|
|
||||||
# Enable automatic cleanup patterns
|
# Enable automatic cleanup patterns
|
||||||
def __enter__(self):
|
def __enter__(self):
|
||||||
@@ -937,15 +707,9 @@ class LeannChat:
|
|||||||
index_path: str,
|
index_path: str,
|
||||||
llm_config: Optional[dict[str, Any]] = None,
|
llm_config: Optional[dict[str, Any]] = None,
|
||||||
enable_warmup: bool = False,
|
enable_warmup: bool = False,
|
||||||
searcher: Optional[LeannSearcher] = None,
|
|
||||||
**kwargs,
|
**kwargs,
|
||||||
):
|
):
|
||||||
if searcher is None:
|
self.searcher = LeannSearcher(index_path, enable_warmup=enable_warmup, **kwargs)
|
||||||
self.searcher = LeannSearcher(index_path, enable_warmup=enable_warmup, **kwargs)
|
|
||||||
self._owns_searcher = True
|
|
||||||
else:
|
|
||||||
self.searcher = searcher
|
|
||||||
self._owns_searcher = False
|
|
||||||
self.llm = get_llm(llm_config)
|
self.llm = get_llm(llm_config)
|
||||||
|
|
||||||
def ask(
|
def ask(
|
||||||
@@ -959,9 +723,6 @@ class LeannChat:
|
|||||||
pruning_strategy: Literal["global", "local", "proportional"] = "global",
|
pruning_strategy: Literal["global", "local", "proportional"] = "global",
|
||||||
llm_kwargs: Optional[dict[str, Any]] = None,
|
llm_kwargs: Optional[dict[str, Any]] = None,
|
||||||
expected_zmq_port: int = 5557,
|
expected_zmq_port: int = 5557,
|
||||||
metadata_filters: Optional[dict[str, dict[str, Union[str, int, float, bool, list]]]] = None,
|
|
||||||
batch_size: int = 0,
|
|
||||||
use_grep: bool = False,
|
|
||||||
**search_kwargs,
|
**search_kwargs,
|
||||||
):
|
):
|
||||||
if llm_kwargs is None:
|
if llm_kwargs is None:
|
||||||
@@ -976,12 +737,10 @@ class LeannChat:
|
|||||||
recompute_embeddings=recompute_embeddings,
|
recompute_embeddings=recompute_embeddings,
|
||||||
pruning_strategy=pruning_strategy,
|
pruning_strategy=pruning_strategy,
|
||||||
expected_zmq_port=expected_zmq_port,
|
expected_zmq_port=expected_zmq_port,
|
||||||
metadata_filters=metadata_filters,
|
|
||||||
batch_size=batch_size,
|
|
||||||
**search_kwargs,
|
**search_kwargs,
|
||||||
)
|
)
|
||||||
search_time = time.time() - search_time
|
search_time = time.time() - search_time
|
||||||
logger.info(f" Search time: {search_time} seconds")
|
# logger.info(f" Search time: {search_time} seconds")
|
||||||
context = "\n\n".join([r.text for r in results])
|
context = "\n\n".join([r.text for r in results])
|
||||||
prompt = (
|
prompt = (
|
||||||
"Here is some retrieved context that might help answer your question:\n\n"
|
"Here is some retrieved context that might help answer your question:\n\n"
|
||||||
@@ -1017,9 +776,7 @@ class LeannChat:
|
|||||||
This method should be called after you're done using the chat interface,
|
This method should be called after you're done using the chat interface,
|
||||||
especially in test environments or batch processing scenarios.
|
especially in test environments or batch processing scenarios.
|
||||||
"""
|
"""
|
||||||
# Only stop the embedding server if this LeannChat instance created the searcher.
|
if hasattr(self.searcher, "cleanup"):
|
||||||
# When a shared searcher is passed in, avoid shutting down the server to enable reuse.
|
|
||||||
if getattr(self, "_owns_searcher", False) and hasattr(self.searcher, "cleanup"):
|
|
||||||
self.searcher.cleanup()
|
self.searcher.cleanup()
|
||||||
|
|
||||||
# Enable automatic cleanup patterns
|
# Enable automatic cleanup patterns
|
||||||
@@ -1032,405 +789,8 @@ class LeannChat:
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
# ------------------------------
|
|
||||||
# Incremental Add Utilities (HNSW no-recompute only)
|
|
||||||
# ------------------------------
|
|
||||||
|
|
||||||
|
|
||||||
def _resolve_index_paths(index_path: str) -> tuple[Path, Path, Path]:
|
|
||||||
"""Given base index path (without extension), return (passages.jsonl, passages.idx, vector.index).
|
|
||||||
|
|
||||||
For HNSW, vector index file is typically <stem>.index (e.g., documents.index) even when base is
|
|
||||||
'documents.leann'. We prefer an existing <stem>.index, otherwise fall back to <name>.index.
|
|
||||||
"""
|
|
||||||
base = Path(index_path)
|
|
||||||
passages_file = base.parent / f"{base.name}.passages.jsonl"
|
|
||||||
offsets_file = base.parent / f"{base.name}.passages.idx"
|
|
||||||
candidate_name_index = base.parent / f"{base.name}.index"
|
|
||||||
candidate_stem_index = base.parent / f"{base.stem}.index"
|
|
||||||
vector_index_file = (
|
|
||||||
candidate_stem_index if candidate_stem_index.exists() else candidate_name_index
|
|
||||||
)
|
|
||||||
return passages_file, offsets_file, vector_index_file
|
|
||||||
|
|
||||||
|
|
||||||
def _read_meta_file(index_path: str) -> dict[str, Any]:
|
|
||||||
meta_path = Path(f"{index_path}.meta.json")
|
|
||||||
if not meta_path.exists():
|
|
||||||
raise FileNotFoundError(f"Leann metadata file not found: {meta_path}")
|
|
||||||
with open(meta_path, encoding="utf-8") as f:
|
|
||||||
return json.load(f)
|
|
||||||
|
|
||||||
|
|
||||||
def _load_offset_map_pickle(offsets_file: Path) -> dict[str, int]:
|
|
||||||
if not offsets_file.exists():
|
|
||||||
return {}
|
|
||||||
with open(offsets_file, "rb") as f:
|
|
||||||
return pickle.load(f)
|
|
||||||
|
|
||||||
|
|
||||||
def _append_passages_and_update_offsets(
|
|
||||||
passages_file: Path, offsets_file: Path, new_texts: list[str]
|
|
||||||
) -> list[str]:
|
|
||||||
"""Append new texts to passages file, update offset map, and return assigned string IDs.
|
|
||||||
|
|
||||||
IDs are assigned as incrementing integers based on existing keys in the offset map.
|
|
||||||
"""
|
|
||||||
offset_map = _load_offset_map_pickle(offsets_file)
|
|
||||||
# Compute next numeric id
|
|
||||||
numeric_ids = [int(x) for x in offset_map.keys() if str(x).isdigit()]
|
|
||||||
next_id_num = (max(numeric_ids) + 1) if numeric_ids else 0
|
|
||||||
assigned_ids: list[str] = []
|
|
||||||
|
|
||||||
with open(passages_file, "a", encoding="utf-8") as f:
|
|
||||||
for text in new_texts:
|
|
||||||
offset = f.tell()
|
|
||||||
str_id = str(next_id_num)
|
|
||||||
json.dump({"id": str_id, "text": text, "metadata": {}}, f, ensure_ascii=False)
|
|
||||||
f.write("\n")
|
|
||||||
offset_map[str_id] = offset
|
|
||||||
assigned_ids.append(str_id)
|
|
||||||
next_id_num += 1
|
|
||||||
|
|
||||||
with open(offsets_file, "wb") as f:
|
|
||||||
pickle.dump(offset_map, f)
|
|
||||||
|
|
||||||
return assigned_ids
|
|
||||||
|
|
||||||
|
|
||||||
def incremental_add_texts(
|
|
||||||
index_path: str,
|
|
||||||
texts: list[str],
|
|
||||||
*,
|
|
||||||
embedding_model: Optional[str] = None,
|
|
||||||
embedding_mode: Optional[str] = None,
|
|
||||||
ef_construction: Optional[int] = None,
|
|
||||||
recompute: bool = False,
|
|
||||||
) -> int:
|
|
||||||
"""Incrementally add text chunks to an existing HNSW index built with no-recompute.
|
|
||||||
|
|
||||||
- Validates backend is HNSW and index is non-compact (no-recompute path)
|
|
||||||
- Appends passages and offsets
|
|
||||||
- Computes embeddings and appends to the HNSW vector index
|
|
||||||
|
|
||||||
Returns number of added chunks.
|
|
||||||
"""
|
|
||||||
if not texts:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
meta = _read_meta_file(index_path)
|
|
||||||
if meta.get("backend_name") != "hnsw":
|
|
||||||
raise RuntimeError("Incremental add is currently supported only for HNSW backend")
|
|
||||||
if meta.get("is_compact", True):
|
|
||||||
raise RuntimeError(
|
|
||||||
"Index is compact/pruned. Rebuild base with is_recompute=False and is_compact=False for incremental add."
|
|
||||||
)
|
|
||||||
|
|
||||||
passages_file, offsets_file, vector_index_file = _resolve_index_paths(index_path)
|
|
||||||
if not vector_index_file.exists():
|
|
||||||
raise FileNotFoundError(
|
|
||||||
f"Vector index file missing: {vector_index_file}. Build base first with LeannBuilder."
|
|
||||||
)
|
|
||||||
|
|
||||||
# Resolve embedding config from meta if not provided
|
|
||||||
model_name = embedding_model or meta.get("embedding_model", "facebook/contriever")
|
|
||||||
mode_name = embedding_mode or meta.get("embedding_mode", "sentence-transformers")
|
|
||||||
|
|
||||||
# Append passages and update offsets
|
|
||||||
assigned_ids = _append_passages_and_update_offsets(passages_file, offsets_file, texts)
|
|
||||||
|
|
||||||
# Compute embeddings
|
|
||||||
# Embedding computation path
|
|
||||||
esm = None
|
|
||||||
port = None
|
|
||||||
if recompute:
|
|
||||||
# Determine distance metric early for server config
|
|
||||||
distance_metric = meta.get("backend_kwargs", {}).get("distance_metric", "mips").lower()
|
|
||||||
# Start embedding server and compute via ZMQ for consistency with recompute semantics
|
|
||||||
passages_source_file = f"{index_path}.meta.json"
|
|
||||||
esm = EmbeddingServerManager(
|
|
||||||
backend_module_name="leann_backend_hnsw.hnsw_embedding_server",
|
|
||||||
)
|
|
||||||
started, port = esm.start_server(
|
|
||||||
port=5557,
|
|
||||||
model_name=model_name,
|
|
||||||
embedding_mode=mode_name,
|
|
||||||
passages_file=passages_source_file,
|
|
||||||
distance_metric=distance_metric,
|
|
||||||
enable_warmup=False,
|
|
||||||
)
|
|
||||||
if not started:
|
|
||||||
raise RuntimeError("Failed to start embedding server for recompute add")
|
|
||||||
embeddings = compute_embeddings_via_server(texts, model_name, port)
|
|
||||||
else:
|
|
||||||
embeddings = compute_embeddings(
|
|
||||||
texts,
|
|
||||||
model_name=model_name,
|
|
||||||
mode=mode_name,
|
|
||||||
use_server=False,
|
|
||||||
is_build=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Normalize for cosine if needed
|
|
||||||
if "distance_metric" not in locals():
|
|
||||||
distance_metric = meta.get("backend_kwargs", {}).get("distance_metric", "mips").lower()
|
|
||||||
if distance_metric == "cosine":
|
|
||||||
norms = np.linalg.norm(embeddings, axis=1, keepdims=True)
|
|
||||||
norms[norms == 0] = 1
|
|
||||||
embeddings = embeddings / norms
|
|
||||||
|
|
||||||
# Append via backend helper (supports ef_construction/recompute plumbing)
|
|
||||||
try:
|
|
||||||
from leann_backend_hnsw.hnsw_backend import add_vectors as hnsw_add_vectors # type: ignore
|
|
||||||
except Exception as e:
|
|
||||||
raise RuntimeError(
|
|
||||||
"Failed to import HNSW backend add helper. Ensure HNSW backend is installed."
|
|
||||||
) from e
|
|
||||||
|
|
||||||
# Propagate ZMQ port to FAISS add path when recompute is True
|
|
||||||
if recompute and port is not None:
|
|
||||||
os.environ["LEANN_ZMQ_PORT"] = str(port)
|
|
||||||
|
|
||||||
hnsw_add_vectors(
|
|
||||||
str(vector_index_file),
|
|
||||||
embeddings,
|
|
||||||
ef_construction=ef_construction,
|
|
||||||
recompute=recompute,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Stop server after add when recompute path used
|
|
||||||
if esm is not None:
|
|
||||||
try:
|
try:
|
||||||
esm.stop_server()
|
self.cleanup()
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
# Sanity: ids length should match embeddings rows
|
|
||||||
if len(assigned_ids) != embeddings.shape[0]:
|
|
||||||
warnings.warn(
|
|
||||||
f"Assigned {len(assigned_ids)} IDs but computed {embeddings.shape[0]} embeddings.",
|
|
||||||
UserWarning,
|
|
||||||
stacklevel=2,
|
|
||||||
)
|
|
||||||
|
|
||||||
return len(assigned_ids)
|
|
||||||
|
|
||||||
|
|
||||||
def create_incremental_add_context(
|
|
||||||
index_path: str,
|
|
||||||
*,
|
|
||||||
# Optional embedding choices; if None will use meta
|
|
||||||
embedding_model: Optional[str] = None,
|
|
||||||
embedding_mode: Optional[str] = None,
|
|
||||||
# Optional data-to-text preparation in context
|
|
||||||
data_dir: Optional[str] = None,
|
|
||||||
required_exts: Optional[list[str]] = None,
|
|
||||||
chunk_size: int = 256,
|
|
||||||
chunk_overlap: int = 128,
|
|
||||||
max_items: int = -1,
|
|
||||||
) -> IncrementalAddContext:
|
|
||||||
"""Validate index and prepare context for repeated incremental adds.
|
|
||||||
|
|
||||||
Additionally, if data_dir is provided, this function will load documents,
|
|
||||||
chunk them to texts with the specified parameters, and store them in ctx.prepared_texts.
|
|
||||||
"""
|
|
||||||
meta = _read_meta_file(index_path)
|
|
||||||
if meta.get("backend_name") != "hnsw":
|
|
||||||
raise RuntimeError("Incremental add is currently supported only for HNSW backend")
|
|
||||||
if meta.get("is_compact", True):
|
|
||||||
raise RuntimeError(
|
|
||||||
"Index is compact/pruned. Rebuild base with is_recompute=False and is_compact=False for incremental add."
|
|
||||||
)
|
|
||||||
|
|
||||||
passages_file, offsets_file, vector_index_file = _resolve_index_paths(index_path)
|
|
||||||
if not vector_index_file.exists():
|
|
||||||
raise FileNotFoundError(
|
|
||||||
f"Vector index file missing: {vector_index_file}. Build base first with LeannBuilder."
|
|
||||||
)
|
|
||||||
|
|
||||||
model_name = embedding_model or meta.get("embedding_model", "facebook/contriever")
|
|
||||||
mode_name = embedding_mode or meta.get("embedding_mode", "sentence-transformers")
|
|
||||||
distance_metric = meta.get("backend_kwargs", {}).get("distance_metric", "mips").lower()
|
|
||||||
|
|
||||||
prepared_texts: Optional[list[str]] = None
|
|
||||||
if data_dir is not None:
|
|
||||||
try:
|
|
||||||
from llama_index.core import SimpleDirectoryReader # type: ignore
|
|
||||||
from llama_index.core.node_parser import SentenceSplitter # type: ignore
|
|
||||||
except Exception as e:
|
|
||||||
raise RuntimeError(
|
|
||||||
"llama-index-core is required when using data_dir in create_incremental_add_context"
|
|
||||||
) from e
|
|
||||||
|
|
||||||
reader_kwargs: dict[str, Any] = {"recursive": True, "encoding": "utf-8"}
|
|
||||||
if required_exts:
|
|
||||||
reader_kwargs["required_exts"] = required_exts
|
|
||||||
documents = SimpleDirectoryReader(data_dir, **reader_kwargs).load_data(show_progress=True)
|
|
||||||
if documents:
|
|
||||||
splitter = SentenceSplitter(
|
|
||||||
chunk_size=chunk_size,
|
|
||||||
chunk_overlap=chunk_overlap,
|
|
||||||
separator=" ",
|
|
||||||
paragraph_separator="\n\n",
|
|
||||||
)
|
|
||||||
prepared_texts = []
|
|
||||||
for doc in documents:
|
|
||||||
try:
|
|
||||||
nodes = splitter.get_nodes_from_documents([doc])
|
|
||||||
if nodes:
|
|
||||||
prepared_texts.extend([node.get_content() for node in nodes])
|
|
||||||
except Exception:
|
|
||||||
content = doc.get_content()
|
|
||||||
if content and content.strip():
|
|
||||||
prepared_texts.append(content.strip())
|
|
||||||
if max_items > 0 and len(prepared_texts) > max_items:
|
|
||||||
prepared_texts = prepared_texts[:max_items]
|
|
||||||
|
|
||||||
return IncrementalAddContext(
|
|
||||||
index_path=index_path,
|
|
||||||
passages_file=passages_file,
|
|
||||||
offsets_file=offsets_file,
|
|
||||||
vector_index_file=vector_index_file,
|
|
||||||
embedding_model=model_name,
|
|
||||||
embedding_mode=mode_name,
|
|
||||||
distance_metric=distance_metric,
|
|
||||||
prepared_texts=prepared_texts,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def incremental_add_texts_with_context(
|
|
||||||
ctx: IncrementalAddContext,
|
|
||||||
texts: list[str],
|
|
||||||
*,
|
|
||||||
ef_construction: Optional[int] = None,
|
|
||||||
recompute: bool = False,
|
|
||||||
) -> int:
|
|
||||||
"""Incrementally add texts using a prepared context (no repeated validation).
|
|
||||||
|
|
||||||
For non-compact HNSW, ef_construction (efConstruction) can be overridden during insertion.
|
|
||||||
"""
|
|
||||||
if not texts:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
# Append passages & offsets
|
|
||||||
_append_passages_and_update_offsets(ctx.passages_file, ctx.offsets_file, texts)
|
|
||||||
|
|
||||||
# Compute embeddings
|
|
||||||
# Embedding computation path
|
|
||||||
esm = None
|
|
||||||
port = None
|
|
||||||
if recompute:
|
|
||||||
passages_source_file = f"{ctx.index_path}.meta.json"
|
|
||||||
esm = EmbeddingServerManager(
|
|
||||||
backend_module_name="leann_backend_hnsw.hnsw_embedding_server",
|
|
||||||
)
|
|
||||||
started, port = esm.start_server(
|
|
||||||
port=5557,
|
|
||||||
model_name=ctx.embedding_model,
|
|
||||||
embedding_mode=ctx.embedding_mode,
|
|
||||||
passages_file=passages_source_file,
|
|
||||||
distance_metric=ctx.distance_metric,
|
|
||||||
enable_warmup=False,
|
|
||||||
)
|
|
||||||
if not started:
|
|
||||||
raise RuntimeError("Failed to start embedding server for recompute add")
|
|
||||||
embeddings = compute_embeddings_via_server(texts, ctx.embedding_model, port)
|
|
||||||
else:
|
|
||||||
embeddings = compute_embeddings(
|
|
||||||
texts,
|
|
||||||
model_name=ctx.embedding_model,
|
|
||||||
mode=ctx.embedding_mode,
|
|
||||||
use_server=False,
|
|
||||||
is_build=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Normalize for cosine if needed
|
|
||||||
if ctx.distance_metric == "cosine":
|
|
||||||
norms = np.linalg.norm(embeddings, axis=1, keepdims=True)
|
|
||||||
norms[norms == 0] = 1
|
|
||||||
embeddings = embeddings / norms
|
|
||||||
|
|
||||||
# Append via backend helper (supports ef_construction/recompute plumbing)
|
|
||||||
try:
|
|
||||||
from leann_backend_hnsw.hnsw_backend import add_vectors as hnsw_add_vectors # type: ignore
|
|
||||||
except Exception as e:
|
|
||||||
raise RuntimeError(
|
|
||||||
"Failed to import HNSW backend add helper. Ensure HNSW backend is installed."
|
|
||||||
) from e
|
|
||||||
|
|
||||||
if recompute and port is not None:
|
|
||||||
os.environ["LEANN_ZMQ_PORT"] = str(port)
|
|
||||||
|
|
||||||
hnsw_add_vectors(
|
|
||||||
str(ctx.vector_index_file),
|
|
||||||
embeddings,
|
|
||||||
ef_construction=ef_construction,
|
|
||||||
recompute=recompute,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Stop server after add when recompute path used
|
|
||||||
if esm is not None:
|
|
||||||
try:
|
|
||||||
esm.stop_server()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
return embeddings.shape[0]
|
|
||||||
|
|
||||||
|
|
||||||
def incremental_add_directory(
|
|
||||||
index_path: str,
|
|
||||||
data_dir: str,
|
|
||||||
*,
|
|
||||||
chunk_size: int = 256,
|
|
||||||
chunk_overlap: int = 128,
|
|
||||||
required_exts: Optional[list[str]] = None,
|
|
||||||
max_items: int = -1,
|
|
||||||
embedding_model: Optional[str] = None,
|
|
||||||
embedding_mode: Optional[str] = None,
|
|
||||||
) -> int:
|
|
||||||
"""Load documents from a directory, chunk them, and incrementally add to an index.
|
|
||||||
|
|
||||||
Chunking uses LlamaIndex SentenceSplitter for simplicity and avoids external app dependencies.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
from llama_index.core import SimpleDirectoryReader # type: ignore
|
|
||||||
from llama_index.core.node_parser import SentenceSplitter # type: ignore
|
|
||||||
except Exception as e:
|
|
||||||
raise RuntimeError("llama-index-core is required for incremental_add_directory") from e
|
|
||||||
|
|
||||||
reader_kwargs: dict[str, Any] = {"recursive": True, "encoding": "utf-8"}
|
|
||||||
if required_exts:
|
|
||||||
reader_kwargs["required_exts"] = required_exts
|
|
||||||
documents = SimpleDirectoryReader(data_dir, **reader_kwargs).load_data(show_progress=True)
|
|
||||||
if not documents:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
# Traditional text chunking
|
|
||||||
splitter = SentenceSplitter(
|
|
||||||
chunk_size=chunk_size,
|
|
||||||
chunk_overlap=chunk_overlap,
|
|
||||||
separator=" ",
|
|
||||||
paragraph_separator="\n\n",
|
|
||||||
)
|
|
||||||
all_texts: list[str] = []
|
|
||||||
for doc in documents:
|
|
||||||
try:
|
|
||||||
nodes = splitter.get_nodes_from_documents([doc])
|
|
||||||
if nodes:
|
|
||||||
all_texts.extend([node.get_content() for node in nodes])
|
|
||||||
except Exception:
|
|
||||||
content = doc.get_content()
|
|
||||||
if content and content.strip():
|
|
||||||
all_texts.append(content.strip())
|
|
||||||
|
|
||||||
if max_items > 0 and len(all_texts) > max_items:
|
|
||||||
all_texts = all_texts[:max_items]
|
|
||||||
|
|
||||||
return incremental_add_texts(
|
|
||||||
index_path,
|
|
||||||
all_texts,
|
|
||||||
embedding_model=embedding_model,
|
|
||||||
embedding_mode=embedding_mode,
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -707,28 +707,20 @@ class GeminiChat(LLMInterface):
|
|||||||
logger.info(f"Sending request to Gemini with model {self.model}")
|
logger.info(f"Sending request to Gemini with model {self.model}")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from google.genai.types import GenerateContentConfig
|
# Set generation configuration
|
||||||
|
generation_config = {
|
||||||
generation_config = GenerateContentConfig(
|
"temperature": kwargs.get("temperature", 0.7),
|
||||||
temperature=kwargs.get("temperature", 0.7),
|
"max_output_tokens": kwargs.get("max_tokens", 1000),
|
||||||
max_output_tokens=kwargs.get("max_tokens", 1000),
|
}
|
||||||
)
|
|
||||||
|
|
||||||
# Handle top_p parameter
|
# Handle top_p parameter
|
||||||
if "top_p" in kwargs:
|
if "top_p" in kwargs:
|
||||||
generation_config.top_p = kwargs["top_p"]
|
generation_config["top_p"] = kwargs["top_p"]
|
||||||
|
|
||||||
response = self.client.models.generate_content(
|
response = self.client.models.generate_content(
|
||||||
model=self.model,
|
model=self.model, contents=prompt, config=generation_config
|
||||||
contents=prompt,
|
|
||||||
config=generation_config,
|
|
||||||
)
|
)
|
||||||
# Handle potential None response text
|
return response.text.strip()
|
||||||
response_text = response.text
|
|
||||||
if response_text is None:
|
|
||||||
logger.warning("Gemini returned None response text")
|
|
||||||
return ""
|
|
||||||
return response_text.strip()
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(f"Error communicating with Gemini: {e}")
|
logger.error(f"Error communicating with Gemini: {e}")
|
||||||
return f"Error: Could not get a response from Gemini. Details: {e}"
|
return f"Error: Could not get a response from Gemini. Details: {e}"
|
||||||
|
|||||||
@@ -1,220 +0,0 @@
|
|||||||
"""
|
|
||||||
Enhanced chunking utilities with AST-aware code chunking support.
|
|
||||||
Packaged within leann-core so installed wheels can import it reliably.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from llama_index.core.node_parser import SentenceSplitter
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
# Code file extensions supported by astchunk
|
|
||||||
CODE_EXTENSIONS = {
|
|
||||||
".py": "python",
|
|
||||||
".java": "java",
|
|
||||||
".cs": "csharp",
|
|
||||||
".ts": "typescript",
|
|
||||||
".tsx": "typescript",
|
|
||||||
".js": "typescript",
|
|
||||||
".jsx": "typescript",
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def detect_code_files(documents, code_extensions=None) -> tuple[list, list]:
|
|
||||||
"""Separate documents into code files and regular text files."""
|
|
||||||
if code_extensions is None:
|
|
||||||
code_extensions = CODE_EXTENSIONS
|
|
||||||
|
|
||||||
code_docs = []
|
|
||||||
text_docs = []
|
|
||||||
|
|
||||||
for doc in documents:
|
|
||||||
file_path = doc.metadata.get("file_path", "") or doc.metadata.get("file_name", "")
|
|
||||||
if file_path:
|
|
||||||
file_ext = Path(file_path).suffix.lower()
|
|
||||||
if file_ext in code_extensions:
|
|
||||||
doc.metadata["language"] = code_extensions[file_ext]
|
|
||||||
doc.metadata["is_code"] = True
|
|
||||||
code_docs.append(doc)
|
|
||||||
else:
|
|
||||||
doc.metadata["is_code"] = False
|
|
||||||
text_docs.append(doc)
|
|
||||||
else:
|
|
||||||
doc.metadata["is_code"] = False
|
|
||||||
text_docs.append(doc)
|
|
||||||
|
|
||||||
logger.info(f"Detected {len(code_docs)} code files and {len(text_docs)} text files")
|
|
||||||
return code_docs, text_docs
|
|
||||||
|
|
||||||
|
|
||||||
def get_language_from_extension(file_path: str) -> Optional[str]:
|
|
||||||
"""Return language string from a filename/extension using CODE_EXTENSIONS."""
|
|
||||||
ext = Path(file_path).suffix.lower()
|
|
||||||
return CODE_EXTENSIONS.get(ext)
|
|
||||||
|
|
||||||
|
|
||||||
def create_ast_chunks(
|
|
||||||
documents,
|
|
||||||
max_chunk_size: int = 512,
|
|
||||||
chunk_overlap: int = 64,
|
|
||||||
metadata_template: str = "default",
|
|
||||||
) -> list[str]:
|
|
||||||
"""Create AST-aware chunks from code documents using astchunk.
|
|
||||||
|
|
||||||
Falls back to traditional chunking if astchunk is unavailable.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
from astchunk import ASTChunkBuilder # optional dependency
|
|
||||||
except ImportError as e:
|
|
||||||
logger.error(f"astchunk not available: {e}")
|
|
||||||
logger.info("Falling back to traditional chunking for code files")
|
|
||||||
return create_traditional_chunks(documents, max_chunk_size, chunk_overlap)
|
|
||||||
|
|
||||||
all_chunks = []
|
|
||||||
for doc in documents:
|
|
||||||
language = doc.metadata.get("language")
|
|
||||||
if not language:
|
|
||||||
logger.warning("No language detected; falling back to traditional chunking")
|
|
||||||
all_chunks.extend(create_traditional_chunks([doc], max_chunk_size, chunk_overlap))
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
|
||||||
configs = {
|
|
||||||
"max_chunk_size": max_chunk_size,
|
|
||||||
"language": language,
|
|
||||||
"metadata_template": metadata_template,
|
|
||||||
"chunk_overlap": chunk_overlap if chunk_overlap > 0 else 0,
|
|
||||||
}
|
|
||||||
|
|
||||||
repo_metadata = {
|
|
||||||
"file_path": doc.metadata.get("file_path", ""),
|
|
||||||
"file_name": doc.metadata.get("file_name", ""),
|
|
||||||
"creation_date": doc.metadata.get("creation_date", ""),
|
|
||||||
"last_modified_date": doc.metadata.get("last_modified_date", ""),
|
|
||||||
}
|
|
||||||
configs["repo_level_metadata"] = repo_metadata
|
|
||||||
|
|
||||||
chunk_builder = ASTChunkBuilder(**configs)
|
|
||||||
code_content = doc.get_content()
|
|
||||||
if not code_content or not code_content.strip():
|
|
||||||
logger.warning("Empty code content, skipping")
|
|
||||||
continue
|
|
||||||
|
|
||||||
chunks = chunk_builder.chunkify(code_content)
|
|
||||||
for chunk in chunks:
|
|
||||||
if hasattr(chunk, "text"):
|
|
||||||
chunk_text = chunk.text
|
|
||||||
elif isinstance(chunk, dict) and "text" in chunk:
|
|
||||||
chunk_text = chunk["text"]
|
|
||||||
elif isinstance(chunk, str):
|
|
||||||
chunk_text = chunk
|
|
||||||
else:
|
|
||||||
chunk_text = str(chunk)
|
|
||||||
|
|
||||||
if chunk_text and chunk_text.strip():
|
|
||||||
all_chunks.append(chunk_text.strip())
|
|
||||||
|
|
||||||
logger.info(
|
|
||||||
f"Created {len(chunks)} AST chunks from {language} file: {doc.metadata.get('file_name', 'unknown')}"
|
|
||||||
)
|
|
||||||
except Exception as e:
|
|
||||||
logger.warning(f"AST chunking failed for {language} file: {e}")
|
|
||||||
logger.info("Falling back to traditional chunking")
|
|
||||||
all_chunks.extend(create_traditional_chunks([doc], max_chunk_size, chunk_overlap))
|
|
||||||
|
|
||||||
return all_chunks
|
|
||||||
|
|
||||||
|
|
||||||
def create_traditional_chunks(
|
|
||||||
documents, chunk_size: int = 256, chunk_overlap: int = 128
|
|
||||||
) -> list[str]:
|
|
||||||
"""Create traditional text chunks using LlamaIndex SentenceSplitter."""
|
|
||||||
if chunk_size <= 0:
|
|
||||||
logger.warning(f"Invalid chunk_size={chunk_size}, using default value of 256")
|
|
||||||
chunk_size = 256
|
|
||||||
if chunk_overlap < 0:
|
|
||||||
chunk_overlap = 0
|
|
||||||
if chunk_overlap >= chunk_size:
|
|
||||||
chunk_overlap = chunk_size // 2
|
|
||||||
|
|
||||||
node_parser = SentenceSplitter(
|
|
||||||
chunk_size=chunk_size,
|
|
||||||
chunk_overlap=chunk_overlap,
|
|
||||||
separator=" ",
|
|
||||||
paragraph_separator="\n\n",
|
|
||||||
)
|
|
||||||
|
|
||||||
all_texts = []
|
|
||||||
for doc in documents:
|
|
||||||
try:
|
|
||||||
nodes = node_parser.get_nodes_from_documents([doc])
|
|
||||||
if nodes:
|
|
||||||
all_texts.extend(node.get_content() for node in nodes)
|
|
||||||
except Exception as e:
|
|
||||||
logger.error(f"Traditional chunking failed for document: {e}")
|
|
||||||
content = doc.get_content()
|
|
||||||
if content and content.strip():
|
|
||||||
all_texts.append(content.strip())
|
|
||||||
|
|
||||||
return all_texts
|
|
||||||
|
|
||||||
|
|
||||||
def create_text_chunks(
|
|
||||||
documents,
|
|
||||||
chunk_size: int = 256,
|
|
||||||
chunk_overlap: int = 128,
|
|
||||||
use_ast_chunking: bool = False,
|
|
||||||
ast_chunk_size: int = 512,
|
|
||||||
ast_chunk_overlap: int = 64,
|
|
||||||
code_file_extensions: Optional[list[str]] = None,
|
|
||||||
ast_fallback_traditional: bool = True,
|
|
||||||
) -> list[str]:
|
|
||||||
"""Create text chunks from documents with optional AST support for code files."""
|
|
||||||
if not documents:
|
|
||||||
logger.warning("No documents provided for chunking")
|
|
||||||
return []
|
|
||||||
|
|
||||||
local_code_extensions = CODE_EXTENSIONS.copy()
|
|
||||||
if code_file_extensions:
|
|
||||||
ext_mapping = {
|
|
||||||
".py": "python",
|
|
||||||
".java": "java",
|
|
||||||
".cs": "c_sharp",
|
|
||||||
".ts": "typescript",
|
|
||||||
".tsx": "typescript",
|
|
||||||
}
|
|
||||||
for ext in code_file_extensions:
|
|
||||||
if ext.lower() not in local_code_extensions:
|
|
||||||
if ext.lower() in ext_mapping:
|
|
||||||
local_code_extensions[ext.lower()] = ext_mapping[ext.lower()]
|
|
||||||
else:
|
|
||||||
logger.warning(f"Unsupported extension {ext}, will use traditional chunking")
|
|
||||||
|
|
||||||
all_chunks = []
|
|
||||||
if use_ast_chunking:
|
|
||||||
code_docs, text_docs = detect_code_files(documents, local_code_extensions)
|
|
||||||
if code_docs:
|
|
||||||
try:
|
|
||||||
all_chunks.extend(
|
|
||||||
create_ast_chunks(
|
|
||||||
code_docs, max_chunk_size=ast_chunk_size, chunk_overlap=ast_chunk_overlap
|
|
||||||
)
|
|
||||||
)
|
|
||||||
except Exception as e:
|
|
||||||
logger.error(f"AST chunking failed: {e}")
|
|
||||||
if ast_fallback_traditional:
|
|
||||||
all_chunks.extend(
|
|
||||||
create_traditional_chunks(code_docs, chunk_size, chunk_overlap)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
raise
|
|
||||||
if text_docs:
|
|
||||||
all_chunks.extend(create_traditional_chunks(text_docs, chunk_size, chunk_overlap))
|
|
||||||
else:
|
|
||||||
all_chunks = create_traditional_chunks(documents, chunk_size, chunk_overlap)
|
|
||||||
|
|
||||||
logger.info(f"Total chunks created: {len(all_chunks)}")
|
|
||||||
return all_chunks
|
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
import argparse
|
import argparse
|
||||||
import asyncio
|
import asyncio
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Any, Optional, Union
|
from typing import Union
|
||||||
|
|
||||||
from llama_index.core import SimpleDirectoryReader
|
from llama_index.core import SimpleDirectoryReader
|
||||||
from llama_index.core.node_parser import SentenceSplitter
|
from llama_index.core.node_parser import SentenceSplitter
|
||||||
from tqdm import tqdm
|
from tqdm import tqdm
|
||||||
|
|
||||||
from .api import LeannBuilder, LeannChat, LeannSearcher
|
from .api import LeannBuilder, LeannChat, LeannSearcher
|
||||||
from .registry import register_project_directory
|
|
||||||
|
|
||||||
|
|
||||||
def extract_pdf_text_with_pymupdf(file_path: str) -> str:
|
def extract_pdf_text_with_pymupdf(file_path: str) -> str:
|
||||||
@@ -180,29 +179,6 @@ Examples:
|
|||||||
default=50,
|
default=50,
|
||||||
help="Code chunk overlap (default: 50)",
|
help="Code chunk overlap (default: 50)",
|
||||||
)
|
)
|
||||||
build_parser.add_argument(
|
|
||||||
"--use-ast-chunking",
|
|
||||||
action="store_true",
|
|
||||||
help="Enable AST-aware chunking for code files (requires astchunk)",
|
|
||||||
)
|
|
||||||
build_parser.add_argument(
|
|
||||||
"--ast-chunk-size",
|
|
||||||
type=int,
|
|
||||||
default=768,
|
|
||||||
help="AST chunk size in characters (default: 768)",
|
|
||||||
)
|
|
||||||
build_parser.add_argument(
|
|
||||||
"--ast-chunk-overlap",
|
|
||||||
type=int,
|
|
||||||
default=96,
|
|
||||||
help="AST chunk overlap in characters (default: 96)",
|
|
||||||
)
|
|
||||||
build_parser.add_argument(
|
|
||||||
"--ast-fallback-traditional",
|
|
||||||
action="store_true",
|
|
||||||
default=True,
|
|
||||||
help="Fall back to traditional chunking if AST chunking fails (default: True)",
|
|
||||||
)
|
|
||||||
|
|
||||||
# Search command
|
# Search command
|
||||||
search_parser = subparsers.add_parser("search", help="Search documents")
|
search_parser = subparsers.add_parser("search", help="Search documents")
|
||||||
@@ -229,11 +205,6 @@ Examples:
|
|||||||
default="global",
|
default="global",
|
||||||
help="Pruning strategy (default: global)",
|
help="Pruning strategy (default: global)",
|
||||||
)
|
)
|
||||||
search_parser.add_argument(
|
|
||||||
"--non-interactive",
|
|
||||||
action="store_true",
|
|
||||||
help="Non-interactive mode: automatically select index without prompting",
|
|
||||||
)
|
|
||||||
|
|
||||||
# Ask command
|
# Ask command
|
||||||
ask_parser = subparsers.add_parser("ask", help="Ask questions")
|
ask_parser = subparsers.add_parser("ask", help="Ask questions")
|
||||||
@@ -292,7 +263,31 @@ Examples:
|
|||||||
|
|
||||||
def register_project_dir(self):
|
def register_project_dir(self):
|
||||||
"""Register current project directory in global registry"""
|
"""Register current project directory in global registry"""
|
||||||
register_project_directory()
|
global_registry = Path.home() / ".leann" / "projects.json"
|
||||||
|
global_registry.parent.mkdir(exist_ok=True)
|
||||||
|
|
||||||
|
current_dir = str(Path.cwd())
|
||||||
|
|
||||||
|
# Load existing registry
|
||||||
|
projects = []
|
||||||
|
if global_registry.exists():
|
||||||
|
try:
|
||||||
|
import json
|
||||||
|
|
||||||
|
with open(global_registry) as f:
|
||||||
|
projects = json.load(f)
|
||||||
|
except Exception:
|
||||||
|
projects = []
|
||||||
|
|
||||||
|
# Add current directory if not already present
|
||||||
|
if current_dir not in projects:
|
||||||
|
projects.append(current_dir)
|
||||||
|
|
||||||
|
# Save registry
|
||||||
|
import json
|
||||||
|
|
||||||
|
with open(global_registry, "w") as f:
|
||||||
|
json.dump(projects, f, indent=2)
|
||||||
|
|
||||||
def _build_gitignore_parser(self, docs_dir: str):
|
def _build_gitignore_parser(self, docs_dir: str):
|
||||||
"""Build gitignore parser using gitignore-parser library."""
|
"""Build gitignore parser using gitignore-parser library."""
|
||||||
@@ -321,17 +316,9 @@ Examples:
|
|||||||
|
|
||||||
return basic_matches
|
return basic_matches
|
||||||
|
|
||||||
def _should_exclude_file(self, file_path: Path, gitignore_matches) -> bool:
|
def _should_exclude_file(self, relative_path: Path, gitignore_matches) -> bool:
|
||||||
"""Check if a file should be excluded using gitignore parser.
|
"""Check if a file should be excluded using gitignore parser."""
|
||||||
|
return gitignore_matches(str(relative_path))
|
||||||
Always match against absolute, posix-style paths for consistency with
|
|
||||||
gitignore_parser expectations.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
absolute_path = file_path.resolve()
|
|
||||||
except Exception:
|
|
||||||
absolute_path = Path(str(file_path))
|
|
||||||
return gitignore_matches(absolute_path.as_posix())
|
|
||||||
|
|
||||||
def _is_git_submodule(self, path: Path) -> bool:
|
def _is_git_submodule(self, path: Path) -> bool:
|
||||||
"""Check if a path is a git submodule."""
|
"""Check if a path is a git submodule."""
|
||||||
@@ -386,10 +373,13 @@ Examples:
|
|||||||
valid_projects.append(current_path)
|
valid_projects.append(current_path)
|
||||||
|
|
||||||
# Separate current and other projects
|
# Separate current and other projects
|
||||||
|
current_project = None
|
||||||
other_projects = []
|
other_projects = []
|
||||||
|
|
||||||
for project_path in valid_projects:
|
for project_path in valid_projects:
|
||||||
if project_path != current_path:
|
if project_path == current_path:
|
||||||
|
current_project = project_path
|
||||||
|
else:
|
||||||
other_projects.append(project_path)
|
other_projects.append(project_path)
|
||||||
|
|
||||||
print("📚 LEANN Indexes")
|
print("📚 LEANN Indexes")
|
||||||
@@ -399,22 +389,35 @@ Examples:
|
|||||||
current_indexes_count = 0
|
current_indexes_count = 0
|
||||||
|
|
||||||
# Show current project first (most important)
|
# Show current project first (most important)
|
||||||
print("\n🏠 Current Project")
|
if current_project:
|
||||||
print(f" {current_path}")
|
current_indexes_dir = current_project / ".leann" / "indexes"
|
||||||
print(" " + "─" * 45)
|
if current_indexes_dir.exists():
|
||||||
|
current_index_dirs = [d for d in current_indexes_dir.iterdir() if d.is_dir()]
|
||||||
|
|
||||||
current_indexes = self._discover_indexes_in_project(
|
print("\n🏠 Current Project")
|
||||||
current_path, exclude_dirs=other_projects
|
print(f" {current_project}")
|
||||||
)
|
print(" " + "─" * 45)
|
||||||
if current_indexes:
|
|
||||||
for idx in current_indexes:
|
if current_index_dirs:
|
||||||
total_indexes += 1
|
for index_dir in current_index_dirs:
|
||||||
current_indexes_count += 1
|
total_indexes += 1
|
||||||
type_icon = "📁" if idx["type"] == "cli" else "📄"
|
current_indexes_count += 1
|
||||||
print(f" {current_indexes_count}. {type_icon} {idx['name']} {idx['status']}")
|
index_name = index_dir.name
|
||||||
if idx["size_mb"] > 0:
|
meta_file = index_dir / "documents.leann.meta.json"
|
||||||
print(f" 📦 Size: {idx['size_mb']:.1f} MB")
|
status = "✅" if meta_file.exists() else "❌"
|
||||||
|
|
||||||
|
print(f" {current_indexes_count}. {index_name} {status}")
|
||||||
|
if meta_file.exists():
|
||||||
|
size_mb = sum(
|
||||||
|
f.stat().st_size for f in index_dir.iterdir() if f.is_file()
|
||||||
|
) / (1024 * 1024)
|
||||||
|
print(f" 📦 Size: {size_mb:.1f} MB")
|
||||||
|
else:
|
||||||
|
print(" 📭 No indexes in current project")
|
||||||
else:
|
else:
|
||||||
|
print("\n🏠 Current Project")
|
||||||
|
print(f" {current_path}")
|
||||||
|
print(" " + "─" * 45)
|
||||||
print(" 📭 No indexes in current project")
|
print(" 📭 No indexes in current project")
|
||||||
|
|
||||||
# Show other projects (reference information)
|
# Show other projects (reference information)
|
||||||
@@ -423,19 +426,29 @@ Examples:
|
|||||||
print(" " + "─" * 45)
|
print(" " + "─" * 45)
|
||||||
|
|
||||||
for project_path in other_projects:
|
for project_path in other_projects:
|
||||||
project_indexes = self._discover_indexes_in_project(project_path)
|
indexes_dir = project_path / ".leann" / "indexes"
|
||||||
if not project_indexes:
|
if not indexes_dir.exists():
|
||||||
|
continue
|
||||||
|
|
||||||
|
index_dirs = [d for d in indexes_dir.iterdir() if d.is_dir()]
|
||||||
|
if not index_dirs:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
print(f"\n 📂 {project_path.name}")
|
print(f"\n 📂 {project_path.name}")
|
||||||
print(f" {project_path}")
|
print(f" {project_path}")
|
||||||
|
|
||||||
for idx in project_indexes:
|
for index_dir in index_dirs:
|
||||||
total_indexes += 1
|
total_indexes += 1
|
||||||
type_icon = "📁" if idx["type"] == "cli" else "📄"
|
index_name = index_dir.name
|
||||||
print(f" • {type_icon} {idx['name']} {idx['status']}")
|
meta_file = index_dir / "documents.leann.meta.json"
|
||||||
if idx["size_mb"] > 0:
|
status = "✅" if meta_file.exists() else "❌"
|
||||||
print(f" 📦 {idx['size_mb']:.1f} MB")
|
|
||||||
|
print(f" • {index_name} {status}")
|
||||||
|
if meta_file.exists():
|
||||||
|
size_mb = sum(
|
||||||
|
f.stat().st_size for f in index_dir.iterdir() if f.is_file()
|
||||||
|
) / (1024 * 1024)
|
||||||
|
print(f" 📦 {size_mb:.1f} MB")
|
||||||
|
|
||||||
# Summary and usage info
|
# Summary and usage info
|
||||||
print("\n" + "=" * 50)
|
print("\n" + "=" * 50)
|
||||||
@@ -443,15 +456,14 @@ Examples:
|
|||||||
print("💡 Get started:")
|
print("💡 Get started:")
|
||||||
print(" leann build my-docs --docs ./documents")
|
print(" leann build my-docs --docs ./documents")
|
||||||
else:
|
else:
|
||||||
# Count only projects that have at least one discoverable index
|
projects_count = len(
|
||||||
projects_count = 0
|
[
|
||||||
for p in valid_projects:
|
p
|
||||||
if p == current_path:
|
for p in valid_projects
|
||||||
discovered = self._discover_indexes_in_project(p, exclude_dirs=other_projects)
|
if (p / ".leann" / "indexes").exists()
|
||||||
else:
|
and list((p / ".leann" / "indexes").iterdir())
|
||||||
discovered = self._discover_indexes_in_project(p)
|
]
|
||||||
if len(discovered) > 0:
|
)
|
||||||
projects_count += 1
|
|
||||||
print(f"📊 Total: {total_indexes} indexes across {projects_count} projects")
|
print(f"📊 Total: {total_indexes} indexes across {projects_count} projects")
|
||||||
|
|
||||||
if current_indexes_count > 0:
|
if current_indexes_count > 0:
|
||||||
@@ -468,100 +480,6 @@ Examples:
|
|||||||
print("\n💡 Create your first index:")
|
print("\n💡 Create your first index:")
|
||||||
print(" leann build my-docs --docs ./documents")
|
print(" leann build my-docs --docs ./documents")
|
||||||
|
|
||||||
def _discover_indexes_in_project(
|
|
||||||
self, project_path: Path, exclude_dirs: Optional[list[Path]] = None
|
|
||||||
):
|
|
||||||
"""Discover all indexes in a project directory (both CLI and apps formats)
|
|
||||||
|
|
||||||
exclude_dirs: when provided, skip any APP-format index files that are
|
|
||||||
located under these directories. This prevents duplicates when the
|
|
||||||
current project is a parent directory of other registered projects.
|
|
||||||
"""
|
|
||||||
indexes = []
|
|
||||||
exclude_dirs = exclude_dirs or []
|
|
||||||
# normalize to resolved paths once for comparison
|
|
||||||
try:
|
|
||||||
exclude_dirs_resolved = [p.resolve() for p in exclude_dirs]
|
|
||||||
except Exception:
|
|
||||||
exclude_dirs_resolved = exclude_dirs
|
|
||||||
|
|
||||||
# 1. CLI format: .leann/indexes/index_name/
|
|
||||||
cli_indexes_dir = project_path / ".leann" / "indexes"
|
|
||||||
if cli_indexes_dir.exists():
|
|
||||||
for index_dir in cli_indexes_dir.iterdir():
|
|
||||||
if index_dir.is_dir():
|
|
||||||
meta_file = index_dir / "documents.leann.meta.json"
|
|
||||||
status = "✅" if meta_file.exists() else "❌"
|
|
||||||
|
|
||||||
size_mb = 0
|
|
||||||
if meta_file.exists():
|
|
||||||
try:
|
|
||||||
size_mb = sum(
|
|
||||||
f.stat().st_size for f in index_dir.iterdir() if f.is_file()
|
|
||||||
) / (1024 * 1024)
|
|
||||||
except (OSError, PermissionError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
indexes.append(
|
|
||||||
{
|
|
||||||
"name": index_dir.name,
|
|
||||||
"type": "cli",
|
|
||||||
"status": status,
|
|
||||||
"size_mb": size_mb,
|
|
||||||
"path": index_dir,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
# 2. Apps format: *.leann.meta.json files anywhere in the project
|
|
||||||
cli_indexes_dir = project_path / ".leann" / "indexes"
|
|
||||||
for meta_file in project_path.rglob("*.leann.meta.json"):
|
|
||||||
if meta_file.is_file():
|
|
||||||
# Skip CLI-built indexes (which store meta under .leann/indexes/<name>/)
|
|
||||||
try:
|
|
||||||
if cli_indexes_dir.exists() and cli_indexes_dir in meta_file.parents:
|
|
||||||
continue
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
# Skip meta files that live under excluded directories
|
|
||||||
try:
|
|
||||||
meta_parent_resolved = meta_file.parent.resolve()
|
|
||||||
if any(
|
|
||||||
meta_parent_resolved.is_relative_to(ex_dir)
|
|
||||||
for ex_dir in exclude_dirs_resolved
|
|
||||||
):
|
|
||||||
continue
|
|
||||||
except Exception:
|
|
||||||
# best effort; if resolve or comparison fails, do not exclude
|
|
||||||
pass
|
|
||||||
# Use the parent directory name as the app index display name
|
|
||||||
display_name = meta_file.parent.name
|
|
||||||
# Extract file base used to store files
|
|
||||||
file_base = meta_file.name.replace(".leann.meta.json", "")
|
|
||||||
|
|
||||||
# Apps indexes are considered complete if the .leann.meta.json file exists
|
|
||||||
status = "✅"
|
|
||||||
|
|
||||||
# Calculate total size of all related files (use file base)
|
|
||||||
size_mb = 0
|
|
||||||
try:
|
|
||||||
index_dir = meta_file.parent
|
|
||||||
for related_file in index_dir.glob(f"{file_base}.leann*"):
|
|
||||||
size_mb += related_file.stat().st_size / (1024 * 1024)
|
|
||||||
except (OSError, PermissionError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
indexes.append(
|
|
||||||
{
|
|
||||||
"name": display_name,
|
|
||||||
"type": "app",
|
|
||||||
"status": status,
|
|
||||||
"size_mb": size_mb,
|
|
||||||
"path": meta_file,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
return indexes
|
|
||||||
|
|
||||||
def remove_index(self, index_name: str, force: bool = False):
|
def remove_index(self, index_name: str, force: bool = False):
|
||||||
"""Safely remove an index - always show all matches for transparency"""
|
"""Safely remove an index - always show all matches for transparency"""
|
||||||
|
|
||||||
@@ -606,79 +524,13 @@ Examples:
|
|||||||
if not project_path.exists():
|
if not project_path.exists():
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# 1) CLI-format index under .leann/indexes/<name>
|
|
||||||
index_dir = project_path / ".leann" / "indexes" / index_name
|
index_dir = project_path / ".leann" / "indexes" / index_name
|
||||||
if index_dir.exists():
|
if index_dir.exists():
|
||||||
is_current = project_path == current_path
|
is_current = project_path == current_path
|
||||||
matches.append(
|
matches.append(
|
||||||
{
|
{"project_path": project_path, "index_dir": index_dir, "is_current": is_current}
|
||||||
"project_path": project_path,
|
|
||||||
"index_dir": index_dir,
|
|
||||||
"is_current": is_current,
|
|
||||||
"kind": "cli",
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# 2) App-format indexes
|
|
||||||
# We support two ways of addressing apps:
|
|
||||||
# a) by the file base (e.g., `pdf_documents`)
|
|
||||||
# b) by the parent directory name (e.g., `new_txt`)
|
|
||||||
seen_app_meta = set()
|
|
||||||
|
|
||||||
# 2a) by file base
|
|
||||||
for meta_file in project_path.rglob(f"{index_name}.leann.meta.json"):
|
|
||||||
if meta_file.is_file():
|
|
||||||
# Skip CLI-built indexes' meta under .leann/indexes
|
|
||||||
try:
|
|
||||||
cli_indexes_dir = project_path / ".leann" / "indexes"
|
|
||||||
if cli_indexes_dir.exists() and cli_indexes_dir in meta_file.parents:
|
|
||||||
continue
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
is_current = project_path == current_path
|
|
||||||
key = (str(project_path), str(meta_file))
|
|
||||||
if key in seen_app_meta:
|
|
||||||
continue
|
|
||||||
seen_app_meta.add(key)
|
|
||||||
matches.append(
|
|
||||||
{
|
|
||||||
"project_path": project_path,
|
|
||||||
"files_dir": meta_file.parent,
|
|
||||||
"meta_file": meta_file,
|
|
||||||
"is_current": is_current,
|
|
||||||
"kind": "app",
|
|
||||||
"display_name": meta_file.parent.name,
|
|
||||||
"file_base": meta_file.name.replace(".leann.meta.json", ""),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
# 2b) by parent directory name
|
|
||||||
for meta_file in project_path.rglob("*.leann.meta.json"):
|
|
||||||
if meta_file.is_file() and meta_file.parent.name == index_name:
|
|
||||||
# Skip CLI-built indexes' meta under .leann/indexes
|
|
||||||
try:
|
|
||||||
cli_indexes_dir = project_path / ".leann" / "indexes"
|
|
||||||
if cli_indexes_dir.exists() and cli_indexes_dir in meta_file.parents:
|
|
||||||
continue
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
is_current = project_path == current_path
|
|
||||||
key = (str(project_path), str(meta_file))
|
|
||||||
if key in seen_app_meta:
|
|
||||||
continue
|
|
||||||
seen_app_meta.add(key)
|
|
||||||
matches.append(
|
|
||||||
{
|
|
||||||
"project_path": project_path,
|
|
||||||
"files_dir": meta_file.parent,
|
|
||||||
"meta_file": meta_file,
|
|
||||||
"is_current": is_current,
|
|
||||||
"kind": "app",
|
|
||||||
"display_name": meta_file.parent.name,
|
|
||||||
"file_base": meta_file.name.replace(".leann.meta.json", ""),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
# Sort: current project first, then by project name
|
# Sort: current project first, then by project name
|
||||||
matches.sort(key=lambda x: (not x["is_current"], x["project_path"].name))
|
matches.sort(key=lambda x: (not x["is_current"], x["project_path"].name))
|
||||||
return matches
|
return matches
|
||||||
@@ -686,8 +538,8 @@ Examples:
|
|||||||
def _remove_single_match(self, match, index_name: str, force: bool):
|
def _remove_single_match(self, match, index_name: str, force: bool):
|
||||||
"""Handle removal when only one match is found"""
|
"""Handle removal when only one match is found"""
|
||||||
project_path = match["project_path"]
|
project_path = match["project_path"]
|
||||||
|
index_dir = match["index_dir"]
|
||||||
is_current = match["is_current"]
|
is_current = match["is_current"]
|
||||||
kind = match.get("kind", "cli")
|
|
||||||
|
|
||||||
if is_current:
|
if is_current:
|
||||||
location_info = "current project"
|
location_info = "current project"
|
||||||
@@ -698,10 +550,7 @@ Examples:
|
|||||||
|
|
||||||
print(f"✅ Found 1 index named '{index_name}':")
|
print(f"✅ Found 1 index named '{index_name}':")
|
||||||
print(f" {emoji} Location: {location_info}")
|
print(f" {emoji} Location: {location_info}")
|
||||||
if kind == "cli":
|
print(f" 📍 Path: {project_path}")
|
||||||
print(f" 📍 Path: {project_path / '.leann' / 'indexes' / index_name}")
|
|
||||||
else:
|
|
||||||
print(f" 📍 Meta: {match['meta_file']}")
|
|
||||||
|
|
||||||
if not force:
|
if not force:
|
||||||
if not is_current:
|
if not is_current:
|
||||||
@@ -713,22 +562,9 @@ Examples:
|
|||||||
print(" ❌ Removal cancelled.")
|
print(" ❌ Removal cancelled.")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if kind == "cli":
|
return self._delete_index_directory(
|
||||||
return self._delete_index_directory(
|
index_dir, index_name, project_path if not is_current else None
|
||||||
match["index_dir"],
|
)
|
||||||
index_name,
|
|
||||||
project_path if not is_current else None,
|
|
||||||
is_app=False,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return self._delete_index_directory(
|
|
||||||
match["files_dir"],
|
|
||||||
match.get("display_name", index_name),
|
|
||||||
project_path if not is_current else None,
|
|
||||||
is_app=True,
|
|
||||||
meta_file=match.get("meta_file"),
|
|
||||||
app_file_base=match.get("file_base"),
|
|
||||||
)
|
|
||||||
|
|
||||||
def _remove_from_multiple_matches(self, matches, index_name: str, force: bool):
|
def _remove_from_multiple_matches(self, matches, index_name: str, force: bool):
|
||||||
"""Handle removal when multiple matches are found"""
|
"""Handle removal when multiple matches are found"""
|
||||||
@@ -739,34 +575,19 @@ Examples:
|
|||||||
for i, match in enumerate(matches, 1):
|
for i, match in enumerate(matches, 1):
|
||||||
project_path = match["project_path"]
|
project_path = match["project_path"]
|
||||||
is_current = match["is_current"]
|
is_current = match["is_current"]
|
||||||
kind = match.get("kind", "cli")
|
|
||||||
|
|
||||||
if is_current:
|
if is_current:
|
||||||
print(f" {i}. 🏠 Current project ({'CLI' if kind == 'cli' else 'APP'})")
|
print(f" {i}. 🏠 Current project")
|
||||||
|
print(f" 📍 {project_path}")
|
||||||
else:
|
else:
|
||||||
print(f" {i}. 📂 {project_path.name} ({'CLI' if kind == 'cli' else 'APP'})")
|
print(f" {i}. 📂 {project_path.name}")
|
||||||
|
print(f" 📍 {project_path}")
|
||||||
# Show path details
|
|
||||||
if kind == "cli":
|
|
||||||
print(f" 📍 {project_path / '.leann' / 'indexes' / index_name}")
|
|
||||||
else:
|
|
||||||
print(f" 📍 {match['meta_file']}")
|
|
||||||
|
|
||||||
# Show size info
|
# Show size info
|
||||||
try:
|
try:
|
||||||
if kind == "cli":
|
size_mb = sum(
|
||||||
size_mb = sum(
|
f.stat().st_size for f in match["index_dir"].iterdir() if f.is_file()
|
||||||
f.stat().st_size for f in match["index_dir"].iterdir() if f.is_file()
|
) / (1024 * 1024)
|
||||||
) / (1024 * 1024)
|
|
||||||
else:
|
|
||||||
file_base = match.get("file_base")
|
|
||||||
size_mb = 0.0
|
|
||||||
if file_base:
|
|
||||||
size_mb = sum(
|
|
||||||
f.stat().st_size
|
|
||||||
for f in match["files_dir"].glob(f"{file_base}.leann*")
|
|
||||||
if f.is_file()
|
|
||||||
) / (1024 * 1024)
|
|
||||||
print(f" 📦 Size: {size_mb:.1f} MB")
|
print(f" 📦 Size: {size_mb:.1f} MB")
|
||||||
except (OSError, PermissionError):
|
except (OSError, PermissionError):
|
||||||
pass
|
pass
|
||||||
@@ -790,8 +611,8 @@ Examples:
|
|||||||
if 0 <= choice_idx < len(matches):
|
if 0 <= choice_idx < len(matches):
|
||||||
selected_match = matches[choice_idx]
|
selected_match = matches[choice_idx]
|
||||||
project_path = selected_match["project_path"]
|
project_path = selected_match["project_path"]
|
||||||
|
index_dir = selected_match["index_dir"]
|
||||||
is_current = selected_match["is_current"]
|
is_current = selected_match["is_current"]
|
||||||
kind = selected_match.get("kind", "cli")
|
|
||||||
|
|
||||||
location = "current project" if is_current else f"'{project_path.name}' project"
|
location = "current project" if is_current else f"'{project_path.name}' project"
|
||||||
print(f" 🎯 Selected: Remove from {location}")
|
print(f" 🎯 Selected: Remove from {location}")
|
||||||
@@ -804,22 +625,9 @@ Examples:
|
|||||||
print(" ❌ Confirmation failed. Removal cancelled.")
|
print(" ❌ Confirmation failed. Removal cancelled.")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if kind == "cli":
|
return self._delete_index_directory(
|
||||||
return self._delete_index_directory(
|
index_dir, index_name, project_path if not is_current else None
|
||||||
selected_match["index_dir"],
|
)
|
||||||
index_name,
|
|
||||||
project_path if not is_current else None,
|
|
||||||
is_app=False,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return self._delete_index_directory(
|
|
||||||
selected_match["files_dir"],
|
|
||||||
selected_match.get("display_name", index_name),
|
|
||||||
project_path if not is_current else None,
|
|
||||||
is_app=True,
|
|
||||||
meta_file=selected_match.get("meta_file"),
|
|
||||||
app_file_base=selected_match.get("file_base"),
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
print(" ❌ Invalid choice. Removal cancelled.")
|
print(" ❌ Invalid choice. Removal cancelled.")
|
||||||
return False
|
return False
|
||||||
@@ -829,65 +637,21 @@ Examples:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def _delete_index_directory(
|
def _delete_index_directory(
|
||||||
self,
|
self, index_dir: Path, index_name: str, project_path: Path | None = None
|
||||||
index_dir: Path,
|
|
||||||
index_display_name: str,
|
|
||||||
project_path: Optional[Path] = None,
|
|
||||||
is_app: bool = False,
|
|
||||||
meta_file: Optional[Path] = None,
|
|
||||||
app_file_base: Optional[str] = None,
|
|
||||||
):
|
):
|
||||||
"""Delete a CLI index directory or APP index files safely."""
|
"""Actually delete the index directory"""
|
||||||
try:
|
try:
|
||||||
if is_app:
|
import shutil
|
||||||
removed = 0
|
|
||||||
errors = 0
|
|
||||||
# Delete only files that belong to this app index (based on file base)
|
|
||||||
pattern_base = app_file_base or ""
|
|
||||||
for f in index_dir.glob(f"{pattern_base}.leann*"):
|
|
||||||
try:
|
|
||||||
f.unlink()
|
|
||||||
removed += 1
|
|
||||||
except Exception:
|
|
||||||
errors += 1
|
|
||||||
# Best-effort: also remove the meta file if specified and still exists
|
|
||||||
if meta_file and meta_file.exists():
|
|
||||||
try:
|
|
||||||
meta_file.unlink()
|
|
||||||
removed += 1
|
|
||||||
except Exception:
|
|
||||||
errors += 1
|
|
||||||
|
|
||||||
if removed > 0 and errors == 0:
|
shutil.rmtree(index_dir)
|
||||||
if project_path:
|
|
||||||
print(
|
if project_path:
|
||||||
f"✅ App index '{index_display_name}' removed from {project_path.name}"
|
print(f"✅ Index '{index_name}' removed from {project_path.name}")
|
||||||
)
|
|
||||||
else:
|
|
||||||
print(f"✅ App index '{index_display_name}' removed successfully")
|
|
||||||
return True
|
|
||||||
elif removed > 0 and errors > 0:
|
|
||||||
print(
|
|
||||||
f"⚠️ App index '{index_display_name}' partially removed (some files couldn't be deleted)"
|
|
||||||
)
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
print(
|
|
||||||
f"❌ No files found to remove for app index '{index_display_name}' in {index_dir}"
|
|
||||||
)
|
|
||||||
return False
|
|
||||||
else:
|
else:
|
||||||
import shutil
|
print(f"✅ Index '{index_name}' removed successfully")
|
||||||
|
return True
|
||||||
shutil.rmtree(index_dir)
|
|
||||||
|
|
||||||
if project_path:
|
|
||||||
print(f"✅ Index '{index_display_name}' removed from {project_path.name}")
|
|
||||||
else:
|
|
||||||
print(f"✅ Index '{index_display_name}' removed successfully")
|
|
||||||
return True
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(f"❌ Error removing index '{index_display_name}': {e}")
|
print(f"❌ Error removing index '{index_name}': {e}")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def load_documents(
|
def load_documents(
|
||||||
@@ -895,7 +659,6 @@ Examples:
|
|||||||
docs_paths: Union[str, list],
|
docs_paths: Union[str, list],
|
||||||
custom_file_types: Union[str, None] = None,
|
custom_file_types: Union[str, None] = None,
|
||||||
include_hidden: bool = False,
|
include_hidden: bool = False,
|
||||||
args: Optional[dict[str, Any]] = None,
|
|
||||||
):
|
):
|
||||||
# Handle both single path (string) and multiple paths (list) for backward compatibility
|
# Handle both single path (string) and multiple paths (list) for backward compatibility
|
||||||
if isinstance(docs_paths, str):
|
if isinstance(docs_paths, str):
|
||||||
@@ -1060,8 +823,7 @@ Examples:
|
|||||||
|
|
||||||
# Try to use better PDF parsers first, but only if PDFs are requested
|
# Try to use better PDF parsers first, but only if PDFs are requested
|
||||||
documents = []
|
documents = []
|
||||||
# Use resolved absolute paths to avoid mismatches (symlinks, relative vs absolute)
|
docs_path = Path(docs_dir)
|
||||||
docs_path = Path(docs_dir).resolve()
|
|
||||||
|
|
||||||
# Check if we should process PDFs
|
# Check if we should process PDFs
|
||||||
should_process_pdfs = custom_file_types is None or ".pdf" in custom_file_types
|
should_process_pdfs = custom_file_types is None or ".pdf" in custom_file_types
|
||||||
@@ -1070,15 +832,10 @@ Examples:
|
|||||||
for file_path in docs_path.rglob("*.pdf"):
|
for file_path in docs_path.rglob("*.pdf"):
|
||||||
# Check if file matches any exclude pattern
|
# Check if file matches any exclude pattern
|
||||||
try:
|
try:
|
||||||
# Ensure both paths are resolved before computing relativity
|
|
||||||
file_path_resolved = file_path.resolve()
|
|
||||||
# Determine directory scope using the non-resolved path to avoid
|
|
||||||
# misclassifying symlinked entries as outside the docs directory
|
|
||||||
relative_path = file_path.relative_to(docs_path)
|
relative_path = file_path.relative_to(docs_path)
|
||||||
if not include_hidden and _path_has_hidden_segment(relative_path):
|
if not include_hidden and _path_has_hidden_segment(relative_path):
|
||||||
continue
|
continue
|
||||||
# Use absolute path for gitignore matching
|
if self._should_exclude_file(relative_path, gitignore_matches):
|
||||||
if self._should_exclude_file(file_path_resolved, gitignore_matches):
|
|
||||||
continue
|
continue
|
||||||
except ValueError:
|
except ValueError:
|
||||||
# Skip files that can't be made relative to docs_path
|
# Skip files that can't be made relative to docs_path
|
||||||
@@ -1121,11 +878,10 @@ Examples:
|
|||||||
) -> bool:
|
) -> bool:
|
||||||
"""Return True if file should be included (not excluded)"""
|
"""Return True if file should be included (not excluded)"""
|
||||||
try:
|
try:
|
||||||
docs_path_obj = Path(docs_dir).resolve()
|
docs_path_obj = Path(docs_dir)
|
||||||
file_path_obj = Path(file_path).resolve()
|
file_path_obj = Path(file_path)
|
||||||
# Use absolute path for gitignore matching
|
relative_path = file_path_obj.relative_to(docs_path_obj)
|
||||||
_ = file_path_obj.relative_to(docs_path_obj) # validate scope
|
return not self._should_exclude_file(relative_path, gitignore_matches)
|
||||||
return not self._should_exclude_file(file_path_obj, gitignore_matches)
|
|
||||||
except (ValueError, OSError):
|
except (ValueError, OSError):
|
||||||
return True # Include files that can't be processed
|
return True # Include files that can't be processed
|
||||||
|
|
||||||
@@ -1208,47 +964,18 @@ Examples:
|
|||||||
}
|
}
|
||||||
|
|
||||||
print("start chunking documents")
|
print("start chunking documents")
|
||||||
|
# Add progress bar for document chunking
|
||||||
|
for doc in tqdm(documents, desc="Chunking documents", unit="doc"):
|
||||||
|
# Check if this is a code file based on source path
|
||||||
|
source_path = doc.metadata.get("source", "")
|
||||||
|
is_code_file = any(source_path.endswith(ext) for ext in code_file_exts)
|
||||||
|
|
||||||
# Check if AST chunking is requested
|
# Use appropriate parser based on file type
|
||||||
use_ast = getattr(args, "use_ast_chunking", False)
|
parser = self.code_parser if is_code_file else self.node_parser
|
||||||
|
nodes = parser.get_nodes_from_documents([doc])
|
||||||
|
|
||||||
if use_ast:
|
for node in nodes:
|
||||||
print("🧠 Using AST-aware chunking for code files")
|
all_texts.append(node.get_content())
|
||||||
try:
|
|
||||||
# Import enhanced chunking utilities from packaged module
|
|
||||||
from .chunking_utils import create_text_chunks
|
|
||||||
|
|
||||||
# Use enhanced chunking with AST support
|
|
||||||
all_texts = create_text_chunks(
|
|
||||||
documents,
|
|
||||||
chunk_size=self.node_parser.chunk_size,
|
|
||||||
chunk_overlap=self.node_parser.chunk_overlap,
|
|
||||||
use_ast_chunking=True,
|
|
||||||
ast_chunk_size=getattr(args, "ast_chunk_size", 768),
|
|
||||||
ast_chunk_overlap=getattr(args, "ast_chunk_overlap", 96),
|
|
||||||
code_file_extensions=None, # Use defaults
|
|
||||||
ast_fallback_traditional=getattr(args, "ast_fallback_traditional", True),
|
|
||||||
)
|
|
||||||
|
|
||||||
except ImportError as e:
|
|
||||||
print(
|
|
||||||
f"⚠️ AST chunking utilities not available in package ({e}), falling back to traditional chunking"
|
|
||||||
)
|
|
||||||
use_ast = False
|
|
||||||
|
|
||||||
if not use_ast:
|
|
||||||
# Use traditional chunking logic
|
|
||||||
for doc in tqdm(documents, desc="Chunking documents", unit="doc"):
|
|
||||||
# Check if this is a code file based on source path
|
|
||||||
source_path = doc.metadata.get("source", "")
|
|
||||||
is_code_file = any(source_path.endswith(ext) for ext in code_file_exts)
|
|
||||||
|
|
||||||
# Use appropriate parser based on file type
|
|
||||||
parser = self.code_parser if is_code_file else self.node_parser
|
|
||||||
nodes = parser.get_nodes_from_documents([doc])
|
|
||||||
|
|
||||||
for node in nodes:
|
|
||||||
all_texts.append(node.get_content())
|
|
||||||
|
|
||||||
print(f"Loaded {len(documents)} documents, {len(all_texts)} chunks")
|
print(f"Loaded {len(documents)} documents, {len(all_texts)} chunks")
|
||||||
return all_texts
|
return all_texts
|
||||||
@@ -1315,7 +1042,7 @@ Examples:
|
|||||||
)
|
)
|
||||||
|
|
||||||
all_texts = self.load_documents(
|
all_texts = self.load_documents(
|
||||||
docs_paths, args.file_types, include_hidden=args.include_hidden, args=args
|
docs_paths, args.file_types, include_hidden=args.include_hidden
|
||||||
)
|
)
|
||||||
if not all_texts:
|
if not all_texts:
|
||||||
print("No documents found")
|
print("No documents found")
|
||||||
@@ -1348,101 +1075,13 @@ Examples:
|
|||||||
async def search_documents(self, args):
|
async def search_documents(self, args):
|
||||||
index_name = args.index_name
|
index_name = args.index_name
|
||||||
query = args.query
|
query = args.query
|
||||||
|
|
||||||
# First try to find the index in current project
|
|
||||||
index_path = self.get_index_path(index_name)
|
index_path = self.get_index_path(index_name)
|
||||||
if self.index_exists(index_name):
|
|
||||||
# Found in current project, use it
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
# Search across all registered projects (like list_indexes does)
|
|
||||||
all_matches = self._find_all_matching_indexes(index_name)
|
|
||||||
if not all_matches:
|
|
||||||
print(
|
|
||||||
f"Index '{index_name}' not found. Use 'leann build {index_name} --docs <dir> [<dir2> ...]' to create it."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
elif len(all_matches) == 1:
|
|
||||||
# Found exactly one match, use it
|
|
||||||
match = all_matches[0]
|
|
||||||
if match["kind"] == "cli":
|
|
||||||
index_path = str(match["index_dir"] / "documents.leann")
|
|
||||||
else:
|
|
||||||
# App format: use the meta file to construct the path
|
|
||||||
meta_file = match["meta_file"]
|
|
||||||
file_base = match["file_base"]
|
|
||||||
index_path = str(meta_file.parent / f"{file_base}.leann")
|
|
||||||
|
|
||||||
project_info = (
|
if not self.index_exists(index_name):
|
||||||
"current project"
|
print(
|
||||||
if match["is_current"]
|
f"Index '{index_name}' not found. Use 'leann build {index_name} --docs <dir> [<dir2> ...]' to create it."
|
||||||
else f"project '{match['project_path'].name}'"
|
)
|
||||||
)
|
return
|
||||||
print(f"Using index '{index_name}' from {project_info}")
|
|
||||||
else:
|
|
||||||
# Multiple matches found
|
|
||||||
if args.non_interactive:
|
|
||||||
# Non-interactive mode: automatically select the best match
|
|
||||||
# Priority: current project first, then first available
|
|
||||||
current_matches = [m for m in all_matches if m["is_current"]]
|
|
||||||
if current_matches:
|
|
||||||
match = current_matches[0]
|
|
||||||
location_desc = "current project"
|
|
||||||
else:
|
|
||||||
match = all_matches[0]
|
|
||||||
location_desc = f"project '{match['project_path'].name}'"
|
|
||||||
|
|
||||||
if match["kind"] == "cli":
|
|
||||||
index_path = str(match["index_dir"] / "documents.leann")
|
|
||||||
else:
|
|
||||||
meta_file = match["meta_file"]
|
|
||||||
file_base = match["file_base"]
|
|
||||||
index_path = str(meta_file.parent / f"{file_base}.leann")
|
|
||||||
|
|
||||||
print(
|
|
||||||
f"Found {len(all_matches)} indexes named '{index_name}', using index from {location_desc}"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
# Interactive mode: ask user to choose
|
|
||||||
print(f"Found {len(all_matches)} indexes named '{index_name}':")
|
|
||||||
for i, match in enumerate(all_matches, 1):
|
|
||||||
project_path = match["project_path"]
|
|
||||||
is_current = match["is_current"]
|
|
||||||
kind = match.get("kind", "cli")
|
|
||||||
|
|
||||||
if is_current:
|
|
||||||
print(
|
|
||||||
f" {i}. 🏠 Current project ({'CLI' if kind == 'cli' else 'APP'})"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
print(
|
|
||||||
f" {i}. 📂 {project_path.name} ({'CLI' if kind == 'cli' else 'APP'})"
|
|
||||||
)
|
|
||||||
|
|
||||||
try:
|
|
||||||
choice = input(f"Which index to search? (1-{len(all_matches)}): ").strip()
|
|
||||||
choice_idx = int(choice) - 1
|
|
||||||
if 0 <= choice_idx < len(all_matches):
|
|
||||||
match = all_matches[choice_idx]
|
|
||||||
if match["kind"] == "cli":
|
|
||||||
index_path = str(match["index_dir"] / "documents.leann")
|
|
||||||
else:
|
|
||||||
meta_file = match["meta_file"]
|
|
||||||
file_base = match["file_base"]
|
|
||||||
index_path = str(meta_file.parent / f"{file_base}.leann")
|
|
||||||
|
|
||||||
project_info = (
|
|
||||||
"current project"
|
|
||||||
if match["is_current"]
|
|
||||||
else f"project '{match['project_path'].name}'"
|
|
||||||
)
|
|
||||||
print(f"Using index '{index_name}' from {project_info}")
|
|
||||||
else:
|
|
||||||
print("Invalid choice. Aborting search.")
|
|
||||||
return
|
|
||||||
except (ValueError, KeyboardInterrupt):
|
|
||||||
print("Invalid input. Aborting search.")
|
|
||||||
return
|
|
||||||
|
|
||||||
searcher = LeannSearcher(index_path=index_path)
|
searcher = LeannSearcher(index_path=index_path)
|
||||||
results = searcher.search(
|
results = searcher.search(
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ Preserves all optimization parameters to ensure performance
|
|||||||
|
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import time
|
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
import numpy as np
|
import numpy as np
|
||||||
@@ -29,8 +28,6 @@ def compute_embeddings(
|
|||||||
is_build: bool = False,
|
is_build: bool = False,
|
||||||
batch_size: int = 32,
|
batch_size: int = 32,
|
||||||
adaptive_optimization: bool = True,
|
adaptive_optimization: bool = True,
|
||||||
manual_tokenize: bool = False,
|
|
||||||
max_length: int = 512,
|
|
||||||
) -> np.ndarray:
|
) -> np.ndarray:
|
||||||
"""
|
"""
|
||||||
Unified embedding computation entry point
|
Unified embedding computation entry point
|
||||||
@@ -53,8 +50,6 @@ def compute_embeddings(
|
|||||||
is_build=is_build,
|
is_build=is_build,
|
||||||
batch_size=batch_size,
|
batch_size=batch_size,
|
||||||
adaptive_optimization=adaptive_optimization,
|
adaptive_optimization=adaptive_optimization,
|
||||||
manual_tokenize=manual_tokenize,
|
|
||||||
max_length=max_length,
|
|
||||||
)
|
)
|
||||||
elif mode == "openai":
|
elif mode == "openai":
|
||||||
return compute_embeddings_openai(texts, model_name)
|
return compute_embeddings_openai(texts, model_name)
|
||||||
@@ -76,8 +71,6 @@ def compute_embeddings_sentence_transformers(
|
|||||||
batch_size: int = 32,
|
batch_size: int = 32,
|
||||||
is_build: bool = False,
|
is_build: bool = False,
|
||||||
adaptive_optimization: bool = True,
|
adaptive_optimization: bool = True,
|
||||||
manual_tokenize: bool = False,
|
|
||||||
max_length: int = 512,
|
|
||||||
) -> np.ndarray:
|
) -> np.ndarray:
|
||||||
"""
|
"""
|
||||||
Compute embeddings using SentenceTransformer with model caching and adaptive optimization
|
Compute embeddings using SentenceTransformer with model caching and adaptive optimization
|
||||||
@@ -221,130 +214,20 @@ def compute_embeddings_sentence_transformers(
|
|||||||
logger.info(f"Model cached: {cache_key}")
|
logger.info(f"Model cached: {cache_key}")
|
||||||
|
|
||||||
# Compute embeddings with optimized inference mode
|
# Compute embeddings with optimized inference mode
|
||||||
logger.info(
|
logger.info(f"Starting embedding computation... (batch_size: {batch_size})")
|
||||||
f"Starting embedding computation... (batch_size: {batch_size}, manual_tokenize={manual_tokenize})"
|
|
||||||
)
|
|
||||||
|
|
||||||
start_time = time.time()
|
# Use torch.inference_mode for optimal performance
|
||||||
if not manual_tokenize:
|
with torch.inference_mode():
|
||||||
# Use SentenceTransformer's optimized encode path (default)
|
embeddings = model.encode(
|
||||||
with torch.inference_mode():
|
texts,
|
||||||
embeddings = model.encode(
|
batch_size=batch_size,
|
||||||
texts,
|
show_progress_bar=is_build, # Don't show progress bar in server environment
|
||||||
batch_size=batch_size,
|
convert_to_numpy=True,
|
||||||
show_progress_bar=is_build, # Don't show progress bar in server environment
|
normalize_embeddings=False,
|
||||||
convert_to_numpy=True,
|
device=device,
|
||||||
normalize_embeddings=False,
|
)
|
||||||
device=device,
|
|
||||||
)
|
|
||||||
# Synchronize if CUDA to measure accurate wall time
|
|
||||||
try:
|
|
||||||
if torch.cuda.is_available():
|
|
||||||
torch.cuda.synchronize()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
# Manual tokenization + forward pass using HF AutoTokenizer/AutoModel
|
|
||||||
try:
|
|
||||||
from transformers import AutoModel, AutoTokenizer # type: ignore
|
|
||||||
except Exception as e:
|
|
||||||
raise ImportError(f"transformers is required for manual_tokenize=True: {e}")
|
|
||||||
|
|
||||||
# Cache tokenizer and model
|
|
||||||
tok_cache_key = f"hf_tokenizer_{model_name}"
|
|
||||||
mdl_cache_key = f"hf_model_{model_name}_{device}_{use_fp16}"
|
|
||||||
if tok_cache_key in _model_cache and mdl_cache_key in _model_cache:
|
|
||||||
hf_tokenizer = _model_cache[tok_cache_key]
|
|
||||||
hf_model = _model_cache[mdl_cache_key]
|
|
||||||
logger.info("Using cached HF tokenizer/model for manual path")
|
|
||||||
else:
|
|
||||||
logger.info("Loading HF tokenizer/model for manual tokenization path")
|
|
||||||
hf_tokenizer = AutoTokenizer.from_pretrained(model_name, use_fast=True)
|
|
||||||
torch_dtype = torch.float16 if (use_fp16 and device == "cuda") else torch.float32
|
|
||||||
hf_model = AutoModel.from_pretrained(model_name, torch_dtype=torch_dtype)
|
|
||||||
hf_model.to(device)
|
|
||||||
hf_model.eval()
|
|
||||||
# Optional compile on supported devices
|
|
||||||
if device in ["cuda", "mps"]:
|
|
||||||
try:
|
|
||||||
hf_model = torch.compile(hf_model, mode="reduce-overhead", dynamic=True) # type: ignore
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
_model_cache[tok_cache_key] = hf_tokenizer
|
|
||||||
_model_cache[mdl_cache_key] = hf_model
|
|
||||||
|
|
||||||
all_embeddings: list[np.ndarray] = []
|
|
||||||
# Progress bar when building or for large inputs
|
|
||||||
show_progress = is_build or len(texts) > 32
|
|
||||||
try:
|
|
||||||
if show_progress:
|
|
||||||
from tqdm import tqdm # type: ignore
|
|
||||||
|
|
||||||
batch_iter = tqdm(
|
|
||||||
range(0, len(texts), batch_size),
|
|
||||||
desc="Embedding (manual)",
|
|
||||||
unit="batch",
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
batch_iter = range(0, len(texts), batch_size)
|
|
||||||
except Exception:
|
|
||||||
batch_iter = range(0, len(texts), batch_size)
|
|
||||||
|
|
||||||
start_time_manual = time.time()
|
|
||||||
with torch.inference_mode():
|
|
||||||
for start_index in batch_iter:
|
|
||||||
end_index = min(start_index + batch_size, len(texts))
|
|
||||||
batch_texts = texts[start_index:end_index]
|
|
||||||
tokenize_start_time = time.time()
|
|
||||||
inputs = hf_tokenizer(
|
|
||||||
batch_texts,
|
|
||||||
padding=True,
|
|
||||||
truncation=True,
|
|
||||||
max_length=max_length,
|
|
||||||
return_tensors="pt",
|
|
||||||
)
|
|
||||||
tokenize_end_time = time.time()
|
|
||||||
logger.info(
|
|
||||||
f"Tokenize time taken: {tokenize_end_time - tokenize_start_time} seconds"
|
|
||||||
)
|
|
||||||
# Print shapes of all input tensors for debugging
|
|
||||||
for k, v in inputs.items():
|
|
||||||
print(f"inputs[{k!r}] shape: {getattr(v, 'shape', type(v))}")
|
|
||||||
to_device_start_time = time.time()
|
|
||||||
inputs = {k: v.to(device) for k, v in inputs.items()}
|
|
||||||
to_device_end_time = time.time()
|
|
||||||
logger.info(
|
|
||||||
f"To device time taken: {to_device_end_time - to_device_start_time} seconds"
|
|
||||||
)
|
|
||||||
forward_start_time = time.time()
|
|
||||||
outputs = hf_model(**inputs)
|
|
||||||
forward_end_time = time.time()
|
|
||||||
logger.info(f"Forward time taken: {forward_end_time - forward_start_time} seconds")
|
|
||||||
last_hidden_state = outputs.last_hidden_state # (B, L, H)
|
|
||||||
attention_mask = inputs.get("attention_mask")
|
|
||||||
if attention_mask is None:
|
|
||||||
# Fallback: assume all tokens are valid
|
|
||||||
pooled = last_hidden_state.mean(dim=1)
|
|
||||||
else:
|
|
||||||
mask = attention_mask.unsqueeze(-1).to(last_hidden_state.dtype)
|
|
||||||
masked = last_hidden_state * mask
|
|
||||||
lengths = mask.sum(dim=1).clamp(min=1)
|
|
||||||
pooled = masked.sum(dim=1) / lengths
|
|
||||||
# Move to CPU float32
|
|
||||||
batch_embeddings = pooled.detach().to("cpu").float().numpy()
|
|
||||||
all_embeddings.append(batch_embeddings)
|
|
||||||
|
|
||||||
embeddings = np.vstack(all_embeddings).astype(np.float32, copy=False)
|
|
||||||
try:
|
|
||||||
if torch.cuda.is_available():
|
|
||||||
torch.cuda.synchronize()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
end_time = time.time()
|
|
||||||
logger.info(f"Manual tokenize time taken: {end_time - start_time_manual} seconds")
|
|
||||||
end_time = time.time()
|
|
||||||
logger.info(f"Generated {len(embeddings)} embeddings, dimension: {embeddings.shape[1]}")
|
logger.info(f"Generated {len(embeddings)} embeddings, dimension: {embeddings.shape[1]}")
|
||||||
logger.info(f"Time taken: {end_time - start_time} seconds")
|
|
||||||
|
|
||||||
# Validate results
|
# Validate results
|
||||||
if np.isnan(embeddings).any() or np.isinf(embeddings).any():
|
if np.isnan(embeddings).any() or np.isinf(embeddings).any():
|
||||||
|
|||||||
@@ -192,7 +192,6 @@ class EmbeddingServerManager:
|
|||||||
stderr_target = None # Direct to console for visible logs
|
stderr_target = None # Direct to console for visible logs
|
||||||
|
|
||||||
# Start embedding server subprocess
|
# Start embedding server subprocess
|
||||||
logger.info(f"Starting server process with command: {' '.join(command)}")
|
|
||||||
self.server_process = subprocess.Popen(
|
self.server_process = subprocess.Popen(
|
||||||
command,
|
command,
|
||||||
cwd=project_root,
|
cwd=project_root,
|
||||||
|
|||||||
@@ -94,7 +94,7 @@ def handle_request(request):
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
# Build simplified command with non-interactive flag for MCP compatibility
|
# Build simplified command
|
||||||
cmd = [
|
cmd = [
|
||||||
"leann",
|
"leann",
|
||||||
"search",
|
"search",
|
||||||
@@ -102,7 +102,6 @@ def handle_request(request):
|
|||||||
args["query"],
|
args["query"],
|
||||||
f"--top-k={args.get('top_k', 5)}",
|
f"--top-k={args.get('top_k', 5)}",
|
||||||
f"--complexity={args.get('complexity', 32)}",
|
f"--complexity={args.get('complexity', 32)}",
|
||||||
"--non-interactive",
|
|
||||||
]
|
]
|
||||||
result = subprocess.run(cmd, capture_output=True, text=True)
|
result = subprocess.run(cmd, capture_output=True, text=True)
|
||||||
|
|
||||||
|
|||||||
@@ -1,240 +0,0 @@
|
|||||||
"""
|
|
||||||
Metadata filtering engine for LEANN search results.
|
|
||||||
|
|
||||||
This module provides generic metadata filtering capabilities that can be applied
|
|
||||||
to search results from any LEANN backend. The filtering supports various
|
|
||||||
operators for different data types including numbers, strings, booleans, and lists.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import logging
|
|
||||||
from typing import Any, Union
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
# Type alias for filter specifications
|
|
||||||
FilterValue = Union[str, int, float, bool, list]
|
|
||||||
FilterSpec = dict[str, FilterValue]
|
|
||||||
MetadataFilters = dict[str, FilterSpec]
|
|
||||||
|
|
||||||
|
|
||||||
class MetadataFilterEngine:
|
|
||||||
"""
|
|
||||||
Engine for evaluating metadata filters against search results.
|
|
||||||
|
|
||||||
Supports various operators for filtering based on metadata fields:
|
|
||||||
- Comparison: ==, !=, <, <=, >, >=
|
|
||||||
- Membership: in, not_in
|
|
||||||
- String operations: contains, starts_with, ends_with
|
|
||||||
- Boolean operations: is_true, is_false
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
"""Initialize the filter engine with supported operators."""
|
|
||||||
self.operators = {
|
|
||||||
"==": self._equals,
|
|
||||||
"!=": self._not_equals,
|
|
||||||
"<": self._less_than,
|
|
||||||
"<=": self._less_than_or_equal,
|
|
||||||
">": self._greater_than,
|
|
||||||
">=": self._greater_than_or_equal,
|
|
||||||
"in": self._in,
|
|
||||||
"not_in": self._not_in,
|
|
||||||
"contains": self._contains,
|
|
||||||
"starts_with": self._starts_with,
|
|
||||||
"ends_with": self._ends_with,
|
|
||||||
"is_true": self._is_true,
|
|
||||||
"is_false": self._is_false,
|
|
||||||
}
|
|
||||||
|
|
||||||
def apply_filters(
|
|
||||||
self, search_results: list[dict[str, Any]], metadata_filters: MetadataFilters
|
|
||||||
) -> list[dict[str, Any]]:
|
|
||||||
"""
|
|
||||||
Apply metadata filters to a list of search results.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
search_results: List of result dictionaries, each containing 'metadata' field
|
|
||||||
metadata_filters: Dictionary of filter specifications
|
|
||||||
Format: {"field_name": {"operator": value}}
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Filtered list of search results
|
|
||||||
"""
|
|
||||||
if not metadata_filters:
|
|
||||||
return search_results
|
|
||||||
|
|
||||||
logger.debug(f"Applying filters: {metadata_filters}")
|
|
||||||
logger.debug(f"Input results count: {len(search_results)}")
|
|
||||||
|
|
||||||
filtered_results = []
|
|
||||||
for result in search_results:
|
|
||||||
if self._evaluate_filters(result, metadata_filters):
|
|
||||||
filtered_results.append(result)
|
|
||||||
|
|
||||||
logger.debug(f"Filtered results count: {len(filtered_results)}")
|
|
||||||
return filtered_results
|
|
||||||
|
|
||||||
def _evaluate_filters(self, result: dict[str, Any], filters: MetadataFilters) -> bool:
|
|
||||||
"""
|
|
||||||
Evaluate all filters against a single search result.
|
|
||||||
|
|
||||||
All filters must pass (AND logic) for the result to be included.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
result: Full search result dictionary (including metadata, text, etc.)
|
|
||||||
filters: Filter specifications to evaluate
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
True if all filters pass, False otherwise
|
|
||||||
"""
|
|
||||||
for field_name, filter_spec in filters.items():
|
|
||||||
if not self._evaluate_field_filter(result, field_name, filter_spec):
|
|
||||||
return False
|
|
||||||
return True
|
|
||||||
|
|
||||||
def _evaluate_field_filter(
|
|
||||||
self, result: dict[str, Any], field_name: str, filter_spec: FilterSpec
|
|
||||||
) -> bool:
|
|
||||||
"""
|
|
||||||
Evaluate a single field filter against a search result.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
result: Full search result dictionary
|
|
||||||
field_name: Name of the field to filter on
|
|
||||||
filter_spec: Filter specification for this field
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
True if the filter passes, False otherwise
|
|
||||||
"""
|
|
||||||
# First check top-level fields, then check metadata
|
|
||||||
field_value = result.get(field_name)
|
|
||||||
if field_value is None:
|
|
||||||
# Try to get from metadata if not found at top level
|
|
||||||
metadata = result.get("metadata", {})
|
|
||||||
field_value = metadata.get(field_name)
|
|
||||||
|
|
||||||
# Handle missing fields - they fail all filters except existence checks
|
|
||||||
if field_value is None:
|
|
||||||
logger.debug(f"Field '{field_name}' not found in result or metadata")
|
|
||||||
return False
|
|
||||||
|
|
||||||
# Evaluate each operator in the filter spec
|
|
||||||
for operator, expected_value in filter_spec.items():
|
|
||||||
if operator not in self.operators:
|
|
||||||
logger.warning(f"Unsupported operator: {operator}")
|
|
||||||
return False
|
|
||||||
|
|
||||||
try:
|
|
||||||
if not self.operators[operator](field_value, expected_value):
|
|
||||||
logger.debug(
|
|
||||||
f"Filter failed: {field_name} {operator} {expected_value} "
|
|
||||||
f"(actual: {field_value})"
|
|
||||||
)
|
|
||||||
return False
|
|
||||||
except Exception as e:
|
|
||||||
logger.warning(
|
|
||||||
f"Error evaluating filter {field_name} {operator} {expected_value}: {e}"
|
|
||||||
)
|
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
# Comparison operators
|
|
||||||
def _equals(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value equals expected value."""
|
|
||||||
return field_value == expected_value
|
|
||||||
|
|
||||||
def _not_equals(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value does not equal expected value."""
|
|
||||||
return field_value != expected_value
|
|
||||||
|
|
||||||
def _less_than(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is less than expected value."""
|
|
||||||
return self._numeric_compare(field_value, expected_value, lambda a, b: a < b)
|
|
||||||
|
|
||||||
def _less_than_or_equal(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is less than or equal to expected value."""
|
|
||||||
return self._numeric_compare(field_value, expected_value, lambda a, b: a <= b)
|
|
||||||
|
|
||||||
def _greater_than(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is greater than expected value."""
|
|
||||||
return self._numeric_compare(field_value, expected_value, lambda a, b: a > b)
|
|
||||||
|
|
||||||
def _greater_than_or_equal(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is greater than or equal to expected value."""
|
|
||||||
return self._numeric_compare(field_value, expected_value, lambda a, b: a >= b)
|
|
||||||
|
|
||||||
# Membership operators
|
|
||||||
def _in(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is in the expected list/collection."""
|
|
||||||
if not isinstance(expected_value, (list, tuple, set)):
|
|
||||||
raise ValueError("'in' operator requires a list, tuple, or set")
|
|
||||||
return field_value in expected_value
|
|
||||||
|
|
||||||
def _not_in(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is not in the expected list/collection."""
|
|
||||||
if not isinstance(expected_value, (list, tuple, set)):
|
|
||||||
raise ValueError("'not_in' operator requires a list, tuple, or set")
|
|
||||||
return field_value not in expected_value
|
|
||||||
|
|
||||||
# String operators
|
|
||||||
def _contains(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value contains the expected substring."""
|
|
||||||
field_str = str(field_value)
|
|
||||||
expected_str = str(expected_value)
|
|
||||||
return expected_str in field_str
|
|
||||||
|
|
||||||
def _starts_with(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value starts with the expected prefix."""
|
|
||||||
field_str = str(field_value)
|
|
||||||
expected_str = str(expected_value)
|
|
||||||
return field_str.startswith(expected_str)
|
|
||||||
|
|
||||||
def _ends_with(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value ends with the expected suffix."""
|
|
||||||
field_str = str(field_value)
|
|
||||||
expected_str = str(expected_value)
|
|
||||||
return field_str.endswith(expected_str)
|
|
||||||
|
|
||||||
# Boolean operators
|
|
||||||
def _is_true(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is truthy."""
|
|
||||||
return bool(field_value)
|
|
||||||
|
|
||||||
def _is_false(self, field_value: Any, expected_value: Any) -> bool:
|
|
||||||
"""Check if field value is falsy."""
|
|
||||||
return not bool(field_value)
|
|
||||||
|
|
||||||
# Helper methods
|
|
||||||
def _numeric_compare(self, field_value: Any, expected_value: Any, compare_func) -> bool:
|
|
||||||
"""
|
|
||||||
Helper for numeric comparisons with type coercion.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
field_value: Value from metadata
|
|
||||||
expected_value: Value to compare against
|
|
||||||
compare_func: Comparison function to apply
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Result of comparison
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
# Try to convert both values to numbers for comparison
|
|
||||||
if isinstance(field_value, str) and isinstance(expected_value, str):
|
|
||||||
# String comparison if both are strings
|
|
||||||
return compare_func(field_value, expected_value)
|
|
||||||
|
|
||||||
# Numeric comparison - attempt to convert to float
|
|
||||||
field_num = (
|
|
||||||
float(field_value) if not isinstance(field_value, (int, float)) else field_value
|
|
||||||
)
|
|
||||||
expected_num = (
|
|
||||||
float(expected_value)
|
|
||||||
if not isinstance(expected_value, (int, float))
|
|
||||||
else expected_value
|
|
||||||
)
|
|
||||||
|
|
||||||
return compare_func(field_num, expected_num)
|
|
||||||
except (ValueError, TypeError):
|
|
||||||
# Fall back to string comparison if numeric conversion fails
|
|
||||||
return compare_func(str(field_value), str(expected_value))
|
|
||||||
@@ -2,10 +2,8 @@
|
|||||||
|
|
||||||
import importlib
|
import importlib
|
||||||
import importlib.metadata
|
import importlib.metadata
|
||||||
import json
|
|
||||||
import logging
|
import logging
|
||||||
from pathlib import Path
|
from typing import TYPE_CHECKING
|
||||||
from typing import TYPE_CHECKING, Optional, Union
|
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from leann.interface import LeannBackendFactoryInterface
|
from leann.interface import LeannBackendFactoryInterface
|
||||||
@@ -45,54 +43,3 @@ def autodiscover_backends():
|
|||||||
# print(f"WARN: Could not import backend module '{backend_module_name}': {e}")
|
# print(f"WARN: Could not import backend module '{backend_module_name}': {e}")
|
||||||
pass
|
pass
|
||||||
# print("INFO: Backend auto-discovery finished.")
|
# print("INFO: Backend auto-discovery finished.")
|
||||||
|
|
||||||
|
|
||||||
def register_project_directory(project_dir: Optional[Union[str, Path]] = None):
|
|
||||||
"""
|
|
||||||
Register a project directory in the global LEANN registry.
|
|
||||||
|
|
||||||
This allows `leann list` to discover indexes created by apps or other tools.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
project_dir: Directory to register. If None, uses current working directory.
|
|
||||||
"""
|
|
||||||
if project_dir is None:
|
|
||||||
project_dir = Path.cwd()
|
|
||||||
else:
|
|
||||||
project_dir = Path(project_dir)
|
|
||||||
|
|
||||||
# Only register directories that have some kind of LEANN content
|
|
||||||
# Either .leann/indexes/ (CLI format) or *.leann.meta.json files (apps format)
|
|
||||||
has_cli_indexes = (project_dir / ".leann" / "indexes").exists()
|
|
||||||
has_app_indexes = any(project_dir.rglob("*.leann.meta.json"))
|
|
||||||
|
|
||||||
if not (has_cli_indexes or has_app_indexes):
|
|
||||||
# Don't register if there are no LEANN indexes
|
|
||||||
return
|
|
||||||
|
|
||||||
global_registry = Path.home() / ".leann" / "projects.json"
|
|
||||||
global_registry.parent.mkdir(exist_ok=True)
|
|
||||||
|
|
||||||
project_str = str(project_dir.resolve())
|
|
||||||
|
|
||||||
# Load existing registry
|
|
||||||
projects = []
|
|
||||||
if global_registry.exists():
|
|
||||||
try:
|
|
||||||
with open(global_registry) as f:
|
|
||||||
projects = json.load(f)
|
|
||||||
except Exception:
|
|
||||||
logger.debug("Could not load existing project registry")
|
|
||||||
projects = []
|
|
||||||
|
|
||||||
# Add project if not already present
|
|
||||||
if project_str not in projects:
|
|
||||||
projects.append(project_str)
|
|
||||||
|
|
||||||
# Save updated registry
|
|
||||||
try:
|
|
||||||
with open(global_registry, "w") as f:
|
|
||||||
json.dump(projects, f, indent=2)
|
|
||||||
logger.debug(f"Registered project directory: {project_str}")
|
|
||||||
except Exception as e:
|
|
||||||
logger.warning(f"Could not save project registry: {e}")
|
|
||||||
|
|||||||
@@ -2,8 +2,6 @@
|
|||||||
|
|
||||||
Transform your development workflow with intelligent code assistance using LEANN's semantic search directly in Claude Code.
|
Transform your development workflow with intelligent code assistance using LEANN's semantic search directly in Claude Code.
|
||||||
|
|
||||||
For agent-facing discovery details, see `llms.txt` in the repository root.
|
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
|
|
||||||
Install LEANN globally for MCP integration (with default backend):
|
Install LEANN globally for MCP integration (with default backend):
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "leann"
|
name = "leann"
|
||||||
version = "0.3.4"
|
version = "0.3.0"
|
||||||
description = "LEANN - The smallest vector index in the world. RAG Everything with LEANN!"
|
description = "LEANN - The smallest vector index in the world. RAG Everything with LEANN!"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
requires-python = ">=3.9"
|
requires-python = ">=3.9"
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ dependencies = [
|
|||||||
"numpy>=1.26.0",
|
"numpy>=1.26.0",
|
||||||
"torch",
|
"torch",
|
||||||
"tqdm",
|
"tqdm",
|
||||||
|
"flask",
|
||||||
|
"flask_compress",
|
||||||
"datasets>=2.15.0",
|
"datasets>=2.15.0",
|
||||||
"evaluate",
|
"evaluate",
|
||||||
"colorama",
|
"colorama",
|
||||||
@@ -46,13 +48,6 @@ dependencies = [
|
|||||||
"pathspec>=0.12.1",
|
"pathspec>=0.12.1",
|
||||||
"nbconvert>=7.16.6",
|
"nbconvert>=7.16.6",
|
||||||
"gitignore-parser>=0.1.12",
|
"gitignore-parser>=0.1.12",
|
||||||
# AST-aware code chunking dependencies
|
|
||||||
"astchunk>=0.1.0",
|
|
||||||
"tree-sitter>=0.20.0",
|
|
||||||
"tree-sitter-python>=0.20.0",
|
|
||||||
"tree-sitter-java>=0.20.0",
|
|
||||||
"tree-sitter-c-sharp>=0.20.0",
|
|
||||||
"tree-sitter-typescript>=0.20.0",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[project.optional-dependencies]
|
[project.optional-dependencies]
|
||||||
@@ -71,7 +66,9 @@ test = [
|
|||||||
"pytest>=7.0",
|
"pytest>=7.0",
|
||||||
"pytest-timeout>=2.0",
|
"pytest-timeout>=2.0",
|
||||||
"llama-index-core>=0.12.0",
|
"llama-index-core>=0.12.0",
|
||||||
|
"llama-index-readers-file>=0.4.0",
|
||||||
"python-dotenv>=1.0.0",
|
"python-dotenv>=1.0.0",
|
||||||
|
"sentence-transformers>=2.2.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
diskann = [
|
diskann = [
|
||||||
@@ -99,13 +96,17 @@ wechat-exporter = "wechat_exporter.main:main"
|
|||||||
leann-core = { path = "packages/leann-core", editable = true }
|
leann-core = { path = "packages/leann-core", editable = true }
|
||||||
leann-backend-diskann = { path = "packages/leann-backend-diskann", editable = true }
|
leann-backend-diskann = { path = "packages/leann-backend-diskann", editable = true }
|
||||||
leann-backend-hnsw = { path = "packages/leann-backend-hnsw", editable = true }
|
leann-backend-hnsw = { path = "packages/leann-backend-hnsw", editable = true }
|
||||||
astchunk = { path = "packages/astchunk-leann", editable = true }
|
|
||||||
|
|
||||||
[tool.ruff]
|
[tool.ruff]
|
||||||
target-version = "py39"
|
target-version = "py39"
|
||||||
line-length = 100
|
line-length = 100
|
||||||
extend-exclude = ["third_party"]
|
extend-exclude = [
|
||||||
|
"third_party",
|
||||||
|
"*.egg-info",
|
||||||
|
"__pycache__",
|
||||||
|
".git",
|
||||||
|
".venv",
|
||||||
|
]
|
||||||
|
|
||||||
[tool.ruff.lint]
|
[tool.ruff.lint]
|
||||||
select = [
|
select = [
|
||||||
@@ -128,12 +129,21 @@ ignore = [
|
|||||||
"RUF012", # mutable class attributes should be annotated with typing.ClassVar
|
"RUF012", # mutable class attributes should be annotated with typing.ClassVar
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[tool.ruff.lint.per-file-ignores]
|
||||||
|
"test/**/*.py" = ["E402"] # module level import not at top of file (common in tests)
|
||||||
|
"examples/**/*.py" = ["E402"] # module level import not at top of file (common in examples)
|
||||||
|
|
||||||
[tool.ruff.format]
|
[tool.ruff.format]
|
||||||
quote-style = "double"
|
quote-style = "double"
|
||||||
indent-style = "space"
|
indent-style = "space"
|
||||||
skip-magic-trailing-comma = false
|
skip-magic-trailing-comma = false
|
||||||
line-ending = "auto"
|
line-ending = "auto"
|
||||||
|
|
||||||
|
[dependency-groups]
|
||||||
|
dev = [
|
||||||
|
"ruff>=0.12.4",
|
||||||
|
]
|
||||||
|
|
||||||
[tool.lychee]
|
[tool.lychee]
|
||||||
accept = ["200", "403", "429", "503"]
|
accept = ["200", "403", "429", "503"]
|
||||||
timeout = 20
|
timeout = 20
|
||||||
|
|||||||
@@ -1,397 +0,0 @@
|
|||||||
"""
|
|
||||||
Test suite for astchunk integration with LEANN.
|
|
||||||
Tests AST-aware chunking functionality, language detection, and fallback mechanisms.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import os
|
|
||||||
import subprocess
|
|
||||||
import sys
|
|
||||||
import tempfile
|
|
||||||
from pathlib import Path
|
|
||||||
from unittest.mock import patch
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
# Add apps directory to path for imports
|
|
||||||
sys.path.insert(0, str(Path(__file__).parent.parent / "apps"))
|
|
||||||
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from chunking import (
|
|
||||||
create_ast_chunks,
|
|
||||||
create_text_chunks,
|
|
||||||
create_traditional_chunks,
|
|
||||||
detect_code_files,
|
|
||||||
get_language_from_extension,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class MockDocument:
|
|
||||||
"""Mock LlamaIndex Document for testing."""
|
|
||||||
|
|
||||||
def __init__(self, content: str, file_path: str = "", metadata: Optional[dict] = None):
|
|
||||||
self.content = content
|
|
||||||
self.metadata = metadata or {}
|
|
||||||
if file_path:
|
|
||||||
self.metadata["file_path"] = file_path
|
|
||||||
|
|
||||||
def get_content(self) -> str:
|
|
||||||
return self.content
|
|
||||||
|
|
||||||
|
|
||||||
class TestCodeFileDetection:
|
|
||||||
"""Test code file detection and language mapping."""
|
|
||||||
|
|
||||||
def test_detect_code_files_python(self):
|
|
||||||
"""Test detection of Python files."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("print('hello')", "/path/to/file.py"),
|
|
||||||
MockDocument("This is text", "/path/to/file.txt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
code_docs, text_docs = detect_code_files(docs)
|
|
||||||
|
|
||||||
assert len(code_docs) == 1
|
|
||||||
assert len(text_docs) == 1
|
|
||||||
assert code_docs[0].metadata["language"] == "python"
|
|
||||||
assert code_docs[0].metadata["is_code"] is True
|
|
||||||
assert text_docs[0].metadata["is_code"] is False
|
|
||||||
|
|
||||||
def test_detect_code_files_multiple_languages(self):
|
|
||||||
"""Test detection of multiple programming languages."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("def func():", "/path/to/script.py"),
|
|
||||||
MockDocument("public class Test {}", "/path/to/Test.java"),
|
|
||||||
MockDocument("interface ITest {}", "/path/to/test.ts"),
|
|
||||||
MockDocument("using System;", "/path/to/Program.cs"),
|
|
||||||
MockDocument("Regular text content", "/path/to/document.txt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
code_docs, text_docs = detect_code_files(docs)
|
|
||||||
|
|
||||||
assert len(code_docs) == 4
|
|
||||||
assert len(text_docs) == 1
|
|
||||||
|
|
||||||
languages = [doc.metadata["language"] for doc in code_docs]
|
|
||||||
assert "python" in languages
|
|
||||||
assert "java" in languages
|
|
||||||
assert "typescript" in languages
|
|
||||||
assert "csharp" in languages
|
|
||||||
|
|
||||||
def test_detect_code_files_no_file_path(self):
|
|
||||||
"""Test handling of documents without file paths."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("some content"),
|
|
||||||
MockDocument("other content", metadata={"some_key": "value"}),
|
|
||||||
]
|
|
||||||
|
|
||||||
code_docs, text_docs = detect_code_files(docs)
|
|
||||||
|
|
||||||
assert len(code_docs) == 0
|
|
||||||
assert len(text_docs) == 2
|
|
||||||
for doc in text_docs:
|
|
||||||
assert doc.metadata["is_code"] is False
|
|
||||||
|
|
||||||
def test_get_language_from_extension(self):
|
|
||||||
"""Test language detection from file extensions."""
|
|
||||||
assert get_language_from_extension("test.py") == "python"
|
|
||||||
assert get_language_from_extension("Test.java") == "java"
|
|
||||||
assert get_language_from_extension("component.tsx") == "typescript"
|
|
||||||
assert get_language_from_extension("Program.cs") == "csharp"
|
|
||||||
assert get_language_from_extension("document.txt") is None
|
|
||||||
assert get_language_from_extension("") is None
|
|
||||||
|
|
||||||
|
|
||||||
class TestChunkingFunctions:
|
|
||||||
"""Test various chunking functionality."""
|
|
||||||
|
|
||||||
def test_create_traditional_chunks(self):
|
|
||||||
"""Test traditional text chunking."""
|
|
||||||
docs = [
|
|
||||||
MockDocument(
|
|
||||||
"This is a test document. It has multiple sentences. We want to test chunking."
|
|
||||||
)
|
|
||||||
]
|
|
||||||
|
|
||||||
chunks = create_traditional_chunks(docs, chunk_size=50, chunk_overlap=10)
|
|
||||||
|
|
||||||
assert len(chunks) > 0
|
|
||||||
assert all(isinstance(chunk, str) for chunk in chunks)
|
|
||||||
assert all(len(chunk.strip()) > 0 for chunk in chunks)
|
|
||||||
|
|
||||||
def test_create_traditional_chunks_empty_docs(self):
|
|
||||||
"""Test traditional chunking with empty documents."""
|
|
||||||
chunks = create_traditional_chunks([], chunk_size=50, chunk_overlap=10)
|
|
||||||
assert chunks == []
|
|
||||||
|
|
||||||
@pytest.mark.skipif(
|
|
||||||
os.environ.get("CI") == "true",
|
|
||||||
reason="Skip astchunk tests in CI - dependency may not be available",
|
|
||||||
)
|
|
||||||
def test_create_ast_chunks_with_astchunk_available(self):
|
|
||||||
"""Test AST chunking when astchunk is available."""
|
|
||||||
python_code = '''
|
|
||||||
def hello_world():
|
|
||||||
"""Print hello world message."""
|
|
||||||
print("Hello, World!")
|
|
||||||
|
|
||||||
def add_numbers(a, b):
|
|
||||||
"""Add two numbers and return the result."""
|
|
||||||
return a + b
|
|
||||||
|
|
||||||
class Calculator:
|
|
||||||
"""A simple calculator class."""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.history = []
|
|
||||||
|
|
||||||
def add(self, a, b):
|
|
||||||
result = a + b
|
|
||||||
self.history.append(f"{a} + {b} = {result}")
|
|
||||||
return result
|
|
||||||
'''
|
|
||||||
|
|
||||||
docs = [MockDocument(python_code, "/test/calculator.py", {"language": "python"})]
|
|
||||||
|
|
||||||
try:
|
|
||||||
chunks = create_ast_chunks(docs, max_chunk_size=200, chunk_overlap=50)
|
|
||||||
|
|
||||||
# Should have multiple chunks due to different functions/classes
|
|
||||||
assert len(chunks) > 0
|
|
||||||
assert all(isinstance(chunk, str) for chunk in chunks)
|
|
||||||
assert all(len(chunk.strip()) > 0 for chunk in chunks)
|
|
||||||
|
|
||||||
# Check that code structure is somewhat preserved
|
|
||||||
combined_content = " ".join(chunks)
|
|
||||||
assert "def hello_world" in combined_content
|
|
||||||
assert "class Calculator" in combined_content
|
|
||||||
|
|
||||||
except ImportError:
|
|
||||||
# astchunk not available, should fall back to traditional chunking
|
|
||||||
chunks = create_ast_chunks(docs, max_chunk_size=200, chunk_overlap=50)
|
|
||||||
assert len(chunks) > 0 # Should still get chunks from fallback
|
|
||||||
|
|
||||||
def test_create_ast_chunks_fallback_to_traditional(self):
|
|
||||||
"""Test AST chunking falls back to traditional when astchunk is not available."""
|
|
||||||
docs = [MockDocument("def test(): pass", "/test/script.py", {"language": "python"})]
|
|
||||||
|
|
||||||
# Mock astchunk import to fail
|
|
||||||
with patch("chunking.create_ast_chunks"):
|
|
||||||
# First call (actual test) should import astchunk and potentially fail
|
|
||||||
# Let's call the actual function to test the import error handling
|
|
||||||
chunks = create_ast_chunks(docs)
|
|
||||||
|
|
||||||
# Should return some chunks (either from astchunk or fallback)
|
|
||||||
assert isinstance(chunks, list)
|
|
||||||
|
|
||||||
def test_create_text_chunks_traditional_mode(self):
|
|
||||||
"""Test text chunking in traditional mode."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("def test(): pass", "/test/script.py"),
|
|
||||||
MockDocument("This is regular text.", "/test/doc.txt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
chunks = create_text_chunks(docs, use_ast_chunking=False, chunk_size=50, chunk_overlap=10)
|
|
||||||
|
|
||||||
assert len(chunks) > 0
|
|
||||||
assert all(isinstance(chunk, str) for chunk in chunks)
|
|
||||||
|
|
||||||
def test_create_text_chunks_ast_mode(self):
|
|
||||||
"""Test text chunking in AST mode."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("def test(): pass", "/test/script.py"),
|
|
||||||
MockDocument("This is regular text.", "/test/doc.txt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
chunks = create_text_chunks(
|
|
||||||
docs,
|
|
||||||
use_ast_chunking=True,
|
|
||||||
ast_chunk_size=100,
|
|
||||||
ast_chunk_overlap=20,
|
|
||||||
chunk_size=50,
|
|
||||||
chunk_overlap=10,
|
|
||||||
)
|
|
||||||
|
|
||||||
assert len(chunks) > 0
|
|
||||||
assert all(isinstance(chunk, str) for chunk in chunks)
|
|
||||||
|
|
||||||
def test_create_text_chunks_custom_extensions(self):
|
|
||||||
"""Test text chunking with custom code file extensions."""
|
|
||||||
docs = [
|
|
||||||
MockDocument("function test() {}", "/test/script.js"), # Not in default extensions
|
|
||||||
MockDocument("Regular text", "/test/doc.txt"),
|
|
||||||
]
|
|
||||||
|
|
||||||
# First without custom extensions - should treat .js as text
|
|
||||||
chunks_without = create_text_chunks(docs, use_ast_chunking=True, code_file_extensions=None)
|
|
||||||
|
|
||||||
# Then with custom extensions - should treat .js as code
|
|
||||||
chunks_with = create_text_chunks(
|
|
||||||
docs, use_ast_chunking=True, code_file_extensions=[".js", ".jsx"]
|
|
||||||
)
|
|
||||||
|
|
||||||
# Both should return chunks
|
|
||||||
assert len(chunks_without) > 0
|
|
||||||
assert len(chunks_with) > 0
|
|
||||||
|
|
||||||
|
|
||||||
class TestIntegrationWithDocumentRAG:
|
|
||||||
"""Integration tests with the document RAG system."""
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def temp_code_dir(self):
|
|
||||||
"""Create a temporary directory with sample code files."""
|
|
||||||
with tempfile.TemporaryDirectory() as temp_dir:
|
|
||||||
temp_path = Path(temp_dir)
|
|
||||||
|
|
||||||
# Create sample Python file
|
|
||||||
python_file = temp_path / "example.py"
|
|
||||||
python_file.write_text('''
|
|
||||||
def fibonacci(n):
|
|
||||||
"""Calculate fibonacci number."""
|
|
||||||
if n <= 1:
|
|
||||||
return n
|
|
||||||
return fibonacci(n-1) + fibonacci(n-2)
|
|
||||||
|
|
||||||
class MathUtils:
|
|
||||||
@staticmethod
|
|
||||||
def factorial(n):
|
|
||||||
if n <= 1:
|
|
||||||
return 1
|
|
||||||
return n * MathUtils.factorial(n-1)
|
|
||||||
''')
|
|
||||||
|
|
||||||
# Create sample text file
|
|
||||||
text_file = temp_path / "readme.txt"
|
|
||||||
text_file.write_text("This is a sample text file for testing purposes.")
|
|
||||||
|
|
||||||
yield temp_path
|
|
||||||
|
|
||||||
@pytest.mark.skipif(
|
|
||||||
os.environ.get("CI") == "true",
|
|
||||||
reason="Skip integration tests in CI to avoid dependency issues",
|
|
||||||
)
|
|
||||||
def test_document_rag_with_ast_chunking(self, temp_code_dir):
|
|
||||||
"""Test document RAG with AST chunking enabled."""
|
|
||||||
with tempfile.TemporaryDirectory() as index_dir:
|
|
||||||
cmd = [
|
|
||||||
sys.executable,
|
|
||||||
"apps/document_rag.py",
|
|
||||||
"--llm",
|
|
||||||
"simulated",
|
|
||||||
"--embedding-model",
|
|
||||||
"facebook/contriever",
|
|
||||||
"--embedding-mode",
|
|
||||||
"sentence-transformers",
|
|
||||||
"--index-dir",
|
|
||||||
index_dir,
|
|
||||||
"--data-dir",
|
|
||||||
str(temp_code_dir),
|
|
||||||
"--enable-code-chunking",
|
|
||||||
"--query",
|
|
||||||
"How does the fibonacci function work?",
|
|
||||||
]
|
|
||||||
|
|
||||||
env = os.environ.copy()
|
|
||||||
env["HF_HUB_DISABLE_SYMLINKS"] = "1"
|
|
||||||
env["TOKENIZERS_PARALLELISM"] = "false"
|
|
||||||
|
|
||||||
try:
|
|
||||||
result = subprocess.run(
|
|
||||||
cmd,
|
|
||||||
capture_output=True,
|
|
||||||
text=True,
|
|
||||||
timeout=300, # 5 minutes
|
|
||||||
env=env,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Should succeed even if astchunk is not available (fallback)
|
|
||||||
assert result.returncode == 0, f"Command failed: {result.stderr}"
|
|
||||||
|
|
||||||
output = result.stdout + result.stderr
|
|
||||||
assert "Index saved to" in output or "Using existing index" in output
|
|
||||||
|
|
||||||
except subprocess.TimeoutExpired:
|
|
||||||
pytest.skip("Test timed out - likely due to model download in CI")
|
|
||||||
|
|
||||||
@pytest.mark.skipif(
|
|
||||||
os.environ.get("CI") == "true",
|
|
||||||
reason="Skip integration tests in CI to avoid dependency issues",
|
|
||||||
)
|
|
||||||
def test_code_rag_application(self, temp_code_dir):
|
|
||||||
"""Test the specialized code RAG application."""
|
|
||||||
with tempfile.TemporaryDirectory() as index_dir:
|
|
||||||
cmd = [
|
|
||||||
sys.executable,
|
|
||||||
"apps/code_rag.py",
|
|
||||||
"--llm",
|
|
||||||
"simulated",
|
|
||||||
"--embedding-model",
|
|
||||||
"facebook/contriever",
|
|
||||||
"--index-dir",
|
|
||||||
index_dir,
|
|
||||||
"--repo-dir",
|
|
||||||
str(temp_code_dir),
|
|
||||||
"--query",
|
|
||||||
"What classes are defined in this code?",
|
|
||||||
]
|
|
||||||
|
|
||||||
env = os.environ.copy()
|
|
||||||
env["HF_HUB_DISABLE_SYMLINKS"] = "1"
|
|
||||||
env["TOKENIZERS_PARALLELISM"] = "false"
|
|
||||||
|
|
||||||
try:
|
|
||||||
result = subprocess.run(cmd, capture_output=True, text=True, timeout=300, env=env)
|
|
||||||
|
|
||||||
# Should succeed
|
|
||||||
assert result.returncode == 0, f"Command failed: {result.stderr}"
|
|
||||||
|
|
||||||
output = result.stdout + result.stderr
|
|
||||||
assert "Using AST-aware chunking" in output or "traditional chunking" in output
|
|
||||||
|
|
||||||
except subprocess.TimeoutExpired:
|
|
||||||
pytest.skip("Test timed out - likely due to model download in CI")
|
|
||||||
|
|
||||||
|
|
||||||
class TestErrorHandling:
|
|
||||||
"""Test error handling and edge cases."""
|
|
||||||
|
|
||||||
def test_text_chunking_empty_documents(self):
|
|
||||||
"""Test text chunking with empty document list."""
|
|
||||||
chunks = create_text_chunks([])
|
|
||||||
assert chunks == []
|
|
||||||
|
|
||||||
def test_text_chunking_invalid_parameters(self):
|
|
||||||
"""Test text chunking with invalid parameters."""
|
|
||||||
docs = [MockDocument("test content")]
|
|
||||||
|
|
||||||
# Should handle negative chunk sizes gracefully
|
|
||||||
chunks = create_text_chunks(
|
|
||||||
docs, chunk_size=0, chunk_overlap=0, ast_chunk_size=0, ast_chunk_overlap=0
|
|
||||||
)
|
|
||||||
|
|
||||||
# Should still return some result
|
|
||||||
assert isinstance(chunks, list)
|
|
||||||
|
|
||||||
def test_create_ast_chunks_no_language(self):
|
|
||||||
"""Test AST chunking with documents missing language metadata."""
|
|
||||||
docs = [MockDocument("def test(): pass", "/test/script.py")] # No language set
|
|
||||||
|
|
||||||
chunks = create_ast_chunks(docs)
|
|
||||||
|
|
||||||
# Should fall back to traditional chunking
|
|
||||||
assert isinstance(chunks, list)
|
|
||||||
assert len(chunks) >= 0 # May be empty if fallback also fails
|
|
||||||
|
|
||||||
def test_create_ast_chunks_empty_content(self):
|
|
||||||
"""Test AST chunking with empty content."""
|
|
||||||
docs = [MockDocument("", "/test/script.py", {"language": "python"})]
|
|
||||||
|
|
||||||
chunks = create_ast_chunks(docs)
|
|
||||||
|
|
||||||
# Should handle empty content gracefully
|
|
||||||
assert isinstance(chunks, list)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
pytest.main([__file__, "-v"])
|
|
||||||
@@ -57,51 +57,6 @@ def test_document_rag_simulated(test_data_dir):
|
|||||||
assert "This is a simulated answer" in output
|
assert "This is a simulated answer" in output
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.skipif(
|
|
||||||
os.environ.get("CI") == "true",
|
|
||||||
reason="Skip AST chunking tests in CI to avoid dependency issues",
|
|
||||||
)
|
|
||||||
def test_document_rag_with_ast_chunking(test_data_dir):
|
|
||||||
"""Test document_rag with AST-aware chunking enabled."""
|
|
||||||
with tempfile.TemporaryDirectory() as temp_dir:
|
|
||||||
# Use a subdirectory that doesn't exist yet to force index creation
|
|
||||||
index_dir = Path(temp_dir) / "test_ast_index"
|
|
||||||
cmd = [
|
|
||||||
sys.executable,
|
|
||||||
"apps/document_rag.py",
|
|
||||||
"--llm",
|
|
||||||
"simulated",
|
|
||||||
"--embedding-model",
|
|
||||||
"facebook/contriever",
|
|
||||||
"--embedding-mode",
|
|
||||||
"sentence-transformers",
|
|
||||||
"--index-dir",
|
|
||||||
str(index_dir),
|
|
||||||
"--data-dir",
|
|
||||||
str(test_data_dir),
|
|
||||||
"--enable-code-chunking", # Enable AST chunking
|
|
||||||
"--query",
|
|
||||||
"What is Pride and Prejudice about?",
|
|
||||||
]
|
|
||||||
|
|
||||||
env = os.environ.copy()
|
|
||||||
env["HF_HUB_DISABLE_SYMLINKS"] = "1"
|
|
||||||
env["TOKENIZERS_PARALLELISM"] = "false"
|
|
||||||
|
|
||||||
result = subprocess.run(cmd, capture_output=True, text=True, timeout=600, env=env)
|
|
||||||
|
|
||||||
# Check return code
|
|
||||||
assert result.returncode == 0, f"Command failed: {result.stderr}"
|
|
||||||
|
|
||||||
# Verify output
|
|
||||||
output = result.stdout + result.stderr
|
|
||||||
assert "Index saved to" in output or "Using existing index" in output
|
|
||||||
assert "This is a simulated answer" in output
|
|
||||||
|
|
||||||
# Should mention AST chunking if code files are present
|
|
||||||
# (might not be relevant for the test data, but command should succeed)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.skipif(not os.environ.get("OPENAI_API_KEY"), reason="OpenAI API key not available")
|
@pytest.mark.skipif(not os.environ.get("OPENAI_API_KEY"), reason="OpenAI API key not available")
|
||||||
@pytest.mark.skipif(
|
@pytest.mark.skipif(
|
||||||
os.environ.get("CI") == "true", reason="Skip OpenAI tests in CI to avoid API costs"
|
os.environ.get("CI") == "true", reason="Skip OpenAI tests in CI to avoid API costs"
|
||||||
|
|||||||
@@ -1,365 +0,0 @@
|
|||||||
"""
|
|
||||||
Comprehensive tests for metadata filtering functionality.
|
|
||||||
|
|
||||||
This module tests the MetadataFilterEngine class and its integration
|
|
||||||
with the LEANN search system.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import os
|
|
||||||
|
|
||||||
# Import the modules we're testing
|
|
||||||
import sys
|
|
||||||
from unittest.mock import Mock, patch
|
|
||||||
|
|
||||||
sys.path.insert(0, os.path.join(os.path.dirname(__file__), "../packages/leann-core/src"))
|
|
||||||
|
|
||||||
from leann.api import PassageManager, SearchResult
|
|
||||||
from leann.metadata_filter import MetadataFilterEngine
|
|
||||||
|
|
||||||
|
|
||||||
class TestMetadataFilterEngine:
|
|
||||||
"""Test suite for the MetadataFilterEngine class."""
|
|
||||||
|
|
||||||
def setup_method(self):
|
|
||||||
"""Setup test fixtures."""
|
|
||||||
self.engine = MetadataFilterEngine()
|
|
||||||
|
|
||||||
# Sample search results for testing
|
|
||||||
self.sample_results = [
|
|
||||||
{
|
|
||||||
"id": "doc1",
|
|
||||||
"score": 0.95,
|
|
||||||
"text": "This is chapter 1 content",
|
|
||||||
"metadata": {
|
|
||||||
"chapter": 1,
|
|
||||||
"character": "Alice",
|
|
||||||
"tags": ["adventure", "fantasy"],
|
|
||||||
"word_count": 150,
|
|
||||||
"is_published": True,
|
|
||||||
"genre": "fiction",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "doc2",
|
|
||||||
"score": 0.87,
|
|
||||||
"text": "This is chapter 3 content",
|
|
||||||
"metadata": {
|
|
||||||
"chapter": 3,
|
|
||||||
"character": "Bob",
|
|
||||||
"tags": ["mystery", "thriller"],
|
|
||||||
"word_count": 250,
|
|
||||||
"is_published": True,
|
|
||||||
"genre": "fiction",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "doc3",
|
|
||||||
"score": 0.82,
|
|
||||||
"text": "This is chapter 5 content",
|
|
||||||
"metadata": {
|
|
||||||
"chapter": 5,
|
|
||||||
"character": "Alice",
|
|
||||||
"tags": ["romance", "drama"],
|
|
||||||
"word_count": 300,
|
|
||||||
"is_published": False,
|
|
||||||
"genre": "non-fiction",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "doc4",
|
|
||||||
"score": 0.78,
|
|
||||||
"text": "This is chapter 10 content",
|
|
||||||
"metadata": {
|
|
||||||
"chapter": 10,
|
|
||||||
"character": "Charlie",
|
|
||||||
"tags": ["action", "adventure"],
|
|
||||||
"word_count": 400,
|
|
||||||
"is_published": True,
|
|
||||||
"genre": "fiction",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
def test_engine_initialization(self):
|
|
||||||
"""Test that the filter engine initializes correctly."""
|
|
||||||
assert self.engine is not None
|
|
||||||
assert len(self.engine.operators) > 0
|
|
||||||
assert "==" in self.engine.operators
|
|
||||||
assert "contains" in self.engine.operators
|
|
||||||
assert "in" in self.engine.operators
|
|
||||||
|
|
||||||
def test_direct_instantiation(self):
|
|
||||||
"""Test direct instantiation of the engine."""
|
|
||||||
engine = MetadataFilterEngine()
|
|
||||||
assert isinstance(engine, MetadataFilterEngine)
|
|
||||||
|
|
||||||
def test_no_filters_returns_all_results(self):
|
|
||||||
"""Test that passing None or empty filters returns all results."""
|
|
||||||
# Test with None
|
|
||||||
result = self.engine.apply_filters(self.sample_results, None)
|
|
||||||
assert len(result) == len(self.sample_results)
|
|
||||||
|
|
||||||
# Test with empty dict
|
|
||||||
result = self.engine.apply_filters(self.sample_results, {})
|
|
||||||
assert len(result) == len(self.sample_results)
|
|
||||||
|
|
||||||
# Test comparison operators
|
|
||||||
def test_equals_filter(self):
|
|
||||||
"""Test equals (==) filter."""
|
|
||||||
filters = {"chapter": {"==": 1}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 1
|
|
||||||
assert result[0]["id"] == "doc1"
|
|
||||||
|
|
||||||
def test_not_equals_filter(self):
|
|
||||||
"""Test not equals (!=) filter."""
|
|
||||||
filters = {"genre": {"!=": "fiction"}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 1
|
|
||||||
assert result[0]["metadata"]["genre"] == "non-fiction"
|
|
||||||
|
|
||||||
def test_less_than_filter(self):
|
|
||||||
"""Test less than (<) filter."""
|
|
||||||
filters = {"chapter": {"<": 5}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 2
|
|
||||||
chapters = [r["metadata"]["chapter"] for r in result]
|
|
||||||
assert all(ch < 5 for ch in chapters)
|
|
||||||
|
|
||||||
def test_less_than_or_equal_filter(self):
|
|
||||||
"""Test less than or equal (<=) filter."""
|
|
||||||
filters = {"chapter": {"<=": 5}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 3
|
|
||||||
chapters = [r["metadata"]["chapter"] for r in result]
|
|
||||||
assert all(ch <= 5 for ch in chapters)
|
|
||||||
|
|
||||||
def test_greater_than_filter(self):
|
|
||||||
"""Test greater than (>) filter."""
|
|
||||||
filters = {"word_count": {">": 200}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 3 # Documents with word_count 250, 300, 400
|
|
||||||
word_counts = [r["metadata"]["word_count"] for r in result]
|
|
||||||
assert all(wc > 200 for wc in word_counts)
|
|
||||||
|
|
||||||
def test_greater_than_or_equal_filter(self):
|
|
||||||
"""Test greater than or equal (>=) filter."""
|
|
||||||
filters = {"word_count": {">=": 250}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 3
|
|
||||||
word_counts = [r["metadata"]["word_count"] for r in result]
|
|
||||||
assert all(wc >= 250 for wc in word_counts)
|
|
||||||
|
|
||||||
# Test membership operators
|
|
||||||
def test_in_filter(self):
|
|
||||||
"""Test in filter."""
|
|
||||||
filters = {"character": {"in": ["Alice", "Bob"]}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 3
|
|
||||||
characters = [r["metadata"]["character"] for r in result]
|
|
||||||
assert all(ch in ["Alice", "Bob"] for ch in characters)
|
|
||||||
|
|
||||||
def test_not_in_filter(self):
|
|
||||||
"""Test not_in filter."""
|
|
||||||
filters = {"character": {"not_in": ["Alice", "Bob"]}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 1
|
|
||||||
assert result[0]["metadata"]["character"] == "Charlie"
|
|
||||||
|
|
||||||
# Test string operators
|
|
||||||
def test_contains_filter(self):
|
|
||||||
"""Test contains filter."""
|
|
||||||
filters = {"genre": {"contains": "fiction"}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 4 # Both "fiction" and "non-fiction"
|
|
||||||
|
|
||||||
def test_starts_with_filter(self):
|
|
||||||
"""Test starts_with filter."""
|
|
||||||
filters = {"genre": {"starts_with": "non"}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 1
|
|
||||||
assert result[0]["metadata"]["genre"] == "non-fiction"
|
|
||||||
|
|
||||||
def test_ends_with_filter(self):
|
|
||||||
"""Test ends_with filter."""
|
|
||||||
filters = {"text": {"ends_with": "content"}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 4 # All sample texts end with "content"
|
|
||||||
|
|
||||||
# Test boolean operators
|
|
||||||
def test_is_true_filter(self):
|
|
||||||
"""Test is_true filter."""
|
|
||||||
filters = {"is_published": {"is_true": True}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 3
|
|
||||||
assert all(r["metadata"]["is_published"] for r in result)
|
|
||||||
|
|
||||||
def test_is_false_filter(self):
|
|
||||||
"""Test is_false filter."""
|
|
||||||
filters = {"is_published": {"is_false": False}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 1
|
|
||||||
assert not result[0]["metadata"]["is_published"]
|
|
||||||
|
|
||||||
# Test compound filters (AND logic)
|
|
||||||
def test_compound_filters(self):
|
|
||||||
"""Test multiple filters applied together (AND logic)."""
|
|
||||||
filters = {"genre": {"==": "fiction"}, "chapter": {"<=": 5}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 2
|
|
||||||
for r in result:
|
|
||||||
assert r["metadata"]["genre"] == "fiction"
|
|
||||||
assert r["metadata"]["chapter"] <= 5
|
|
||||||
|
|
||||||
def test_multiple_operators_same_field(self):
|
|
||||||
"""Test multiple operators on the same field."""
|
|
||||||
filters = {"word_count": {">=": 200, "<=": 350}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 2
|
|
||||||
for r in result:
|
|
||||||
wc = r["metadata"]["word_count"]
|
|
||||||
assert 200 <= wc <= 350
|
|
||||||
|
|
||||||
# Test edge cases
|
|
||||||
def test_missing_field_fails_filter(self):
|
|
||||||
"""Test that missing metadata fields fail filters."""
|
|
||||||
filters = {"nonexistent_field": {"==": "value"}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 0
|
|
||||||
|
|
||||||
def test_invalid_operator(self):
|
|
||||||
"""Test that invalid operators are handled gracefully."""
|
|
||||||
filters = {"chapter": {"invalid_op": 1}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 0 # Should filter out all results
|
|
||||||
|
|
||||||
def test_type_coercion_numeric(self):
|
|
||||||
"""Test numeric type coercion in comparisons."""
|
|
||||||
# Add a result with string chapter number
|
|
||||||
test_results = [
|
|
||||||
*self.sample_results,
|
|
||||||
{
|
|
||||||
"id": "doc5",
|
|
||||||
"score": 0.75,
|
|
||||||
"text": "String chapter test",
|
|
||||||
"metadata": {"chapter": "2", "genre": "test"},
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
filters = {"chapter": {"<": 3}}
|
|
||||||
result = self.engine.apply_filters(test_results, filters)
|
|
||||||
# Should include doc1 (chapter=1) and doc5 (chapter="2")
|
|
||||||
assert len(result) == 2
|
|
||||||
ids = [r["id"] for r in result]
|
|
||||||
assert "doc1" in ids
|
|
||||||
assert "doc5" in ids
|
|
||||||
|
|
||||||
def test_list_membership_with_nested_tags(self):
|
|
||||||
"""Test membership operations with list metadata."""
|
|
||||||
# Note: This tests the metadata structure, not list field filtering
|
|
||||||
# For list field filtering, we'd need to modify the test data
|
|
||||||
filters = {"character": {"in": ["Alice"]}}
|
|
||||||
result = self.engine.apply_filters(self.sample_results, filters)
|
|
||||||
assert len(result) == 2
|
|
||||||
assert all(r["metadata"]["character"] == "Alice" for r in result)
|
|
||||||
|
|
||||||
def test_empty_results_list(self):
|
|
||||||
"""Test filtering on empty results list."""
|
|
||||||
filters = {"chapter": {"==": 1}}
|
|
||||||
result = self.engine.apply_filters([], filters)
|
|
||||||
assert len(result) == 0
|
|
||||||
|
|
||||||
|
|
||||||
class TestPassageManagerFiltering:
|
|
||||||
"""Test suite for PassageManager filtering integration."""
|
|
||||||
|
|
||||||
def setup_method(self):
|
|
||||||
"""Setup test fixtures."""
|
|
||||||
# Mock the passage manager without actual file I/O
|
|
||||||
self.passage_manager = Mock(spec=PassageManager)
|
|
||||||
self.passage_manager.filter_engine = MetadataFilterEngine()
|
|
||||||
|
|
||||||
# Sample SearchResult objects
|
|
||||||
self.search_results = [
|
|
||||||
SearchResult(
|
|
||||||
id="doc1",
|
|
||||||
score=0.95,
|
|
||||||
text="Chapter 1 content",
|
|
||||||
metadata={"chapter": 1, "character": "Alice"},
|
|
||||||
),
|
|
||||||
SearchResult(
|
|
||||||
id="doc2",
|
|
||||||
score=0.87,
|
|
||||||
text="Chapter 5 content",
|
|
||||||
metadata={"chapter": 5, "character": "Bob"},
|
|
||||||
),
|
|
||||||
SearchResult(
|
|
||||||
id="doc3",
|
|
||||||
score=0.82,
|
|
||||||
text="Chapter 10 content",
|
|
||||||
metadata={"chapter": 10, "character": "Alice"},
|
|
||||||
),
|
|
||||||
]
|
|
||||||
|
|
||||||
def test_search_result_filtering(self):
|
|
||||||
"""Test filtering SearchResult objects."""
|
|
||||||
# Create a real PassageManager instance just for the filtering method
|
|
||||||
# We'll mock the file operations
|
|
||||||
with patch("builtins.open"), patch("json.loads"), patch("pickle.load"):
|
|
||||||
pm = PassageManager([{"type": "jsonl", "path": "test.jsonl"}])
|
|
||||||
|
|
||||||
filters = {"chapter": {"<=": 5}}
|
|
||||||
result = pm.filter_search_results(self.search_results, filters)
|
|
||||||
|
|
||||||
assert len(result) == 2
|
|
||||||
chapters = [r.metadata["chapter"] for r in result]
|
|
||||||
assert all(ch <= 5 for ch in chapters)
|
|
||||||
|
|
||||||
def test_filter_search_results_no_filters(self):
|
|
||||||
"""Test that None filters return all results."""
|
|
||||||
with patch("builtins.open"), patch("json.loads"), patch("pickle.load"):
|
|
||||||
pm = PassageManager([{"type": "jsonl", "path": "test.jsonl"}])
|
|
||||||
|
|
||||||
result = pm.filter_search_results(self.search_results, None)
|
|
||||||
assert len(result) == len(self.search_results)
|
|
||||||
|
|
||||||
def test_filter_maintains_search_result_type(self):
|
|
||||||
"""Test that filtering returns SearchResult objects."""
|
|
||||||
with patch("builtins.open"), patch("json.loads"), patch("pickle.load"):
|
|
||||||
pm = PassageManager([{"type": "jsonl", "path": "test.jsonl"}])
|
|
||||||
|
|
||||||
filters = {"character": {"==": "Alice"}}
|
|
||||||
result = pm.filter_search_results(self.search_results, filters)
|
|
||||||
|
|
||||||
assert len(result) == 2
|
|
||||||
for r in result:
|
|
||||||
assert isinstance(r, SearchResult)
|
|
||||||
assert r.metadata["character"] == "Alice"
|
|
||||||
|
|
||||||
|
|
||||||
# Integration tests would go here, but they require actual LEANN backend setup
|
|
||||||
# These would test the full pipeline from LeannSearcher.search() with metadata_filters
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
# Run basic smoke tests
|
|
||||||
engine = MetadataFilterEngine()
|
|
||||||
|
|
||||||
sample_data = [
|
|
||||||
{
|
|
||||||
"id": "test1",
|
|
||||||
"score": 0.9,
|
|
||||||
"text": "Test content",
|
|
||||||
"metadata": {"chapter": 1, "published": True},
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
# Test basic filtering
|
|
||||||
result = engine.apply_filters(sample_data, {"chapter": {"==": 1}})
|
|
||||||
assert len(result) == 1
|
|
||||||
print("✅ Basic filtering test passed")
|
|
||||||
|
|
||||||
result = engine.apply_filters(sample_data, {"chapter": {"==": 2}})
|
|
||||||
assert len(result) == 0
|
|
||||||
print("✅ No match filtering test passed")
|
|
||||||
|
|
||||||
print("🎉 All smoke tests passed!")
|
|
||||||
Reference in New Issue
Block a user