---
Kazakh is heavily agglutinative: a single word like `мектептерімізде` carries plural, possessive, and locative suffixes that must all be stripped to reach the root `мектеп`. No existing PostgreSQL or Elasticsearch analyzer handles this. pg_kazsearch fills that gap with a Rust stemmer that plugs into both PostgreSQL (via [pgrx](https://github.com/pgcentralfoundation/pgrx)) and Elasticsearch (via JNI native plugin).
```sql
-- PostgreSQL
CREATE EXTENSION pg_kazsearch;
SELECT to_tsvector('kazakh_cfg', 'президенттің жарлығы');
-- 'жарлық':2 'президент':1
```
```json
// Elasticsearch
{ "filter": { "kaz_stem": { "type": "kazsearch_stem" } } }
// алмаларымыздағы → алма
// мектептеріміздегі → мектеп
// almalar → алма
// mektepterimizdegi → мектеп
```
Latin-script Kazakh is auto-detected and normalized to canonical Cyrillic inside the core stemmer. Successful Latin and Cyrillic inputs therefore converge to the same stem output (always Cyrillic), which keeps indexing and query matching unified across scripts.
Current scope of Latin support:
- Targets the official modern Kazakh Latin orthography first (`ä ö ü ū ğ ş ñ ı`, plus `q`/`w`).
- Leaves mixed-script, unsupported Latin variants (apostrophe/acute/digraph legacy spellings), and low-confidence ASCII tokens unchanged.
---
## Install
### Pre-built package (Debian/Ubuntu)
Download the `.deb` for your PostgreSQL version from [GitHub Releases](https://github.com/darkhanakh/pg-kazsearch/releases):
```bash
# Example: PostgreSQL 18 on amd64
curl -LO https://github.com/darkhanakh/pg-kazsearch/releases/latest/download/postgresql-18-pg-kazsearch_2.3.0_amd64.deb
sudo dpkg -i postgresql-18-pg-kazsearch_2.3.0_amd64.deb
```
Then in psql:
```sql
CREATE EXTENSION pg_kazsearch;
```
### Docker
Use the pre-built image as a drop-in replacement for `postgres`:
```yaml
# docker-compose.yml
services:
db:
image: ghcr.io/darkhanakh/pg-kazsearch:18
```
Or add to your existing Dockerfile:
```dockerfile
FROM ghcr.io/darkhanakh/pg-kazsearch:18 AS kazsearch
FROM postgres:18
COPY --from=kazsearch /usr/share/postgresql/18/extension/pg_kazsearch* /usr/share/postgresql/18/extension/
COPY --from=kazsearch /usr/lib/postgresql/18/lib/pg_kazsearch* /usr/lib/postgresql/18/lib/
COPY --from=kazsearch /usr/share/postgresql/18/tsearch_data/kaz_* /usr/share/postgresql/18/tsearch_data/
```
### From source
```bash
# Requires: Rust toolchain, cargo-pgrx, postgresql-server-dev
cargo install --locked cargo-pgrx --version "=0.17.0"
cargo pgrx init --pg18 $(which pg_config)
git clone https://github.com/darkhanakh/pg-kazsearch.git
cd pg-kazsearch
cargo pgrx install --release -p pg_kazsearch
# Install lexicon (+ verb-lemma sibling) and stopwords
cp data/tsearch_data/kaz_stems.dict $(pg_config --sharedir)/tsearch_data/
cp data/tsearch_data/kaz_stems.dict.verbs $(pg_config --sharedir)/tsearch_data/
cp data/tsearch_data/kaz_stopwords.stop $(pg_config --sharedir)/tsearch_data/
```
---
## Elasticsearch
The same Kazakh stemmer is available as an Elasticsearch analysis plugin (`kazsearch_stem` token filter). All stemmer logic stays in Rust — the Java side is a thin JNI bridge.
### Install from GitHub Releases
Elasticsearch plugins install only on the exact ES version they were built for, so each release ships one ZIP per supported ES version (`8.17.0`, `8.17.10`, `8.18.8`, `8.19.18`). Download the ZIP matching your cluster from [GitHub Releases](https://github.com/darkhanakh/pg-kazsearch/releases) and install:
```bash
# Example: Elasticsearch 8.18.8
bin/elasticsearch-plugin install https://github.com/darkhanakh/pg-kazsearch/releases/latest/download/analysis-kazsearch-2.3.0-es8.18.8.zip
```
Each pre-built ZIP includes native libraries for linux/amd64 and linux/aarch64 plus the bundled lexicon. If your ES version isn't listed, build from source with `-PesVersion=` (see below) — the Java bridge compiles unchanged across 8.17–8.19.
### Configuration
```json
{
"settings": {
"analysis": {
"filter": {
"kaz_stem": { "type": "kazsearch_stem" }
},
"analyzer": {
"kazakh": {
"type": "custom",
"tokenizer": "standard",
"filter": ["lowercase", "kaz_stem"]
}
}
}
}
}
```
Optional filter settings:
- `lexicon_path` — absolute path to a lexicon dict file, overriding the `data/kaz_stems.dict` bundled with the plugin (loaded automatically)
- `script_mode` — `auto` (default; Latin-script Kazakh is transliterated and stemmed) or `cyrillic_only`
The plugin locates and loads its native library (`.so`/`.dylib`) from the installed plugin directory at runtime — no `LD_LIBRARY_PATH` or post-install copy step is needed.
### Verify
```bash
curl -X POST 'localhost:9200/my_index/_analyze' \
-H 'Content-Type: application/json' \
-d '{"analyzer": "kazakh", "text": "алмаларымыздағы мектептеріміздегі"}'
# → tokens: ["алма", "мектеп"]
```
### Build from source
Requires: Rust toolchain, JDK 21, Gradle 8+, and `cargo-zigbuild` for cross-compilation.
```bash
# Build Rust cdylib (native stemmer library)
just es-native
# Build ES plugin ZIP (includes Java bridge + native lib)
just es-build
# → elastic/java/build/distributions/analysis-kazsearch-2.3.0.zip
# Target a specific ES version (stamps plugin-descriptor.properties)
cd elastic/java && gradle bundlePlugin -PesVersion=8.18.8
# → build/distributions/analysis-kazsearch-2.3.0-es8.18.8.zip
# Run tests
just es-up
just es-load-corpus # index 3000 articles
just es-eval # run search quality evaluation
```
---
## Upgrading
**Stemmer upgrades require reindexing.** Releases routinely improve the stemmer, which changes its output for some words. Anything indexed with the old version keeps the *old* stems, while queries are stemmed with the *new* code — the two sides silently stop matching for exactly the words the upgrade improved. Upgrading the binary without reindexing makes search quality *worse*, not better.
**PostgreSQL** — after installing the new package:
```sql
ALTER EXTENSION pg_kazsearch UPDATE;
-- Force recompute of STORED generated tsvector columns; a no-op UPDATE
-- rewrites each row, regenerating the column (the GIN index follows
-- automatically):
UPDATE articles SET title = title;
VACUUM (ANALYZE) articles;
```
If you populate tsvector columns with triggers instead of generated columns, re-run your population query. New sessions pick up the new dictionary automatically; long-lived sessions from before the upgrade should reconnect.
**Elasticsearch** — remove the old plugin, install the version-matching new ZIP on every node, restart, then reindex affected indices (`POST _reindex` into a fresh index or re-ingest from source). The `_analyze` endpoint is a quick way to confirm the new stemmer is live before reindexing.
---
## Usage (PostgreSQL)
The extension creates everything automatically — a text search template, dictionaries, and a ready-to-use configuration called `kazakh_cfg`:
```sql
CREATE EXTENSION pg_kazsearch;
-- Stem individual words
SELECT ts_lexize('pg_kazsearch_dict', 'алмаларымыздағы');
-- {алма}
-- Build tsvectors
SELECT to_tsvector('kazakh_cfg', 'мектептеріміздегі оқушылардың');
-- 'мектеп':1 'оқушы':2
-- Add FTS to a table
ALTER TABLE articles ADD COLUMN fts tsvector
GENERATED ALWAYS AS (
setweight(to_tsvector('kazakh_cfg', title), 'A') ||
setweight(to_tsvector('kazakh_cfg', body), 'B')
) STORED;
CREATE INDEX idx_fts ON articles USING GIN (fts);
-- Search
SELECT title FROM articles
WHERE fts @@ websearch_to_tsquery('kazakh_cfg', 'президенттің жарлығы')
ORDER BY ts_rank_cd(fts, websearch_to_tsquery('kazakh_cfg', 'президенттің жарлығы')) DESC
LIMIT 10;
```
### Recommended query pattern: AND with OR fallback
`websearch_to_tsquery` produces strict AND semantics — one query term with no match anywhere kills the entire result set, even when every other term hits. On our human-query benchmark this is the single largest source of failed searches. Measured on gold_v2 (n=132), same index, same ranking function, only the query construction differs:
| Strategy | P@10 | R@10 | MRR@10 | R@50 | Queries with no results |
| --------------------------- | --------- | --------- | --------- | --------- | ----------------------- |
| AND (`websearch_to_tsquery`)| 0.352 | 0.542 | 0.743 | 0.668 | 16 / 132 |
| Pure OR | 0.230 | 0.392 | 0.389 | 0.800 | 4 / 132 |
| **AND, then OR fallback** | **0.373** | **0.595** | **0.753** | **0.869** | **4 / 132** |
Pure OR is not the answer: weak single-term matches flood the ranking and MRR collapses. The winning pattern runs the strict query first, then tops up with OR-ranked results only when it comes back short — precise queries keep their precise results, and queries that died on one missing term get rescued:
```sql
WITH strict AS (
SELECT title, url, ts_rank_cd(fts, websearch_to_tsquery('kazakh_cfg', :q)) AS rank
FROM articles
WHERE fts @@ websearch_to_tsquery('kazakh_cfg', :q)
ORDER BY rank DESC
LIMIT 10
),
relaxed AS (
-- Re-join the stemmed query terms with OR; cover-density ranking
-- floats documents matching more terms to the top.
SELECT a.title, a.url, ts_rank_cd(a.fts, q.tsq) AS rank
FROM articles a,
(SELECT to_tsquery('simple', string_agg(lexeme, ' | ')) AS tsq
FROM unnest(to_tsvector('kazakh_cfg', :q))) q
WHERE a.fts @@ q.tsq
AND a.url NOT IN (SELECT url FROM strict)
ORDER BY rank DESC
LIMIT 10
)
SELECT * FROM strict
UNION ALL
SELECT * FROM relaxed
LIMIT 10;
```
(The `relaxed` CTE uses the `simple` config for `to_tsquery` because the lexemes coming out of `unnest(to_tsvector('kazakh_cfg', …))` are already stemmed.)
### Tuning weights
Penalty weights are tunable at runtime without restarting PostgreSQL:
```sql
ALTER TEXT SEARCH DICTIONARY pg_kazsearch_dict (w_deriv = 3.5, w_short_char = 100.0);
```
### Script mode controls
`pg_kazsearch_dict` defaults to `script_mode = auto` (Latin auto-detection + canonical Cyrillic output). For debugging or strict Cyrillic-only behavior:
```sql
ALTER TEXT SEARCH DICTIONARY pg_kazsearch_dict (script_mode = cyrillic_only);
```
CLI uses the same core default (`auto`) and exposes `--cyrillic-only` on `stem`, `analyze`, and `bench` commands. Elasticsearch exposes the same knob as the `script_mode` token filter setting (see the Elasticsearch configuration section above).
---
## Benchmarks
Tested on 2,999 Kazakh news articles from [kaz.tengrinews.kz](https://kaz.tengrinews.kz/). Queries fall into three groups that must be read differently:
- **gold_v2** (n=132): human-written queries across 39 themes with deliberate morphological variety, URL-keyed, relevance judged over a pooled top-15 union of three retrieval systems, with a blind 20% re-judgment (95.9% agreement, Cohen's κ 0.899) and adjudicated disagreements — the primary quality benchmark (`eval/gold_queries_v2.jsonl`, methodology in `eval/gold_queries_v2.meta.json`)
- **gold** (n=51): the older, smaller human-written set, kept for continuity
- **auto** (n=8,997): queries mined from the indexed articles themselves (title keywords, body sentences, artificially inflected variants) — useful for regression testing, but *circular*: they overstate absolute quality because each query is derived from the document it must find
All numbers below are reproduced by `just eval-search` and written to `eval/results/report.json` (charts are generated from that file, never hardcoded).
### Does stemming help? (vs identical FTS with no stemming)
Recall@10, same corpus, same ranking, only the dictionary differs:
| Query set | pg_kazsearch | `simple` (no stem) | Effect |
| -------------------------- | ------------ | ------------------ | ------ |
| gold_v2 (human, n=132) | **0.542** | 0.187 | ~2.9x recall (95% CI [0.49, 0.60]) |
| gold (human, n=51) | **0.225** | 0.102 | ~2.2x recall |
| morpho_variant (inflected) | **0.865** | 0.005 | stemming is essential for suffixed queries |
| title_keywords (verbatim) | 0.985 | 0.992 | no stemming needed for exact-word matches |
Human queries in Kazakh naturally contain inflected forms, which is exactly where the stemmer pays off. gold_v2 MRR@10 is 0.743 vs 0.420 without stemming. Stemming is idempotent (`stem(stem(w)) == stem(w)`, enforced by tests), so query-side and document-side inflections of one lexeme always meet at the same index term.
### PostgreSQL: pg_kazsearch vs pg_trgm
Head-to-head on the same 500-query sample (seeded, reproducible):
| Metric | pg_kazsearch | pg_trgm | Improvement |
| --------- | ------------ | ------- | ----------- |
| Recall@10 | **0.920** | 0.619 | +49% |
| MRR@10 | **0.846** | 0.539 | +57% |
| nDCG@10 | **0.861** | 0.555 | +55% |
Note: pg_trgm here matches against titles only (its typical usage); the sample is dominated by auto-queries, so treat this as a relative comparison, not an absolute quality claim.
### Token coverage
Measured over 45,708 corpus tokens with `python3 eval/measure_stem_coverage.py`:
| Rate | Value | Meaning |
| ---- | ----- | ------- |
| Analyzed | 76.5% | a suffix was stripped |
| Stem in lexicon | 74.9% | final stem is a dictionary lemma |
| Recognized | **87.5%** | stemmed or already a dictionary lemma |
### Elasticsearch: kazsearch_stem vs standard analyzer
On human-written queries, the stemmer finds more relevant articles and ranks them higher. Reproduced by `python3 eval/run_eval_es.py` (results in `eval/results/report_es.json`), stratified by query source like the PostgreSQL eval; auto-generated query sources are omitted here because they are mined from the indexed corpus itself:
| Query set | Metric | kazsearch_stem | standard | Improvement |
| ---------------------------------- | --------- | -------------- | -------- | ----------- |
| gold (human, n=51) | Recall@10 | **0.396** | 0.309 | +28% |
| | MRR@10 | **0.663** | 0.591 | +12% |
| gold_v2 (human, URL-keyed, n=132) | Recall@10 | **0.524** | 0.451 | +16% |
| | MRR@10 | **0.669** | 0.569 | +18% |
### vs Tengrinews.kz native search
Re-benchmarked July 2026 against `kaz.tengrinews.kz/search/?text=…`. Raw counts are not directly comparable — tengrinews searches its full archive while our index holds 2,999 articles — so the meaningful signal is *conflation behavior*, shown by the probe below the table. Matched-count columns use each engine's default semantics (tengrinews as-is, ES `multi_match` OR):
| Search query (Kazakh with suffixes) | tengrinews.kz (full archive) | ES + kazsearch_stem (2,999 docs) |
| ----------------------------------- | ---------------------------- | -------------------------------- |
| мектептердегі оқушылар | 55 | **182** |
| балалардың денсаулығы | 203 | **468** |
| мұғалімдердің наразылығы | 0 | **64** |
| спортшылардың жетістіктері | 14 | **227** |
| бензиннің бағасын көтеру | 8 | **471** |
| мектептеріміздегі мәселелер | 0 | **640** |
Tengrinews' search has improved since our first benchmark (several formerly-zero queries now return results), but the morphology probe shows it still matches surface forms, not lexemes — every inflection of "school" is a different search:
| Query form | tengrinews.kz (full archive) | ES + kazsearch_stem (2,999 docs) |
| ---------------- | ---------------------------- | -------------------------------- |
| мектеп | 906 | **156** |
| мектептердегі | 110 | **156** |
| мектептерімізде | 1 | **156** |
With stemming, all three forms hit the same 156 articles. Without it, a user who phrases the query with a possessive gets 1 result from an archive that contains 906 school articles.
### Stemmer examples
| Input | Output | Stripped |
| ---------------- | --------- | ------------------------------ |
| мектептерімізде | мектеп | plural + possessive + locative |
| президенттерінің | президент | plural + possessive + genitive |
| өзгеруі | өзгеру | verbal noun possessive |
| берді | бер | past tense |
| экономикалық | экономика | derivational adjective |
---
## Architecture
```
┌──────────────────────────────────────────────────────────┐
│ Cargo Workspace │
│ │
│ core/ Pure Rust stemmer (no PG/ES deps) │
│ pg_ext/ pgrx PostgreSQL extension │
│ cli/ CLI tool (kazsearch stem/analyze/bench) │
│ elastic/ Elasticsearch plugin (Rust cdylib + JNI) │
│ src/ C ABI + JNI exports calling core::stem() │
│ java/ Java bridge + Lucene TokenFilter (~50 LoC)│
│ docker/ ES with plugin pre-installed │
└──────────────────────────────────────────────────────────┘
```
One stemmer, multiple consumers. The `core/` crate is the single source of truth for all stemming logic — PostgreSQL, Elasticsearch, and CLI all call into it.
The stemmer algorithm:
- **BFS suffix stripper** — breadth-first search over layered morphological rules (predicate, case, possessive, plural, derivational for nouns; person, tense, negation, voice for verbs), with vowel harmony validation
- **Penalty scoring** — candidates scored by syllable count, suffix weakness, derivational depth, and lexicon hits
- **Lexicon** — 21,863 POS-tagged stems from [Apertium-kaz](https://github.com/apertium/apertium-kaz) for overstemming protection
- **Stem repair** — consonant mutation reversal (б→п, г→к, ғ→қ), vowel elision restoration, lexicon-based vowel append
---
## CLI
The `kazsearch` CLI works standalone without PostgreSQL:
```bash
cargo build -p kazsearch-cli --release
# Stem a word
kazsearch stem алмаларымыздағы
# алмаларымыздағы алма
# Morphological analysis
kazsearch analyze мектептеріміздегі
# Benchmark
kazsearch bench wordlist.txt
# Validate lexicon
kazsearch lexicon validate data/tsearch_data/kaz_stems.dict
```
---
## Development
### PostgreSQL
```bash
just up # Start PG container
just build # Build + install extension
just reload # DROP + CREATE extension
just test-core # Core Rust unit tests
just test-ext # Smoke test via SQL
just cli # Build CLI
```
### Elasticsearch
```bash
just es-native # Build Rust cdylib for ES plugin
just es-build # Build plugin ZIP (Gradle)
just es-up # Start ES container with plugin
just es-load-corpus # Index 3000 articles
just es-eval # Run search quality evaluation
just es-down # Stop ES container
```
---
## Contributing
1. Fork the repo and create a feature branch
2. Make your changes — stemmer logic lives in `core/src/`, extension glue in `pg_ext/src/lib.rs`
3. Run `cargo test -p kazsearch-core --test stem_tests` to verify stemmer correctness
4. Run `just up && just reload && just test-ext` to verify the extension works end-to-end
5. Open a PR
Key things to know:
- Penalty weights in `core/src/explore.rs` are empirically tuned via CMA-ES — changing one can affect many test cases
- Layer guards encode real morphotactic constraints, not heuristics
- Vowel harmony (back/front) is mandatory for suffix validation
---
## References
- Krippes, K.A. (1993). *Kazakh (Qazaq-) Grammatical Sketch with Affix List*. ERIC.
- Washington, J., Salimzyanov, I., Tyers, F. (2014). *Finite-state morphological transducers for three Kypchak languages*. LREC.
- Makhambetov, O. et al. (2015). *Data-driven morphological analysis and disambiguation for Kazakh*. CICLing.
---
## License
- **Code:** [LGPL-3.0](LICENSE)
- **Lexicon data** derived from [Apertium-kaz](https://github.com/apertium/apertium-kaz) (GPL-3.0).