Rule-based grapheme-to-phoneme phonemizer for the Mirandese language (mwl)
Project description
Mirandese Phonemizer
This repository contains a Python-based Mirandese phonemizer, designed to convert Mirandese text into its International Phonetic Alphabet (IPA) representation. It supports different Mirandese dialects and incorporates various phonological rules based on linguistic information from Wikipedia.
Features
- Grapheme-to-Phoneme Conversion: Converts Mirandese graphemes (letters and digraphs) to their corresponding IPA phonemes.
- Contextual Rules: Applies phonological rules based on surrounding characters (e.g., lenition of voiced stops, sibilant variations, vowel glides).
- Dialectal Support:
- Handles the specific lh to l pronunciation change in the Sendinese dialect.
- Includes word-level lookup dictionaries for Central, Raiano, and Sendinese dialects to handle irregular pronunciations.
- Latin Cluster Evolution: Converts Latin initial consonant clusters (pl, kl, fl) to /tʃ/.
- Proto-Romance Medial Clusters: Converts Proto-Romance medial clusters (-ly-, -cl-) to /ʎ/.
- Double Consonant Palatalization: Handles palatalization of double ll to /ʎ/ and nn to /ɲ/.
- Proto-Romance -mn-: Converts -mn- to /m/.
- Output Customization: Options to keep or remove optional phonemes (in parentheses) and stress marks/syllable dots.
Usage
from mwl_phonemizer import CRFOrthoCorrector
sample_texts = [
"Muitas lhénguas ténen proua de ls sous pergaminos antigos, de la lhiteratura screbida hai cientos d'anhos i de scritores hai muito afamados, hoije bandeiras dessas lhénguas. Mas outras hai que nun puoden tener proua de nada desso, cumo ye l causo de la lhéngua mirandesa.",
"Todos ls seres houmanos nácen lhibres i eiguales an honra i an dreitos. Dotados de rezon i de cuncéncia, dében de se dar bien uns culs outros i cumo armano",
]
phonemizer = CRFOrthoCorrector()
for text in sample_texts:
print(f"Original: {text}")
print(f"Phonemized: {phonemizer.phonemize_sentence(text)}\n")
Helper Functions
The base class provides static methods for cleaning up IPA output:
strip_markers(ipa: str): Removes syllable dots (.) and optional phoneme parentheses (()).strip_stress(ipa: str): Removes primary (ˈ) and secondary (ˌ) stress markers.
ipa_with_markers = "ˈe(j).ʒɛmˈplu"
clean_ipa = phonemizer.strip_markers(ipa_with_markers)
print(f"Clean IPA: {clean_ipa}")
# Output: Clean IPA: ˈejʒɛmˈplu
ipa_with_stress = "miɾɐ̃ˈdes̺"
stress_agnostic_ipa = phonemizer.strip_stress(ipa_with_stress)
print(f"Stress-Agnostic IPA: {stress_agnostic_ipa}")
# Output: Stress-Agnostic IPA: miɾɐ̃des̺
Phonemizer Comparison
| Phonemizer | PER (Full IPA, Stress) | PER (Stress-Agnostic) | Words Incorrect (ED>0) | Notes |
|---|---|---|---|---|
| Character lookup | 45.47% | 38.66% | 174 | Simple letter/digraph to phoneme lookup table |
| N-gram (n=4) | 44.13% | 30.98% | 173 | Statistical N-gram model for G2P conversion |
| Orthography Rules | 35.86% | 27.91% | 166 | Hand-crafted orthographic rules |
| Orthography Rules + CRF | 14.97% | 2.53% | 161 | Hand-crafted orthographic rules output corrected with a CRF model |
| CRF | 20.25% | 8.58% | 164 | Character-level CRF trained on aligned word–phoneme pairs |
| Espeak + CRF | 61.39% → 11.82% | 40.92% → 7.41% | 103 | Espeak output corrected with a CRF model |
| Epitran + CRF | 51.14% → 20.25% | 44.63% → 8.58% | 164 | Epitran output corrected with a CRF model |
| Epitran + Rules | 51.14% → 47.68% | 44.63% → 40.56% | 169 | Epitran output corrected with hand-crafted rules |
| Espeak + Rules | 61.39% → 53.35% | 40.92% → 32.07% | 174 | Espeak output corrected with hand-crafted rules |
Notes:
- For Epitran and Espeak, the first value is the initial phonemization output; the second is after applying correction rules.
- PER (Phoneme Error Rate) measures the proportion of phonemes differing from the gold standard IPA transcription.
- Stress-Agnostic PER ignores stress marks.
- lower PER does not necessarily mean a better phonemizer
- CRF models are overfitted due to small data size
Future Work
- Stress Prediction: Implement machine learning or rule-based stress placement.
- Larger Dataset: Expand training and evaluation beyond ~150 words.
- Neural Approaches: Explore Transformer or seq2seq models for improved G2P accuracy.
- Dialectal Coverage: Add more irregular forms and dialect-specific exceptions.
Contributing
Feel free to open issues or submit pull requests for improvements, bug fixes, or new dialect support.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mwl_phonemizer-0.1.0a1.tar.gz.
File metadata
- Download URL: mwl_phonemizer-0.1.0a1.tar.gz
- Upload date:
- Size: 46.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
428c6399581649db0563fc0e722eeba991c84be22811fc598901f82ed3834a93
|
|
| MD5 |
859d1ea1f970ef502d4271e020508fba
|
|
| BLAKE2b-256 |
cf634864a6945b88c7ccea8682e169deac890c1cd76c8889270ad917e180192b
|
File details
Details for the file mwl_phonemizer-0.1.0a1-py3-none-any.whl.
File metadata
- Download URL: mwl_phonemizer-0.1.0a1-py3-none-any.whl
- Upload date:
- Size: 53.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
58ca86f46d0df4afb4eb5af6d125c54f54a09b738e94ea4c0908fc7c35f7883a
|
|
| MD5 |
4bddaf1e8571e82ecde81c4448af1716
|
|
| BLAKE2b-256 |
8e59caa9e70032b44e1d6faf001f7f8616213ace7b89d9228331fd8050f8b293
|