| | import re |
| |
|
| | class VietnameseTTSNormalizer: |
| | """ |
| | A text normalizer for Vietnamese Text-to-Speech systems. |
| | Converts numbers, dates, units, and special characters into readable Vietnamese text. |
| | """ |
| | |
| | def __init__(self): |
| | self.units = { |
| | 'km': 'ki lô mét', 'dm': 'đê xi mét', 'cm': 'xen ti mét', |
| | 'mm': 'mi li mét', 'nm': 'na nô mét', 'µm': 'mic rô mét', |
| | 'μm': 'mic rô mét', 'm': 'mét', |
| | |
| | 'kg': 'ki lô gam', 'g': 'gam', 'mg': 'mi li gam', |
| | |
| | 'km²': 'ki lô mét vuông', 'km2': 'ki lô mét vuông', |
| | 'm²': 'mét vuông', 'm2': 'mét vuông', |
| | 'cm²': 'xen ti mét vuông', 'cm2': 'xen ti mét vuông', |
| | 'mm²': 'mi li mét vuông', 'mm2': 'mi li mét vuông', |
| | 'ha': 'héc ta', |
| | |
| | 'km³': 'ki lô mét khối', 'km3': 'ki lô mét khối', |
| | 'm³': 'mét khối', 'm3': 'mét khối', |
| | 'cm³': 'xen ti mét khối', 'cm3': 'xen ti mét khối', |
| | 'mm³': 'mi li mét khối', 'mm3': 'mi li mét khối', |
| | 'l': 'lít', 'dl': 'đê xi lít', 'ml': 'mi li lít', 'hl': 'héc tô lít', |
| | |
| | 'v': 'vôn', 'kv': 'ki lô vôn', 'mv': 'mi li vôn', |
| | 'a': 'am pe', 'ma': 'mi li am pe', 'ka': 'ki lô am pe', |
| | 'w': 'oát', 'kw': 'ki lô oát', 'mw': 'mê ga oát', 'gw': 'gi ga oát', |
| | 'kwh': 'ki lô oát giờ', 'mwh': 'mê ga oát giờ', 'wh': 'oát giờ', |
| | 'ω': 'ôm', 'ohm': 'ôm', 'kω': 'ki lô ôm', 'mω': 'mê ga ôm', |
| | |
| | 'hz': 'héc', 'khz': 'ki lô héc', 'mhz': 'mê ga héc', 'ghz': 'gi ga héc', |
| | |
| | 'pa': 'pát cal', 'kpa': 'ki lô pát cal', 'mpa': 'mê ga pát cal', |
| | 'bar': 'ba', 'mbar': 'mi li ba', 'atm': 'át mốt phia', 'psi': 'pi ét xai', |
| | |
| | 'j': 'giun', 'kj': 'ki lô giun', |
| | 'cal': 'ca lo', 'kcal': 'ki lô ca lo', |
| | } |
| | |
| | self.digits = ['không', 'một', 'hai', 'ba', 'bốn', |
| | 'năm', 'sáu', 'bảy', 'tám', 'chín'] |
| | |
| | def normalize(self, text): |
| | """Main normalization pipeline.""" |
| | text = text.lower() |
| | text = self._normalize_temperature(text) |
| | text = self._normalize_currency(text) |
| | text = self._normalize_percentage(text) |
| | text = self._normalize_units(text) |
| | text = self._normalize_time(text) |
| | text = self._normalize_date(text) |
| | text = self._normalize_phone(text) |
| | text = self._normalize_numbers(text) |
| | text = self._number_to_words(text) |
| | text = self._normalize_special_chars(text) |
| | text = self._normalize_whitespace(text) |
| | return text |
| | |
| | def _normalize_temperature(self, text): |
| | """Convert temperature notation to words.""" |
| | text = re.sub(r'-(\d+(?:[.,]\d+)?)\s*°\s*c\b', r'âm \1 độ xê', text, flags=re.IGNORECASE) |
| | text = re.sub(r'-(\d+(?:[.,]\d+)?)\s*°\s*f\b', r'âm \1 độ ép', text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*°\s*c\b', r'\1 độ xê', text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*°\s*f\b', r'\1 độ ép', text, flags=re.IGNORECASE) |
| | text = re.sub(r'°', ' độ ', text) |
| | return text |
| | |
| | def _normalize_currency(self, text): |
| | """Convert currency notation to words.""" |
| | def decimal_currency(match): |
| | whole = match.group(1) |
| | decimal = match.group(2) |
| | unit = match.group(3) |
| | decimal_words = ' '.join([self.digits[int(d)] for d in decimal]) |
| | unit_map = {'k': 'nghìn', 'm': 'triệu', 'b': 'tỷ'} |
| | unit_word = unit_map.get(unit.lower(), unit) |
| | return f"{whole} phẩy {decimal_words} {unit_word}" |
| | |
| | text = re.sub(r'(\d+)[.,](\d+)\s*([kmb])\b', decimal_currency, text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+)\s*k\b', r'\1 nghìn', text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+)\s*m\b', r'\1 triệu', text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+)\s*b\b', r'\1 tỷ', text, flags=re.IGNORECASE) |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*đ\b', r'\1 đồng', text) |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*vnd\b', r'\1 đồng', text, flags=re.IGNORECASE) |
| | text = re.sub(r'\$\s*(\d+(?:[.,]\d+)?)', r'\1 đô la', text) |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*\$', r'\1 đô la', text) |
| | return text |
| | |
| | def _normalize_percentage(self, text): |
| | """Convert percentage to words.""" |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*%', r'\1 phần trăm', text) |
| | return text |
| | |
| | def _normalize_units(self, text): |
| | """Convert measurement units to words.""" |
| | def expand_compound_with_number(match): |
| | number = match.group(1) |
| | unit1 = match.group(2).lower() |
| | unit2 = match.group(3).lower() |
| | full_unit1 = self.units.get(unit1, unit1) |
| | full_unit2 = self.units.get(unit2, unit2) |
| | return f"{number} {full_unit1} trên {full_unit2}" |
| | |
| | def expand_compound_without_number(match): |
| | unit1 = match.group(1).lower() |
| | unit2 = match.group(2).lower() |
| | full_unit1 = self.units.get(unit1, unit1) |
| | full_unit2 = self.units.get(unit2, unit2) |
| | return f"{full_unit1} trên {full_unit2}" |
| | |
| | text = re.sub(r'(\d+(?:[.,]\d+)?)\s*([a-zA-Zμµ²³°]+)/([a-zA-Zμµ²³°0-9]+)\b', |
| | expand_compound_with_number, text) |
| | text = re.sub(r'\b([a-zA-Zμµ²³°]+)/([a-zA-Zμµ²³°0-9]+)\b', |
| | expand_compound_without_number, text) |
| | |
| | sorted_units = sorted(self.units.items(), key=lambda x: len(x[0]), reverse=True) |
| | for unit, full_name in sorted_units: |
| | pattern = r'(\d+(?:[.,]\d+)?)\s*' + re.escape(unit) + r'\b' |
| | text = re.sub(pattern, rf'\1 {full_name}', text, flags=re.IGNORECASE) |
| | |
| | for unit, full_name in sorted_units: |
| | if any(c in unit for c in '²³°'): |
| | pattern = r'\b' + re.escape(unit) + r'\b' |
| | text = re.sub(pattern, full_name, text, flags=re.IGNORECASE) |
| | |
| | return text |
| | |
| | def _normalize_time(self, text): |
| | """Convert time notation to words with validation.""" |
| | |
| | def validate_and_convert_time(match): |
| | """Validate time components before converting.""" |
| | groups = match.groups() |
| | |
| | |
| | if len(groups) == 3: |
| | hour, minute, second = groups |
| | hour_int, minute_int, second_int = int(hour), int(minute), int(second) |
| | |
| | |
| | if not (0 <= hour_int <= 23): |
| | return match.group(0) |
| | if not (0 <= minute_int <= 59): |
| | return match.group(0) |
| | if not (0 <= second_int <= 59): |
| | return match.group(0) |
| | |
| | return f"{hour} giờ {minute} phút {second} giây" |
| | |
| | |
| | elif len(groups) == 2: |
| | hour, minute = groups |
| | hour_int, minute_int = int(hour), int(minute) |
| | |
| | |
| | if not (0 <= hour_int <= 23): |
| | return match.group(0) |
| | if not (0 <= minute_int <= 59): |
| | return match.group(0) |
| | |
| | return f"{hour} giờ {minute} phút" |
| | |
| | |
| | else: |
| | hour = groups[0] |
| | hour_int = int(hour) |
| | |
| | if not (0 <= hour_int <= 23): |
| | return match.group(0) |
| | |
| | return f"{hour} giờ" |
| | |
| | |
| | text = re.sub(r'(\d{1,2}):(\d{2}):(\d{2})', validate_and_convert_time, text) |
| | text = re.sub(r'(\d{1,2}):(\d{2})', validate_and_convert_time, text) |
| | text = re.sub(r'(\d{1,2})h(\d{2})', validate_and_convert_time, text) |
| | text = re.sub(r'(\d{1,2})h\b', validate_and_convert_time, text) |
| | |
| | return text |
| | |
| | def _normalize_date(self, text): |
| | """Convert date notation to words with validation.""" |
| | |
| | def is_valid_date(day, month, year): |
| | """Check if date components are valid.""" |
| | day, month, year = int(day), int(month), int(year) |
| | |
| | |
| | if not (1 <= day <= 31): |
| | return False |
| | if not (1 <= month <= 12): |
| | return False |
| |
|
| | return True |
| | |
| | def date_to_text(match): |
| | day, month, year = match.groups() |
| | if is_valid_date(day, month, year): |
| | return f"ngày {day} tháng {month} năm {year}" |
| | return match.group(0) |
| | |
| | def date_iso_to_text(match): |
| | year, month, day = match.groups() |
| | if is_valid_date(day, month, year): |
| | return f"ngày {day} tháng {month} năm {year}" |
| | return match.group(0) |
| | |
| | def date_short_year(match): |
| | day, month, year = match.groups() |
| | full_year = f"20{year}" if int(year) < 50 else f"19{year}" |
| | if is_valid_date(day, month, full_year): |
| | return f"ngày {day} tháng {month} năm {full_year}" |
| | return match.group(0) |
| | |
| | |
| | text = re.sub(r'\bngày\s+(\d{1,2})[/\-](\d{1,2})[/\-](\d{4})\b', |
| | lambda m: date_to_text(m).replace('ngày ngày', 'ngày'), text) |
| | text = re.sub(r'\bngày\s+(\d{1,2})[/\-](\d{1,2})[/\-](\d{2})\b', |
| | lambda m: date_short_year(m).replace('ngày ngày', 'ngày'), text) |
| | text = re.sub(r'\b(\d{4})-(\d{1,2})-(\d{1,2})\b', date_iso_to_text, text) |
| | text = re.sub(r'\b(\d{1,2})[/\-](\d{1,2})[/\-](\d{4})\b', date_to_text, text) |
| | text = re.sub(r'\b(\d{1,2})[/\-](\d{1,2})[/\-](\d{2})\b', date_short_year, text) |
| | |
| | return text |
| | |
| | def _normalize_phone(self, text): |
| | """Convert phone numbers to digit-by-digit reading.""" |
| | def phone_to_text(match): |
| | phone = match.group(0) |
| | phone = re.sub(r'[^\d]', '', phone) |
| | |
| | if phone.startswith('84') and len(phone) >= 10: |
| | phone = '0' + phone[2:] |
| | |
| | if 10 <= len(phone) <= 11: |
| | words = [self.digits[int(d)] for d in phone] |
| | return ' '.join(words) + ' ' |
| | |
| | return match.group(0) |
| | |
| | text = re.sub(r'(\+84|84)[\s\-\.]?\d[\d\s\-\.]{7,}', phone_to_text, text) |
| | text = re.sub(r'\b0\d[\d\s\-\.]{8,}', phone_to_text, text) |
| | return text |
| | |
| | def _normalize_numbers(self, text): |
| | text = re.sub(r'(\d+(?:[,.]\d+)?)%', lambda m: f'{m.group(1)} phần trăm', text) |
| | |
| | text = re.sub(r'(\d{1,3})(?:\.(\d{3}))+', lambda m: m.group(0).replace('.', ''), text) |
| | |
| | |
| | def decimal_to_words(match): |
| | whole = match.group(1) |
| | decimal = match.group(2) |
| | decimal_words = ' '.join([self.digits[int(d)] for d in decimal]) |
| | separator = 'phẩy' if ',' in match.group(0) else 'chấm' |
| | return f"{whole} {separator} {decimal_words}" |
| | |
| | |
| | text = re.sub(r'(\d+),(\d+)', decimal_to_words, text) |
| | |
| | text = re.sub(r'(\d+)\.(\d{1,2})\b', decimal_to_words, text) |
| | |
| | return text |
| | |
| | def _read_two_digits(self, n): |
| | """Read two-digit numbers in Vietnamese.""" |
| | if n < 10: |
| | return self.digits[n] |
| | elif n == 10: |
| | return "mười" |
| | elif n < 20: |
| | if n == 15: |
| | return "mười lăm" |
| | return f"mười {self.digits[n % 10]}" |
| | else: |
| | tens = n // 10 |
| | ones = n % 10 |
| | if ones == 0: |
| | return f"{self.digits[tens]} mươi" |
| | elif ones == 1: |
| | return f"{self.digits[tens]} mươi mốt" |
| | elif ones == 5: |
| | return f"{self.digits[tens]} mươi lăm" |
| | else: |
| | return f"{self.digits[tens]} mươi {self.digits[ones]}" |
| | |
| | def _read_three_digits(self, n): |
| | """Read three-digit numbers in Vietnamese.""" |
| | if n < 100: |
| | return self._read_two_digits(n) |
| | |
| | hundreds = n // 100 |
| | remainder = n % 100 |
| | result = f"{self.digits[hundreds]} trăm" |
| | |
| | if remainder == 0: |
| | return result |
| | elif remainder < 10: |
| | result += f" lẻ {self.digits[remainder]}" |
| | else: |
| | result += f" {self._read_two_digits(remainder)}" |
| | |
| | return result |
| | |
| | def _convert_number_to_words(self, num): |
| | """Convert a number to Vietnamese words.""" |
| | if num == 0: |
| | return "không" |
| | |
| | if num < 0: |
| | return f"âm {self._convert_number_to_words(-num)}" |
| | |
| | if num >= 1000000000: |
| | billion = num // 1000000000 |
| | remainder = num % 1000000000 |
| | result = f"{self._read_three_digits(billion)} tỷ" |
| | if remainder > 0: |
| | result += f" {self._convert_number_to_words(remainder)}" |
| | return result |
| | |
| | elif num >= 1000000: |
| | million = num // 1000000 |
| | remainder = num % 1000000 |
| | result = f"{self._read_three_digits(million)} triệu" |
| | if remainder > 0: |
| | result += f" {self._convert_number_to_words(remainder)}" |
| | return result |
| | |
| | elif num >= 1000: |
| | thousand = num // 1000 |
| | remainder = num % 1000 |
| | result = f"{self._read_three_digits(thousand)} nghìn" |
| | if remainder > 0: |
| | if remainder < 100: |
| | result += f" không trăm {self._read_two_digits(remainder)}" |
| | else: |
| | result += f" {self._read_three_digits(remainder)}" |
| | return result |
| | |
| | else: |
| | return self._read_three_digits(num) |
| | |
| | def _number_to_words(self, text): |
| | """Convert all remaining numbers to words.""" |
| | def convert_number(match): |
| | num = int(match.group(0)) |
| | return self._convert_number_to_words(num) |
| | |
| | text = re.sub(r'\b\d+\b', convert_number, text) |
| | return text |
| | |
| | def _normalize_special_chars(self, text): |
| | """Handle special characters.""" |
| | text = text.replace('&', ' và ') |
| | text = text.replace('+', ' cộng ') |
| | text = text.replace('=', ' bằng ') |
| | text = text.replace('#', ' thăng ') |
| | text = re.sub(r'[\[\]\(\)\{\}]', ' ', text) |
| | text = re.sub(r'\s+[-–—]+\s+', ' ', text) |
| | text = re.sub(r'\.{2,}', ' ', text) |
| | text = re.sub(r'\s+\.\s+', ' ', text) |
| | text = re.sub(r'[^\w\sàáảãạăắằẳẵặâấầẩẫậèéẻẽẹêếềểễệìíỉĩịòóỏõọôốồổỗộơớờởỡợùúủũụưứừửữựỳýỷỹỵđ.,!?;:@%]', ' ', text) |
| | return text |
| | |
| | def _normalize_whitespace(self, text): |
| | """Normalize whitespace.""" |
| | text = re.sub(r'\s+', ' ', text) |
| | text = text.strip() |
| | return text |
| |
|
| |
|
| | if __name__ == "__main__": |
| | normalizer = VietnameseTTSNormalizer() |
| | |
| | test_texts = [ |
| | "Giá 2.500.000đ (giảm 50%), mua trước 14h30 ngày 15/12/2025", |
| | "Liên hệ: 0912-345-678 hoặc email@example.com", |
| | "Tốc độ 120km/h, trọng lượng 75kg", |
| | "Nhiệt độ 36,5°C, độ ẩm 80%", |
| | "Số pi = 3,14159", |
| | "Giá trị tăng 2.5M, đạt 10B", |
| | "Nhiệt độ -15°C vào mùa đông", |
| | "Điện áp 220V, công suất 2.5kW, tần số 50Hz", |
| | "Tôi đi lấy l nước về nhà", |
| | "Cần 5l nước cho công thức này", |
| | "Vận tốc ánh sáng 299792km/s", |
| | "Mật độ dân số 450 người/km2", |
| | "Công suất 100 W/m2", |
| | "Hôm nay 2025-01-15", |
| | "Gọi +84 912 345 678", |
| | "Nhiệt độ 25°C lúc 14:30:45", |
| | "Ngày 15/12/25", |
| | "Giá 3.140.159", |
| | ] |
| | |
| | print("=" * 80) |
| | print("VIETNAMESE TTS NORMALIZATION TEST") |
| | print("=" * 80) |
| | |
| | for text in test_texts: |
| | print(f"\n📝 Input: {text}") |
| | normalized = normalizer.normalize(text) |
| | print(f"🎵 Output: {normalized}") |
| | print("-" * 80) |
| |
|