Skip to main content

Modern NLP for Indonesian and regional languages

Project description

BASA

Modern NLP preprocessing for Indonesian and regional languages.

BASA is a lightweight, zero-dependency preprocessing library designed for real-world Indonesian text — the kind found on Twitter/X, TikTok, WhatsApp, Shopee reviews, and Discord. It normalizes informal slang, collapses expressive character repetition, reduces punctuation noise, and optionally corrects typos, all through a single clean API.

from basa import normalize

normalize("GW GKKKK NGERTIII BNGTTTT!!!!!")
# → 'saya tidak mengerti banget!'

Why BASA?

Indonesian social media text is notoriously difficult to process with standard NLP tools:

Raw input After normalize()
gw gk ngerti bngt sihhhh!!! saya tidak mengerti banget sih!
kmrn gamau makan krn baper bgt kemarin tidak mau makan karena bawa perasaan banget
otw gan, rekber dlu ya!!!!! dalam perjalanan saudara, rekening bersama dulu ya!
GW GKKKK NGERTIII BNGTTTT!!!!! saya tidak mengerti banget!

Standard tokenizers and language models often fail on this kind of input because they see "gkkkk", "bngtttt", and "ngertiii" as unknown tokens. BASA normalizes them first.


Installation

pip install basa

Requires Python 3.10+


Quick Start

One-liner (recommended for most use cases)

from basa import normalize

normalize("gw gk ngerti bngt sihhhh!!!")
# → 'saya tidak mengerti banget sih!'

Zero-config alias

from basa import quick

quick("GW GKKKK NGERTIII BNGTTTT!!!!!")
# → 'saya tidak mengerti banget!'

quick() is a thin alias for normalize() with all defaults applied. Use it when you want the shortest possible call.

Batch processing

from basa import normalize

texts = [
    "gw gk ngerti",
    "lu udh makan??",
    "kmrn gamau pergi krn baper bgt",
]

normalize(texts)
# → ['saya tidak mengerti', 'kamu sudah makan?', 'kemarin tidak mau pergi karena bawa perasaan banget']

API Reference

normalize(text, **options)

Normalize informal Indonesian text. Accepts a single string or a list of strings.

normalize(
    text: Union[str, List[str]],
    apply_slang: bool = True,
    apply_typo: bool = False,
    lowercase: bool = True,
    normalize_punctuation: bool = True,
    normalize_whitespace: bool = True,
) -> Union[str, List[str]]

Parameters

Parameter Type Default Description
text str or List[str] Input text or list of texts.
apply_slang bool True Expand slang and reduce expressive repeated characters (e.g. "bngtttt""banget").
apply_typo bool False Correct misspelled words using Levenshtein distance. Opt-in — requires a vocabulary to be loaded first.
lowercase bool True Lowercase the text before processing. Set False for NER and case-sensitive pipelines.
normalize_punctuation bool True Collapse repeated punctuation marks ("!!!!!""!").
normalize_whitespace bool True Strip leading/trailing whitespace and collapse internal multiple spaces.

Processing pipeline (in order)

1. lowercase            → "GW GK NGERTI" → "gw gk ngerti"
2. slang normalization  → "gkkkk" → "gk" → "tidak"
3. typo correction      → "mkan" → "makan"  (opt-in)
4. punctuation          → "!!!!!" → "!"
5. whitespace cleanup   → "  a   b  " → "a b"

Examples

# Preserve case for NER tasks
normalize("Jokowi pergi ke Jakarta", lowercase=False)
# → 'Jokowi pergi ke Jakarta'

# Disable slang (pass through raw tokens)
normalize("gw gk ngerti", apply_slang=False)
# → 'gw gk ngerti'

# Enable typo correction (requires vocab)
from basa import typo
typo.add_to_vocab({"makan", "minum", "pergi"})
normalize("saya mkan dan mnum", apply_typo=True)
# → 'saya makan dan minum'

quick(text)

Zero-config alias for normalize() with all default settings.

from basa import quick

quick("gw gamau pergi krn mager")
# → 'saya tidak mau pergi karena malas bergerak'

typo — Typo Corrector

BASA's typo corrector is vocabulary-driven and opt-in by default. You supply the vocabulary; BASA finds the closest match using Levenshtein distance.

from basa import typo

# Load your domain vocabulary
typo.add_to_vocab({"makan", "minum", "masak", "pergi", "datang"})

typo.correct("mkan")     # → 'makan'
typo.correct("mnm")      # → 'minum'
typo.correct("ok")       # → 'ok'  (too short, skipped by default)

# Correct a full sentence
typo.correct_text("saya mkan dan mnm")
# → 'saya makan dan minum'

# Get multiple suggestions
typo.suggest("mkan", top_k=3)
# → ['makan', 'masak', 'minum']

Why is apply_typo=False by default?

Typo correction is destructive when applied blindly. Without the right vocabulary, domain-specific terms like xgboost, lightgbm, or rekber would be mangled. BASA follows the principle of conservative by default, destructive features opt-in.

Vocabulary management

from basa import typo

typo.add_to_vocab({"kata", "lain"})      # add words
typo.remove_from_vocab({"kata"})         # remove words
typo.clear_vocab()                       # reset entirely
len(typo)                                # vocab size
"makan" in typo                          # membership check

# Check cache statistics (useful for profiling)
typo.cache_info()
# → {'hits': 120, 'misses': 35, 'size': 35}

Typo corrector options

from basa.core.typo import TypoCorrector

corrector = TypoCorrector(
    vocab={"makan", "minum"},
    min_word_length=4,    # tokens shorter than this are skipped (default: 4)
    min_confidence=0.5,   # minimum correction confidence in [0, 1] (default: 0.5)
)

slang — Slang Normalizer

Access the underlying slang engine directly for fine-grained control.

from basa.core.slang import slang, SlangNormalizer

# Use the singleton
slang.normalize("gw gamau pergi krn lg baper bgt")
# → 'saya tidak mau pergi karena sedang bawa perasaan banget'

# Custom dictionary (extend or override defaults)
custom = SlangNormalizer(custom_mapping={
    "gaskeun": "ayo lakukan",
    "jancok":  "ekspresi",
})
custom.normalize("gaskeun bro!")
# → 'ayo lakukan bro!'

# Batch normalize
slang.normalize_batch(["gw makan", "lu minum"])
# → ['saya makan', 'kamu minum']

Slang dictionary categories

The built-in dictionary covers 250+ entries across 13 categories:

Category Examples
Pronouns gw → saya, lu → kamu, dy → dia
Kinship & address kk → kakak, klg → keluarga, ortu → orang tua
Negation ga, gak, nggak → tidak
Compound negation gamau → tidak mau, gabisa → tidak bisa
Conjunctions yg → yang, krn → karena, tp → tapi
Verbs udah → sudah, blm → belum, ngerti → mengerti
Adjectives & adverbs bgt → banget, bener → benar, dikit → sedikit
Question words gmn → bagaimana, knp → kenapa, kmn → kemana
Greetings & responses makasih → terima kasih, sip → baik
Temporal & location skrg → sekarang, kmrn → kemarin, ntr → nanti
Internet slang otw → dalam perjalanan, btw → omong-omong, wkwk → tertawa
E-commerce & finance ongkir → ongkos kirim, rekber → rekening bersama, cod → bayar di tempat
Youth / Gen-Z mager → malas bergerak, baper → bawa perasaan, gabut → tidak ada kegiatan

Real-World Use Cases

Preprocessing for sentiment analysis

from basa import normalize

reviews = [
    "produknya bagus bgt tp ongkirnya mahal bgt!!!",
    "gw kecewa bngt, barang ga sesuai deskripsi smskali",
    "rekber dlu gan, takut kena tipu",
]

clean = normalize(reviews)
# Pass clean into your sentiment model

Preprocessing for a custom NLP pipeline

from basa import normalize, typo

# Load your domain vocabulary (e.g., from a word list file)
with open("vocab.txt") as f:
    domain_vocab = set(f.read().splitlines())

typo.add_to_vocab(domain_vocab)

def preprocess(text: str) -> str:
    return normalize(text, apply_typo=True)

preprocess("gw mkan siang tdi di wrng padang")
# → 'saya makan siang tadi di warung padang'

NER pipeline (preserve casing)

from basa import normalize

text = "Jokowi blg bhw pemerintah akan bantu UMKM"
normalize(text, lowercase=False)
# → 'Jokowi bilang bahwa pemerintah akan bantu UMKM'

Design Philosophy

BASA is built around three principles:

  1. Conservative by default. Only safe, lossless transforms are enabled out of the box. Destructive features (like typo correction) require explicit opt-in.

  2. No bundled vocabularies for correction. Every domain has different vocabulary needs — fintech, e-commerce, ML, healthcare. Callers supply their own word list via typo.add_to_vocab().

  3. Zero required dependencies for core preprocessing. The normalize() and slang modules use only the Python standard library. The optional transformers, torch, and pydantic dependencies are only required for advanced modules (basa.translate, basa.evaluate).


Development

Setup

git clone https://github.com/Muanai/basa.git
cd basa
python -m venv .venv
.venv\Scripts\activate       # Windows
# source .venv/bin/activate  # macOS / Linux
pip install -e ".[dev]"

Running tests

pytest tests/ -v

Optional extras

pip install -e ".[serving]"     # FastAPI serving
pip install -e ".[evaluation]"  # ROUGE, BERTScore, seqeval
pip install -e ".[dev]"         # pytest, ruff, black, mypy

Roadmap

Version Status Features
v0.1 ✅ Current normalize(), quick(), slang (250+ entries), typo corrector
v0.2 🔜 Planned BK-Tree / SymSpell for faster typo correction at large vocab sizes
v0.3 🔜 Planned Emoji handling, remove_emoji flag
v0.4 🔜 Planned Tokenizer module (basa.tokenize)
v1.0 🔜 Planned Stable API, full docs site, PyPI release

Contributing

Contributions are welcome! In particular:

  • Slang dictionary additions — if you spot a common slang word that's missing, open a PR adding it to the appropriate category in src/basa/core/slang.py.
  • Bug reports — please include the exact input string and the unexpected output.
  • Performance improvements — especially for the typo correction module.

Please open an issue before submitting large changes.


License

MIT © 2026 Muanai Khalifah Revindo

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

basa-0.1.0a0.tar.gz (24.4 kB view details)

Uploaded Source

Built Distribution

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

basa-0.1.0a0-py3-none-any.whl (25.7 kB view details)

Uploaded Python 3

File details

Details for the file basa-0.1.0a0.tar.gz.

File metadata

  • Download URL: basa-0.1.0a0.tar.gz
  • Upload date:
  • Size: 24.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.6

File hashes

Hashes for basa-0.1.0a0.tar.gz
Algorithm Hash digest
SHA256 1e4a72b229e21b9c8d7a8e147de3c4fc3bcb3f90e8be00bf723e7dc0b8136d99
MD5 490c6e1b07fbd94895a1e1503082db9c
BLAKE2b-256 b14ee356f94119f44d918b562f2892a89abb02f3814fce78b2b0f45b4055c744

See more details on using hashes here.

File details

Details for the file basa-0.1.0a0-py3-none-any.whl.

File metadata

  • Download URL: basa-0.1.0a0-py3-none-any.whl
  • Upload date:
  • Size: 25.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.6

File hashes

Hashes for basa-0.1.0a0-py3-none-any.whl
Algorithm Hash digest
SHA256 968095053fa870d8b0f73ec63957f74a73368b882967a33863450992f5f87b46
MD5 03c683d0495ce976028e97c60265c2a8
BLAKE2b-256 125caead7e30ccf9123ae869d227cc5a9fc2e2c3f8742d66eb11071b87be610d

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