# -*- coding: utf-8 -*-
# SPDX-FileCopyrightText: 2016-2025 PyThaiNLP Project
# SPDX-FileType: SOURCE
# SPDX-License-Identifier: Apache-2.0
# ruff: noqa: C901
from typing import List, Union
from pythainlp.tokenize import subword_tokenize
from pythainlp.util import remove_tonemark, sound_syllable
[docs]
class KhaveeVerifier:
[docs]
def __init__(self):
"""
KhaveeVerifier: Thai Poetry verifier
"""
[docs]
def check_sara(self, word: str) -> str:
"""
Check the vowels in the Thai word.
:param str word: Thai word
:return: vowel name of the word
:rtype: str
:Example:
::
from pythainlp.khavee import KhaveeVerifier
kv = KhaveeVerifier()
print(kv.check_sara("เริง"))
# output: 'เออ'
"""
sara = []
countoa = 0
# In case of การันย์
if "์" in word[-1]:
word = word[:-2]
# In case of สระเดี่ยว
for i in word:
if i in ("ะ", "ั"):
sara.append("อะ")
elif i == "ิ":
sara.append("อิ")
elif i == "ุ":
sara.append("อุ")
elif i == "ึ":
sara.append("อึ")
elif i == "ี":
sara.append("อี")
elif i == "ู":
sara.append("อู")
elif i == "ื":
sara.append("อือ")
elif i == "เ":
sara.append("เอ")
elif i == "แ":
sara.append("แอ")
elif i == "า":
sara.append("อา")
elif i == "โ":
sara.append("โอ")
elif i == "ำ":
sara.append("อำ")
elif i == "อ":
countoa += 1
sara.append("ออ")
elif i == "ั" and "ว" in word:
sara.append("อัว")
elif i in ("ไ", "ใ"):
sara.append("ไอ")
elif i == "็":
sara.append("ออ")
elif "รร" in word:
if self.check_marttra(word) == "กม":
sara.append("อำ")
else:
sara.append("อะ")
# In case of ออ
if countoa == 1 and "อ" in word[-1] and "เ" not in word:
sara.remove("ออ")
# In case of เอ เอ
countA = 0
for i in sara:
if i == "เอ":
countA = countA + 1
if countA > 1:
sara.remove("เอ")
sara.remove("เอ")
sara.append("แ")
# In case of สระประสม
if "เอ" in sara and "อะ" in sara:
sara.remove("เอ")
sara.remove("อะ")
sara.append("เอะ")
elif "แอ" in sara and "อะ" in sara:
sara.remove("แอ")
sara.remove("อะ")
sara.append("แอะ")
if "เอะ" in sara and "ออ" in sara:
sara.remove("เอะ")
sara.remove("ออ")
sara.append("เออะ")
elif "เอ" in sara and "อิ" in sara:
sara.remove("เอ")
sara.remove("อิ")
sara.append("เออ")
elif "เอ" in sara and "ออ" in sara and "อ" in word[-1]:
sara.remove("เอ")
sara.remove("ออ")
sara.append("เออ")
elif "โอ" in sara and "อะ" in sara:
sara.remove("โอ")
sara.remove("อะ")
sara.append("โอะ")
elif "เอ" in sara and "อี" in sara:
sara.remove("เอ")
sara.remove("อี")
sara.append("เอีย")
elif "เอ" in sara and "อือ" in sara:
sara.remove("เอ")
sara.remove("อือ")
sara.append("อัว")
elif "เอ" in sara and "อา" in sara:
sara.remove("เอ")
sara.remove("อา")
sara.append("เอา")
elif "เ" in word and "า" in word and "ะ" in word:
sara = []
sara.append("เอาะ")
if "อือ" in sara and "เออ" in sara:
sara.remove("เออ")
sara.remove("อือ")
sara.append("เอือ")
elif "ออ" in sara and len(sara) > 1:
sara.remove("ออ")
elif "ว" in word and len(sara) == 0:
sara.append("อัว")
if "ั" in word and self.check_marttra(word) == "กา":
sara = []
sara.append("ไอ")
# In case of อ
if word == "เออะ":
sara = []
sara.append("เออะ")
elif word == "เออ":
sara = []
sara.append("เออ")
elif word == "เอ":
sara = []
sara.append("เอ")
elif word == "เอะ":
sara = []
sara.append("เอะ")
elif word == "เอา":
sara = []
sara.append("เอา")
elif word == "เอาะ":
sara = []
sara.append("เอาะ")
if "ฤา" in word or "ฦา" in word:
sara = []
sara.append("อือ")
elif "ฤ" in word or "ฦ" in word:
sara = []
sara.append("อึ")
# In case of กน
if not sara and len(word) == 2:
if word[-1] != "ร":
sara.append("โอะ")
else:
sara.append("ออ")
elif not sara and len(word) == 3:
sara.append("ออ")
# In case of บ่
if word == "บ่":
sara = []
sara.append("ออ")
if "ํ" in word:
sara = []
sara.append("อำ")
if "เ" in word and "ื" in word and "อ" in word:
sara = []
sara.append("เอือ")
if not sara:
return "Can't find Sara in this word"
return sara[0]
[docs]
def check_marttra(self, word: str) -> str:
"""
Check the Thai spelling Section in the Thai word.
:param str word: Thai word
:return: name of spelling Section of the word.
:rtype: str
:Example:
::
from pythainlp.khavee import KhaveeVerifier
kv = KhaveeVerifier()
print(kv.check_marttra("สาว"))
# output: 'เกอว'
"""
if word[-1] == "ร" and word[-2] in ["ต", "ท"]:
word = word[:-1]
word = self.handle_karun_sound_silence(word)
word = remove_tonemark(word)
if (
"ำ" in word
or ("ํ" in word and "า" in word)
or "ไ" in word
or "ใ" in word
):
return "กา"
elif (
word[-1] in ["า", "ะ", "ิ", "ี", "ุ", "ู", "อ"]
or ("ี" in word and "ย" in word[-1])
or ("ื" in word and "อ" in word[-1])
):
return "กา"
elif word[-1] in ["ง"]:
return "กง"
elif word[-1] in ["ม"]:
return "กม"
elif word[-1] in ["ย"]:
if "ั" in word:
return "กา"
else:
return "เกย"
elif word[-1] in ["ว"]:
return "เกอว"
elif word[-1] in ["ก", "ข", "ค", "ฆ"]:
return "กก"
elif word[-1] in [
"จ",
"ช",
"ซ",
"ฎ",
"ฏ",
"ฐ",
"ฑ",
"ฒ",
"ด",
"ต",
"ถ",
"ท",
"ธ",
"ศ",
"ษ",
"ส",
]:
return "กด"
elif word[-1] in ["ญ", ", ณ", "น", "ร", "ล", "ฬ"]:
return "กน"
elif word[-1] in ["บ", "ป", "พ", "ฟ", "ภ"]:
return "กบ"
else:
if "็" in word:
return "กา"
else:
return "Cant find Marttra in this word"
[docs]
def is_sumpus(self, word1: str, word2: str) -> bool:
"""
Check the rhyme between two words.
:param str word1: Thai word
:param str word2: Thai word
:return: boolean
:rtype: bool
:Example:
::
from pythainlp.khavee import KhaveeVerifier
kv = KhaveeVerifier()
print(kv.is_sumpus("สรร", "อัน"))
# output: True
print(kv.is_sumpus("สรร", "แมว"))
# output: False
"""
marttra1 = self.check_marttra(word1)
marttra2 = self.check_marttra(word2)
sara1 = self.check_sara(word1)
sara2 = self.check_sara(word2)
if sara1 == "อะ" and marttra1 == "เกย":
sara1 = "ไอ"
marttra1 = "กา"
elif sara2 == "อะ" and marttra2 == "เกย":
sara2 = "ไอ"
marttra2 = "กา"
if sara1 == "อำ" and marttra1 == "กม":
sara1 = "อำ"
marttra1 = "กา"
elif sara2 == "อำ" and marttra2 == "กม":
sara2 = "อำ"
marttra2 = "กา"
return bool(marttra1 == marttra2 and sara1 == sara2)
[docs]
def check_karu_lahu(self, text):
if (
self.check_marttra(text) != "กา"
or (
self.check_marttra(text) == "กา"
and self.check_sara(text)
in [
"อา",
"อี",
"อือ",
"อู",
"เอ",
"แอ",
"โอ",
"ออ",
"เออ",
"เอีย",
"เอือ",
"อัว",
]
)
or self.check_sara(text) in ["อำ", "ไอ", "เอา"]
) and text not in ["บ่", "ณ", "ธ", "ก็"]:
return "karu"
else:
return "lahu"
[docs]
def check_klon(self, text: str, k_type: int = 8) -> Union[List[str], str]:
"""
Check the suitability of the poem according to Thai principles.
:param str text: Thai poem
:param int k_type: type of Thai poem
:return: the check results of the suitability of the poem according to Thai principles.
:rtype: Union[List[str], str]
:Example:
::
from pythainlp.khavee import KhaveeVerifier
kv = KhaveeVerifier()
print(kv.check_klon(
'ฉันชื่อหมูกรอบ ฉันชอบกินไก่ แล้วก็วิ่งไล่ หมาชื่อนํ้าทอง ลคคนเก่ง เอ๋งเอ๋งคะนอง \
มีคนจับจอง เขาชื่อน้องเธียร',
k_type=4
))
# output: The poem is correct according to the principle.
print(kv.check_klon(
'ฉันชื่อหมูกรอบ ฉันชอบกินไก่ แล้วก็วิ่งไล่ หมาชื่อนํ้าทอง ลคคนเก่ง \
เอ๋งเอ๋งเสียงหมา มีคนจับจอง เขาชื่อน้องเธียร',
k_type=4
))
# output: [
"Can't find rhyme between paragraphs ('หมา', 'จอง') in paragraph 2",
"Can't find rhyme between paragraphs ('หมา', 'ทอง') in paragraph 2"
]
"""
if k_type == 8:
try:
error = []
list_sumpus_sent1 = []
list_sumpus_sent2h = []
list_sumpus_sent2l = []
list_sumpus_sent3 = []
list_sumpus_sent4 = []
for i, sent in enumerate(text.split()):
sub_sent = subword_tokenize(sent, engine="dict")
if len(sub_sent) > 10:
error.append(
"In sentence "
+ str(i + 2)
+ ", there are more than 10 words. "
+ str(sub_sent)
)
if (i + 1) % 4 == 1:
list_sumpus_sent1.append(sub_sent[-1])
elif (i + 1) % 4 == 2:
list_sumpus_sent2h.append(
[
sub_sent[1],
sub_sent[2],
sub_sent[3],
sub_sent[4],
]
)
list_sumpus_sent2l.append(sub_sent[-1])
elif (i + 1) % 4 == 3:
list_sumpus_sent3.append(sub_sent[-1])
elif (i + 1) % 4 == 0:
list_sumpus_sent4.append(sub_sent[-1])
if (
len(list_sumpus_sent1) != len(list_sumpus_sent2h)
or len(list_sumpus_sent2h) != len(list_sumpus_sent2l)
or len(list_sumpus_sent2l) != len(list_sumpus_sent3)
or len(list_sumpus_sent3) != len(list_sumpus_sent4)
or len(list_sumpus_sent4) != len(list_sumpus_sent1)
):
return "The poem does not have 4 complete sentences."
else:
for i in range(len(list_sumpus_sent1)):
countwrong = 0
for j in list_sumpus_sent2h[i]:
if (
self.is_sumpus(list_sumpus_sent1[i], j)
is False
):
countwrong += 1
if countwrong > 3:
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent1[i],
list_sumpus_sent2h[i],
)
)
+ " in paragraph "
+ str(i + 1)
)
if (
self.is_sumpus(
list_sumpus_sent2l[i], list_sumpus_sent3[i]
)
is False
):
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent2l[i],
list_sumpus_sent3[i],
)
)
+ " in paragraph "
+ str(i + 1)
)
if i > 0:
if (
self.is_sumpus(
list_sumpus_sent2l[i],
list_sumpus_sent4[i - 1],
)
is False
):
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent2l[i],
list_sumpus_sent4[i - 1],
)
)
+ " in paragraph "
+ str(i + 1)
)
if not error:
return (
"The poem is correct according to the principle."
)
else:
return error
except:
return "Something went wrong. Make sure you enter it in the correct form of klon 8."
elif k_type == 4:
try:
error = []
list_sumpus_sent1 = []
list_sumpus_sent2h = []
list_sumpus_sent2l = []
list_sumpus_sent3 = []
list_sumpus_sent4 = []
for i, sent in enumerate(text.split()):
sub_sent = subword_tokenize(sent, engine="dict")
if len(sub_sent) > 5:
error.append(
"In sentence "
+ str(i + 2)
+ ", there are more than 4 words. "
+ str(sub_sent)
)
if (i + 1) % 4 == 1:
list_sumpus_sent1.append(sub_sent[-1])
elif (i + 1) % 4 == 2:
list_sumpus_sent2h.append([sub_sent[1], sub_sent[2]])
list_sumpus_sent2l.append(sub_sent[-1])
elif (i + 1) % 4 == 3:
list_sumpus_sent3.append(sub_sent[-1])
elif (i + 1) % 4 == 0:
list_sumpus_sent4.append(sub_sent[-1])
if (
len(list_sumpus_sent1) != len(list_sumpus_sent2h)
or len(list_sumpus_sent2h) != len(list_sumpus_sent2l)
or len(list_sumpus_sent2l) != len(list_sumpus_sent3)
or len(list_sumpus_sent3) != len(list_sumpus_sent4)
or len(list_sumpus_sent4) != len(list_sumpus_sent1)
):
return "The poem does not have 4 complete sentences."
else:
for i in range(len(list_sumpus_sent1)):
countwrong = 0
for j in list_sumpus_sent2h[i]:
if (
self.is_sumpus(list_sumpus_sent1[i], j)
is False
):
countwrong += 1
if countwrong > 1:
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent1[i],
list_sumpus_sent2h[i],
)
)
+ " in paragraph "
+ str(i + 1)
)
if (
self.is_sumpus(
list_sumpus_sent2l[i], list_sumpus_sent3[i]
)
is False
):
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent2l[i],
list_sumpus_sent3[i],
)
)
+ " in paragraph "
+ str(i + 1)
)
if i > 0:
if (
self.is_sumpus(
list_sumpus_sent2l[i],
list_sumpus_sent4[i - 1],
)
is False
):
error.append(
"Can't find rhyme between paragraphs "
+ str(
(
list_sumpus_sent2l[i],
list_sumpus_sent4[i - 1],
)
)
+ " in paragraph "
+ str(i + 1)
)
if not error:
return (
"The poem is correct according to the principle."
)
else:
return error
except:
return "Something went wrong. Make sure you enter it in the correct form."
else:
return "Something went wrong. Make sure you enter it in the correct form."
[docs]
def check_aek_too(
self, text: Union[List[str], str], dead_syllable_as_aek: bool = False
) -> Union[List[bool], List[str], bool, str]:
"""
Checker of Thai tonal words
:param Union[List[str], str] text: Thai word or list of Thai words
:param bool dead_syllable_as_aek: if True, dead syllable will be considered as aek
:return: the check result if the word is aek or too or False (not both) or list of check results if input is list
:rtype: Union[List[bool], List[str], bool, str]
:Example:
::
from pythainlp.khavee import KhaveeVerifier
kv = KhaveeVerifier()
# การเช็คคำเอกโท
print(
kv.check_aek_too("เอง"),
kv.check_aek_too("เอ่ง"),
kv.check_aek_too("เอ้ง"),
)
# -> False, aek, too
print(kv.check_aek_too(["เอง", "เอ่ง", "เอ้ง"])) # ใช้ List ได้เหมือนกัน
# -> [False, 'aek', 'too']
"""
if isinstance(text, list):
return [self.check_aek_too(t, dead_syllable_as_aek) for t in text]
if not isinstance(text, str):
raise TypeError("text must be str or iterable list[str]")
word_characters = [*text]
if "่" in word_characters and "้" not in word_characters:
return "aek"
elif "้" in word_characters and "่" not in word_characters:
return "too"
if dead_syllable_as_aek and sound_syllable(text) == "dead":
return "aek"
else:
return False
[docs]
def handle_karun_sound_silence(self, word: str) -> str:
"""
Handle silent sounds in Thai words using '์' character (Karun)
by stripping all characters before the 'Karun' character that should be silenced
:param str text: Thai word
:return: Thai word with silent words stripped
:rtype: str
"""
sound_silenced = word.endswith("์")
if not sound_silenced:
return word
thai_consonants = "กขฃคฅฆงจฉชซฌญฎฏฐฑฒณดตถทธนบปผฝพฟภมยรลวศษสหฬอฮ"
locate_silenced = word.rfind("์") - 1
can_silence_two = word[locate_silenced - 2] in thai_consonants
cut_off = 2 if can_silence_two else 1
word = word[: locate_silenced + 1 - cut_off]
return word