Skip to main content

Semantic text clustering using sentence embeddings and agglomerative clustering.

Project description

semaclust

CI PyPI version Python versions License: MIT

semaclust (semantic + clustering) is a small Python library for clustering similar strings using sentence embeddings and agglomerative clustering. It is useful for deduplicating free-text fields, normalizing user-entered values, and collapsing spelling or formatting variants into a canonical form.

Documentation: https://cobanov.github.io/semaclust/

Installation

pip install semaclust
# or
uv add semaclust

Install from source:

pip install git+https://github.com/cobanov/semaclust.git

Quickstart

from semaclust import TextClusterer

texts = [
    "New York", "NYC", "new york city",
    "Los Angeles", "LA",
    "San Francisco", "San Fran", "SF",
]

clusterer = TextClusterer(distance_threshold=1.0)
clusterer.fit(texts)

print(clusterer.n_clusters_)
# 3

print(clusterer.clusters_)
# {0: ['New York', 'NYC', 'new york city'],
#  1: ['Los Angeles', 'LA'],
#  2: ['San Francisco', 'San Fran', 'SF']}

print(clusterer.transform())
# ['NYC', 'NYC', 'NYC', 'LA', 'LA', 'SF', 'SF', 'SF']

fit_transform is the one-call shortcut:

TextClusterer(distance_threshold=1.0).fit_transform(texts)

API at a glance

Method Returns Purpose
fit(texts) self Cluster and store fitted attributes
fit_predict(texts) ndarray[int] Cluster labels per input text
fit_transform(texts) list[str] Each text replaced with its representative
transform(texts=None) list[str] Replace texts seen at fit time
get_replacement_map() dict[str, str] Mapping from original to representative
result_ ClusterResult Frozen dataclass with labels, clusters, reps

Fitted attributes (sklearn convention): labels_, clusters_, representatives_, n_clusters_, texts_.

Choosing a model and threshold

The default encoder is all-MiniLM-L6-v2 (22M params, 384-dim, ~80 MB). It's fast, small, and a sensible drop-in, but it isn't the strongest option for every task. The table below summarizes 9 encoders on three workloads of escalating difficulty: short text with abbreviations (cities), medium text with synonyms (job titles), and long sentences with topical themes (customer feedback). See benchmarks.md for the full methodology, raw thresholds, and per-test breakdowns.

ARI (Adjusted Rand Index) of 1.0 means an exact match with the ground-truth clustering. Bold = perfect.

Model Params Dim cities job titles feedback Good threshold
all-MiniLM-L6-v2 (default) 23M 384 1.000 0.672 1.000 1.0 (cities) / 1.3 (feedback)
all-MiniLM-L12-v2 33M 384 1.000 0.512 1.000 1.20
all-mpnet-base-v2 109M 768 0.619 0.512 1.000 -
BAAI/bge-small-en-v1.5 33M 384 1.000 0.672 1.000 0.95 - 1.00
BAAI/bge-m3 568M 1024 1.000 0.520 1.000 -
nomic-ai/nomic-embed-text-v1.5 * 137M 768 1.000 0.520 1.000 0.70 - 0.85
nomic-ai/nomic-embed-text-v2-moe * 475M 768 1.000 0.672 1.000 1.15 - 1.35
mixedbread-ai/mxbai-embed-large-v1 335M 1024 1.000 0.815 1.000 0.95 - 1.05
Qwen/Qwen3-Embedding-0.6B 596M 1024 0.529 0.672 1.000 -

* Requires a clustering: prefix on each input. The bench applies this automatically; if you swap the model in directly, wrap it in a custom encoder that prepends the prefix.

Practical takeaways:

  • Bigger isn't better. all-mpnet-base-v2 and Qwen3-Embedding-0.6B both fail the simplest test (cities) - they merge Los Angeles with San Francisco before merging SF with San Francisco.
  • Best small drop-in: BAAI/bge-small-en-v1.5 matches the default's size class and has a single threshold window (0.95-1.00) that works for both cities and feedback.
  • Best overall: mixedbread-ai/mxbai-embed-large-v1 - the only model to exceed 0.8 ARI on job titles, with a 0.10-wide cities+feedback overlap at 0.95-1.05. Costs 15x more parameters than the default.
  • No model solves the job-titles case. Synonymy across roles (SWE ~ Programmer ~ Software Engineer, Product Manager ~ Product Owner) defeats every encoder we tested.

To swap the encoder, pass a string or a custom Encoder:

from semaclust import TextClusterer

clusterer = TextClusterer(
    encoder="BAAI/bge-small-en-v1.5",
    distance_threshold=1.0,
)

Plugging in your own encoder

Any object implementing encode(texts: list[str]) -> np.ndarray satisfies the Encoder protocol:

from semaclust import TextClusterer, Encoder
import numpy as np

class MyEncoder:
    def encode(self, texts: list[str]) -> np.ndarray:
        return np.random.rand(len(texts), 384).astype(np.float32)

clusterer = TextClusterer(encoder=MyEncoder())

CLI

semaclust ships a small typer-based CLI:

# Cluster lines from a file, write JSON
semaclust cluster items.txt --threshold 1.0 --output clusters.json

# Replace each line with its cluster representative
cat items.txt | semaclust replace --threshold 1.0

Run semaclust --help for the full reference.

Migration from 0.1.x

The 0.3 release is a breaking change. The single cluster(texts) entry point is gone; the new API mirrors scikit-learn:

0.1.x 0.3.x
clusterer.cluster(texts) clusterer.fit(texts).clusters_
clusterer.get_replacement_map(texts) clusterer.fit(texts).get_replacement_map()
clusterer.replace_values(texts) clusterer.fit_transform(texts)

The representative_selector argument moved from per-call to the constructor.

Development

git clone https://github.com/cobanov/semaclust.git
cd semaclust
uv sync --extra dev
uv run pytest
uv run ruff check src tests
uv run mypy src/semaclust

Install the pre-commit hooks once:

uv run pre-commit install

License

MIT, see LICENSE.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

semaclust-0.4.1.tar.gz (15.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

semaclust-0.4.1-py3-none-any.whl (12.0 kB view details)

Uploaded Python 3

File details

Details for the file semaclust-0.4.1.tar.gz.

File metadata

  • Download URL: semaclust-0.4.1.tar.gz
  • Upload date:
  • Size: 15.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for semaclust-0.4.1.tar.gz
Algorithm Hash digest
SHA256 852b772eab33a094dbcc607e469e32f51e426b3fbce92cbc739e8059adb93dea
MD5 0a80d78bb9b9f61b9f4465582c98797c
BLAKE2b-256 2948b85f43fd0fb51be48bc1fcda2f55ef4e0edfb44dc81f82f073d0030f0f27

See more details on using hashes here.

Provenance

The following attestation bundles were made for semaclust-0.4.1.tar.gz:

Publisher: release.yml on cobanov/semaclust

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file semaclust-0.4.1-py3-none-any.whl.

File metadata

  • Download URL: semaclust-0.4.1-py3-none-any.whl
  • Upload date:
  • Size: 12.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for semaclust-0.4.1-py3-none-any.whl
Algorithm Hash digest
SHA256 1703fe88f0d4e4422c1faaadb6ac62f87ddadfb8baad850f2200f76627e3d6be
MD5 f39fa99cf9cdab1f92002258581ecd86
BLAKE2b-256 cb850b5c8a4deb6ded76099fbea41d8e448a530b6119094638b912175e68a3a6

See more details on using hashes here.

Provenance

The following attestation bundles were made for semaclust-0.4.1-py3-none-any.whl:

Publisher: release.yml on cobanov/semaclust

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page