Skip to main content

Split text into semantically coherent, LLM-categorized chunks

Project description

chunklabel

A Python library for splitting text into categorized chunks using an LLM.

Overview

chunklabel segments text into semantically coherent spans, assigning a free-form category to each. Categories are named by the LLM without a predefined schema. Each chunk's quote is a verbatim excerpt from the source text, aligned back to the original after LLM output.

from chunklabel import ChunkLabeler

labeler = ChunkLabeler()
chunks = labeler.split(
    "The project kicked off in January with a small team. "
    "Budget constraints forced a scope reduction in March. "
    "Despite the setbacks, the product launched successfully in June."
)

# [
#   Chunk(category="initiation", quote="The project kicked off in January with a small team", start=0,   end=51),
#   Chunk(category="obstacle",   quote="Budget constraints forced a scope reduction in March", start=53,  end=104),
#   Chunk(category="outcome",    quote="the product launched successfully in June", start=120, end=160),
# ]

Installation

pip install chunklabel

Data structures

The LLM returns raw chunks without span information. Alignment is performed as a separate step, producing the final Chunk with character-level positions.

# Intermediate: LLM output
@dataclass
class RawChunk:
    category: str   # Free-form category name assigned by the LLM
    quote: str      # Verbatim excerpt (may contain minor transcription noise)

# Final: after alignment
@dataclass
class Chunk:
    category: str   # Same as RawChunk
    quote: str      # Excerpt aligned to source text
    start: int      # Start index in source text
    end: int        # End index in source text

Pipeline

Input text
     │
     ▼
LLM  →  [{category, quote}, ...]   (RawChunk list)
     │
     ▼
rapidfuzz alignment  →  (start, end) resolved per chunk
     │
     ▼
Span post-processing  (lenient mode)
     │  gap-filling / overlap resolution
     ▼
Chunk list

Lenient mode

  • Gaps: unassigned spans between chunks are filled automatically as category="uncategorized"
  • Overlaps: the earlier chunk takes priority; the later chunk's start is pushed forward

Category normalization (offline)

After processing multiple texts, category names can drift across runs. A dedicated normalization step lets the LLM consolidate them in batch.

from chunklabel import Normalizer

normalizer = Normalizer()
normalizer.build_mapping(all_chunks)
# {"kick-off": "initiation", "project start": "initiation", "blocker": "obstacle", ...}

normalized_chunks = normalizer.apply(all_chunks)

The mapping is stored internally after build_mapping, so it can be passed to apply implicitly. To reuse the mapping across runs without calling the LLM again:

# Save after building
normalizer.save("mapping.json")

# Restore later
normalizer = Normalizer.load("mapping.json")
normalized_chunks = normalizer.apply(all_chunks)

Normalization runs offline over the full category inventory, so the LLM can make globally consistent decisions rather than local ones.

Configuration

labeler = ChunkLabeler(
    model="gpt-4o",          # LLM model to use
    fuzzy_threshold=80,      # Match threshold for rapidfuzz alignment (0–100)
)

Using local LLMs

chunklabel uses LangChain's BaseChatModel interface internally, so any compatible model can be passed via the llm parameter.

Ollama

from langchain_ollama import ChatOllama
from chunklabel import ChunkLabeler

labeler = ChunkLabeler(llm=ChatOllama(model="llama3"))

llama.cpp (OpenAI-compatible server)

from langchain_openai import ChatOpenAI
from chunklabel import ChunkLabeler

labeler = ChunkLabeler(llm=ChatOpenAI(
    model="llama3",
    base_url="http://localhost:8080/v1",
    api_key="not-used",
))

Note: local models must support structured output (JSON mode). If with_structured_output is not reliable, wrap the model with a JSON-enforcing layer before passing it in.

Downstream use cases

The Chunk list produced by chunklabel is designed as input for further analysis:

  • NLI: score the relationship between hypotheses and chunk categories
  • NER: analyze co-occurrence between entity labels and categories
  • Relation extraction: map entity-pair relations to chunk categories
  • Conditional generation: use category as a conditioning signal for language models

License

MIT

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

chunklabel-0.1.6.tar.gz (5.4 kB view details)

Uploaded Source

Built Distribution

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

chunklabel-0.1.6-py3-none-any.whl (8.2 kB view details)

Uploaded Python 3

File details

Details for the file chunklabel-0.1.6.tar.gz.

File metadata

  • Download URL: chunklabel-0.1.6.tar.gz
  • Upload date:
  • Size: 5.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.4

File hashes

Hashes for chunklabel-0.1.6.tar.gz
Algorithm Hash digest
SHA256 6a8ee1c61abdc257576686521504594471abf6bbb6bcb499bf2224e34b9ac701
MD5 ec23590482a68cc98a0958c3f8e85ecb
BLAKE2b-256 10b30c0e9fcf0227032ad28d49b2b90da7afc1b6a1aedb13f7f8b6205c34da5f

See more details on using hashes here.

File details

Details for the file chunklabel-0.1.6-py3-none-any.whl.

File metadata

  • Download URL: chunklabel-0.1.6-py3-none-any.whl
  • Upload date:
  • Size: 8.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.4

File hashes

Hashes for chunklabel-0.1.6-py3-none-any.whl
Algorithm Hash digest
SHA256 08e98a3c19af647527daaf04b22c0ce2073ba88d8d1cfd6748838729015779e1
MD5 b9cfadeb56ccc10eefe1b4742b640e43
BLAKE2b-256 58b9a5c1ebcbecc4e18c37bcc7c51f37cfa7ef18c510b55d457c0dab96649b15

See more details on using hashes here.

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