|
import os |
|
PWD = os.path.dirname(__file__) |
|
import re |
|
import regex |
|
import json |
|
import traceback |
|
|
|
from nemo_text_processing.text_normalization.normalize import Normalizer |
|
from indic_numtowords import num2words, supported_langs |
|
from .translator import GoogleTranslator |
|
|
|
indic_acronym_matcher = regex.compile(r"([\p{L}\p{M}]+\.\s*){2,}") |
|
|
|
|
|
|
|
|
|
|
|
short_form_regex = re.compile(r"\b([A-Z][\.\s]+)+([A-Z])?\b") |
|
eng_consonants_regex = re.compile(r"\b[BCDFGHJKLMNPQRSTVWXZbcdfghjklmnpqrstvwxz]+\b") |
|
def get_shortforms_from_string(text): |
|
dotted_shortforms = [m.group() for m in re.finditer(short_form_regex, text)] |
|
non_dotted_shortforms = [m.group() for m in re.finditer(eng_consonants_regex, text)] |
|
return dotted_shortforms + non_dotted_shortforms |
|
|
|
decimal_str_regex = re.compile("\d{1,3}(?:(?:,\d{2,3}){1,3}|(?:\d{1,7}))?(?:\.\d+)") |
|
def get_all_decimals_from_string(text): |
|
return decimal_str_regex.findall(text) |
|
|
|
num_str_regex = re.compile("\d{1,3}(?:(?:,\d{2,3}){1,3}|(?:\d{1,7}))?(?:\.\d+)?") |
|
def get_all_numbers_from_string(text): |
|
return num_str_regex.findall(text) |
|
|
|
multiple_stops_regex = r'\.\.+' |
|
def replace_multiple_stops(text): |
|
return re.sub(multiple_stops_regex, '.', text) |
|
|
|
date_generic_match_regex = re.compile("(?:[^0-9]\d*[./-]\d*[./-]\d*)") |
|
date_str_regex = re.compile("(?:\d{1,2}[./-]\d{1,2}[./-]\d{2,4})|(?:\d{2,4}[./-]\d{1,2}[./-]\d{1,2})") |
|
def get_all_dates_from_string(text): |
|
candidates = date_generic_match_regex.findall(text) |
|
candidates = [c.replace(' ', '') for c in candidates] |
|
candidates = [c for c in candidates if len(c) <= 10] |
|
candidates = ' '.join(candidates) |
|
return date_str_regex.findall(candidates) |
|
|
|
def get_decimal_substitution(decimal): |
|
decimal_parts = decimal.split('.') |
|
l_part = decimal_parts[0] |
|
r_part = "" |
|
for part in decimal_parts[1:]: |
|
r_part += ' '.join(list(part)) |
|
decimal_sub = l_part + " point " + r_part |
|
decimal_sub = decimal_sub.strip() |
|
return decimal_sub |
|
|
|
email_regex = r'[\w.+-]+@[\w-]+\.[\w.-]+' |
|
url_regex = r'((?:\w+://)?\w+\.\w+\.\w+/?[\w\.\?=#]*)|(\w*.com/?[\w\.\?=#]*)' |
|
currency_regex = r"\₹\ ?[+-]?[0-9]{1,3}(?:,?[0-9])*(?:\.[0-9]{1,2})?" |
|
phone_regex = r'\+?\d[ \d-]{6,12}\d' |
|
|
|
|
|
|
|
class TextNormalizer: |
|
def __init__(self): |
|
self.translator = GoogleTranslator() |
|
self.normalizer = Normalizer(input_case='cased', lang='en') |
|
self.symbols2lang2word = json.load(open(os.path.join(PWD, "symbols.json"), encoding="utf-8")) |
|
self.alphabet2phone = json.load(open(os.path.join(PWD, "alphabet2phone.json"), encoding="utf-8")) |
|
|
|
def normalize_text(self, text, lang): |
|
text = text.replace("।", ".").replace("|", ".").replace("꯫", ".").strip() |
|
text = self.expand_shortforms(text, lang) |
|
text = self.normalize_decimals(text, lang) |
|
text = self.replace_punctutations(text, lang) |
|
text = self.convert_dates_to_words(text, lang) |
|
text = self.convert_symbols_to_words(text, lang) |
|
text = self.convert_numbers_to_words(text, lang) |
|
return text |
|
|
|
def normalize_decimals(self, text, lang): |
|
decimal_strs = get_all_decimals_from_string(text) |
|
if not decimal_strs: |
|
return text |
|
decimals = [str(decimal_str.replace(',', '')) for decimal_str in decimal_strs] |
|
decimal_substitutions = [get_decimal_substitution(decimal) for decimal in decimals] |
|
for decimal_str, decimal_sub in zip(decimal_strs, decimal_substitutions): |
|
text = text.replace(decimal_str, decimal_sub) |
|
return text |
|
|
|
def replace_punctutations(self, text, lang): |
|
text = replace_multiple_stops(text) |
|
if lang not in ['brx', 'or']: |
|
text = text.replace('।', '.') |
|
if text[-1] not in ['.', '!', '?', ',', ':', ';']: |
|
text = text + ' .' |
|
else: |
|
text = text.replace('.', '।') |
|
text = text.replace('|', '.') |
|
for bracket in ['(', ')', '{', '}', '[', ']']: |
|
text = text.replace(bracket, ',') |
|
|
|
text = text.replace(';',',') |
|
return text |
|
|
|
def convert_numbers_to_words(self, text, lang): |
|
num_strs = get_all_numbers_from_string(text) |
|
if not num_strs: |
|
return text |
|
|
|
|
|
|
|
numbers = [int(num_str.replace(',', '')) for num_str in num_strs] |
|
|
|
if lang in supported_langs: |
|
|
|
num_words = [num2words(num, lang=lang) for num in numbers] |
|
else: |
|
try: |
|
num_words = [num2words(num, lang="en") for num in numbers] |
|
translated_num_words = [self.translator(text=num_word, from_lang="en", to_lang=lang) for num_word in num_words] |
|
|
|
num_words = translated_num_words |
|
except: |
|
traceback.print_exc() |
|
|
|
for num_str, num_word in zip(num_strs, num_words): |
|
text = text.replace(num_str, ' '+num_word+' ', 1) |
|
return text.replace(" ", ' ') |
|
|
|
def convert_dates_to_words(self, text, lang): |
|
date_strs = get_all_dates_from_string(text) |
|
if not date_strs: |
|
return text |
|
for date_str in date_strs: |
|
normalized_str = self.normalizer.normalize(date_str, verbose=False, punct_post_process=True) |
|
if lang in ['brx', 'en']: |
|
translated_str = normalized_str |
|
else: |
|
translated_str = self.translator(text=normalized_str, from_lang="en", to_lang=lang) |
|
text = text.replace(date_str, translated_str) |
|
return text |
|
|
|
def expand_phones(self, item): |
|
return ' '.join(list(item)) |
|
|
|
def find_valid(self, regex_str, text): |
|
items = re.findall(regex_str, text) |
|
return_items = [] |
|
for item in items: |
|
if isinstance(item, tuple): |
|
for subitem in item: |
|
if len(subitem) > 0: |
|
return_items.append(subitem) |
|
break |
|
elif len(item) > 0: |
|
return_items.append(item) |
|
return return_items |
|
|
|
def convert_symbols_to_words(self, text, lang): |
|
symbols = self.symbols2lang2word.keys() |
|
emails = self.find_valid(email_regex, text) |
|
|
|
urls = self.find_valid(url_regex, text) |
|
|
|
for item in emails + urls: |
|
item_norm = item |
|
for symbol in symbols: |
|
item_norm = item_norm.replace(symbol, f' {self.symbols2lang2word[symbol][lang]} ') |
|
text = text.replace(item, item_norm) |
|
|
|
currencies = self.find_valid(currency_regex, text) |
|
for item in currencies: |
|
item_norm = item.replace('₹','') + '₹' |
|
for symbol in symbols: |
|
item_norm = item_norm.replace(symbol, f' {self.symbols2lang2word[symbol][lang]} ') |
|
text = text.replace(item, item_norm) |
|
|
|
phones = self.find_valid(phone_regex, text) |
|
for item in phones: |
|
item_norm = item.replace('-', ' ') |
|
for symbol in symbols: |
|
item_norm = item_norm.replace(symbol, f' {self.symbols2lang2word[symbol][lang]} ') |
|
item_norm = self.expand_phones(item_norm) |
|
text = text.replace(item, item_norm) |
|
|
|
|
|
text = text.replace('%', self.symbols2lang2word['%'][lang]) |
|
|
|
return text |
|
|
|
def convert_char2phone(self, char): |
|
return self.alphabet2phone[char.lower()] if char.lower() in self.alphabet2phone else '' |
|
|
|
def expand_shortforms(self, text, lang): |
|
if lang!='en': |
|
|
|
|
|
for match in regex.finditer(indic_acronym_matcher, text): |
|
match = match.group() |
|
match_without_dot = match.replace('.', ' ') |
|
text = text.replace(match, match_without_dot) |
|
return text |
|
|
|
shortforms = get_shortforms_from_string(text) |
|
for shortform in shortforms: |
|
shortform = shortform.strip() |
|
if shortform == 'I' or shortform == "A": |
|
|
|
continue |
|
expanded = ' '.join([self.convert_char2phone(char) for char in shortform]) |
|
text = text.replace(shortform, expanded, 1) |
|
return text |
|
|