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.8.tar.gz (6.1 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.8-py3-none-any.whl (9.0 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: chunklabel-0.1.8.tar.gz
  • Upload date:
  • Size: 6.1 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.8.tar.gz
Algorithm Hash digest
SHA256 bcab134734bf9095536d682f2081ce4db298c0391c73448ceef24f10cefab1d8
MD5 15e0de92fa8a1bc629f9081ed4f39aad
BLAKE2b-256 21eb855d8ed9a283564bdcd81b27ee61d0b820a1542363ac3c71f8028afc782d

See more details on using hashes here.

File details

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

File metadata

  • Download URL: chunklabel-0.1.8-py3-none-any.whl
  • Upload date:
  • Size: 9.0 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.8-py3-none-any.whl
Algorithm Hash digest
SHA256 fe56e5301c1ea4f0af8a73af670e7a68043a2366030ac1b4eb3f6ec4b29e0067
MD5 1a1b7cf3703cef3981768841605e9005
BLAKE2b-256 71860cd83a7de59068f42b9e2177bb9a4802e54211ffb263eb6e5e5cd5381968

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