danswer/backend/onyx/utils/text_processing.py
2024-12-13 09:56:10 -08:00

163 lines
4.6 KiB
Python

import codecs
import json
import re
import string
from urllib.parse import quote
from onyx.utils.logger import setup_logger
logger = setup_logger(__name__)
ESCAPE_SEQUENCE_RE = re.compile(
r"""
( \\U........ # 8-digit hex escapes
| \\u.... # 4-digit hex escapes
| \\x.. # 2-digit hex escapes
| \\[0-7]{1,3} # Octal escapes
| \\N\{[^}]+\} # Unicode characters by name
| \\[\\'"abfnrtv] # Single-character escapes
)""",
re.UNICODE | re.VERBOSE,
)
def decode_escapes(s: str) -> str:
def decode_match(match: re.Match) -> str:
return codecs.decode(match.group(0), "unicode-escape")
return ESCAPE_SEQUENCE_RE.sub(decode_match, s)
def make_url_compatible(s: str) -> str:
s_with_underscores = s.replace(" ", "_")
return quote(s_with_underscores, safe="")
def has_unescaped_quote(s: str) -> bool:
pattern = r'(?<!\\)"'
return bool(re.search(pattern, s))
def escape_newlines(s: str) -> str:
return re.sub(r"(?<!\\)\n", "\\\\n", s)
def replace_whitespaces_w_space(s: str) -> str:
return re.sub(r"\s", " ", s)
# Function to remove punctuation from a string
def remove_punctuation(s: str) -> str:
return s.translate(str.maketrans("", "", string.punctuation))
def escape_quotes(original_json_str: str) -> str:
result = []
in_string = False
for i, char in enumerate(original_json_str):
if char == '"':
if not in_string:
in_string = True
result.append(char)
else:
next_char = (
original_json_str[i + 1] if i + 1 < len(original_json_str) else None
)
if result and result[-1] == "\\":
result.append(char)
elif next_char not in [",", ":", "}", "\n"]:
result.append("\\" + char)
else:
result.append(char)
in_string = False
else:
result.append(char)
return "".join(result)
def extract_embedded_json(s: str) -> dict:
first_brace_index = s.find("{")
last_brace_index = s.rfind("}")
if first_brace_index == -1 or last_brace_index == -1:
logger.warning("No valid json found, assuming answer is entire string")
return {"answer": s, "quotes": []}
json_str = s[first_brace_index : last_brace_index + 1]
try:
return json.loads(json_str, strict=False)
except json.JSONDecodeError:
try:
return json.loads(escape_quotes(json_str), strict=False)
except json.JSONDecodeError as e:
raise ValueError("Failed to parse JSON, even after escaping quotes") from e
def clean_up_code_blocks(model_out_raw: str) -> str:
return model_out_raw.strip().strip("```").strip().replace("\\xa0", "")
def clean_model_quote(quote: str, trim_length: int) -> str:
quote_clean = quote.strip()
if quote_clean[0] == '"':
quote_clean = quote_clean[1:]
if quote_clean[-1] == '"':
quote_clean = quote_clean[:-1]
if trim_length > 0:
quote_clean = quote_clean[:trim_length]
return quote_clean
def shared_precompare_cleanup(text: str) -> str:
"""LLMs models sometime restructure whitespaces or edits special characters to fit a more likely
distribution of characters found in its training data, but this hurts exact quote matching
"""
text = text.lower()
# \s: matches any whitespace character (spaces, tabs, newlines, etc.)
# |: acts as an OR.
# \*: matches the asterisk character.
# \\": matches the \" sequence.
# [.,:`"#-]: matches any character inside the square brackets.
text = re.sub(r'\s|\*|\\"|[.,:`"#-]', "", text)
return text
_INITIAL_FILTER = re.compile(
"["
"\U0000FFF0-\U0000FFFF" # Specials
"\U0001F000-\U0001F9FF" # Emoticons
"\U00002000-\U0000206F" # General Punctuation
"\U00002190-\U000021FF" # Arrows
"\U00002700-\U000027BF" # Dingbats
"]+",
flags=re.UNICODE,
)
def clean_text(text: str) -> str:
# Remove specific Unicode ranges that might cause issues
cleaned = _INITIAL_FILTER.sub("", text)
# Remove any control characters except for newline and tab
cleaned = "".join(ch for ch in cleaned if ch >= " " or ch in "\n\t")
return cleaned
def is_valid_email(text: str) -> bool:
"""Can use a library instead if more detailed checks are needed"""
regex = r"^[a-zA-Z0-9._-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$"
if re.match(regex, text):
return True
else:
return False
def count_punctuation(text: str) -> int:
return sum(1 for char in text if char in string.punctuation)