Spaces:
Sleeping
Sleeping
fruitpicker01
commited on
Update app.py
Browse files
app.py
CHANGED
@@ -8,7 +8,7 @@ from langchain_community.chat_models.gigachat import GigaChat
|
|
8 |
from openpyxl import load_workbook
|
9 |
import plotly.graph_objects as go
|
10 |
import random
|
11 |
-
import
|
12 |
import string
|
13 |
import json
|
14 |
from mistralai import Mistral
|
@@ -17,37 +17,37 @@ import requests
|
|
17 |
import base64
|
18 |
import io
|
19 |
|
20 |
-
|
21 |
token = os.getenv('GITHUB_TOKEN')
|
22 |
|
23 |
# Клиент для генерации сообщений
|
24 |
-
|
25 |
|
26 |
# Клиент для выполнения проверок
|
27 |
-
|
28 |
|
29 |
-
morph =
|
30 |
|
31 |
# Авторизация в GigaChat Pro
|
32 |
-
gc_key = os.getenv('GC_KEY')
|
33 |
#chat_pro = GigaChat(credentials=gc_key, model='GigaChat', max_tokens=68, temperature=1.15, verify_ssl_certs=False)
|
34 |
-
chat_pro = GigaChat(
|
35 |
-
credentials=gc_key,
|
36 |
-
model='GigaChat-Pro-preview',
|
37 |
-
base_url='https://gigachat-preview.devices.sberbank.ru/api/v1/',
|
38 |
-
max_tokens=68,
|
39 |
-
temperature=1.15,
|
40 |
-
verify_ssl_certs=False
|
41 |
-
)
|
42 |
-
|
43 |
-
chat_pro_check = GigaChat(
|
44 |
-
credentials=gc_key,
|
45 |
-
model='GigaChat-Pro-preview',
|
46 |
-
base_url='https://gigachat-preview.devices.sberbank.ru/api/v1/',
|
47 |
-
max_tokens=3000,
|
48 |
-
temperature=0.8,
|
49 |
-
verify_ssl_certs=False
|
50 |
-
)
|
51 |
|
52 |
approach_stats = {
|
53 |
"Начни сообщение с призыва к действию с продуктом.": {"failed_checks": defaultdict(int), "total_attempts": 0},
|
@@ -74,30 +74,6 @@ for sheet_name, df in data.items():
|
|
74 |
print(f"Ошибка при обработке данных листа {sheet_name}: {e}")
|
75 |
features[sheet_name] = {}
|
76 |
|
77 |
-
# Функция для создания спидометра
|
78 |
-
def create_gauge(value):
|
79 |
-
fig = go.Figure(go.Indicator(
|
80 |
-
mode="gauge+number",
|
81 |
-
value=value,
|
82 |
-
gauge={
|
83 |
-
'axis': {'range': [0, 100]},
|
84 |
-
'bar': {'color': "black"}, # Цвет стрелки
|
85 |
-
'steps': [
|
86 |
-
{'range': [0, 40], 'color': "#55efc4"}, # Мягкий зеленый
|
87 |
-
{'range': [40, 70], 'color': "#ffeaa7"}, # Желтый
|
88 |
-
{'range': [70, 100], 'color': "#ff7675"} # Мягкий красный
|
89 |
-
],
|
90 |
-
'threshold': {
|
91 |
-
'line': {'color': "black", 'width': 4},
|
92 |
-
'thickness': 0.75,
|
93 |
-
'value': value
|
94 |
-
}
|
95 |
-
},
|
96 |
-
number={'font': {'size': 48}} # Размер шрифта числа
|
97 |
-
))
|
98 |
-
fig.update_layout(paper_bgcolor="#f8f9fa", font={'color': "#2d3436", 'family': "Arial"}, width=250, height=150)
|
99 |
-
return fig
|
100 |
-
|
101 |
def save_statistics_to_github(approach_stats):
|
102 |
repo = "fruitpicker01/Storage_dev"
|
103 |
timestamp = int(time.time())
|
@@ -170,10 +146,6 @@ def save_statistics_to_github(approach_stats):
|
|
170 |
else:
|
171 |
print(f"Ошибка при сохранении CSV-файла: {response.status_code} {response.text}")
|
172 |
|
173 |
-
# Функция для генерации случайных значений спидометров
|
174 |
-
def generate_random_gauges():
|
175 |
-
return create_gauge(random.randint(80, 95)), create_gauge(random.randint(80, 95)), create_gauge(random.randint(80, 95))
|
176 |
-
|
177 |
# Функция для смены вкладки
|
178 |
def change_tab(id):
|
179 |
return gr.Tabs(selected=id)
|
@@ -191,77 +163,77 @@ def clean_message(message):
|
|
191 |
return message
|
192 |
|
193 |
# Функция для генерации сообщения с GigaChat Pro
|
194 |
-
def generate_message_gigachat_pro(prompt):
|
195 |
-
try:
|
196 |
-
messages = [SystemMessage(content=prompt)]
|
197 |
-
res = chat_pro(messages)
|
198 |
-
cleaned_message = clean_message(res.content.strip())
|
199 |
-
return cleaned_message
|
200 |
-
except Exception as e:
|
201 |
-
return f"Ошибка при обращении к GigaChat-Pro: {e}"
|
202 |
-
|
203 |
-
|
204 |
-
|
205 |
-
|
206 |
-
|
207 |
-
|
208 |
-
|
209 |
-
|
210 |
-
|
211 |
-
|
212 |
-
|
213 |
-
|
214 |
-
|
215 |
-
|
216 |
-
|
217 |
-
|
218 |
-
|
219 |
-
|
220 |
-
|
221 |
-
|
222 |
-
|
223 |
-
|
224 |
-
|
225 |
-
|
226 |
-
|
227 |
-
|
228 |
-
|
229 |
-
|
230 |
-
|
231 |
-
|
232 |
-
|
233 |
-
|
234 |
-
|
235 |
-
|
236 |
-
|
237 |
-
|
238 |
-
|
239 |
-
|
240 |
-
|
241 |
-
|
242 |
-
|
243 |
-
|
244 |
-
|
245 |
-
|
246 |
-
|
247 |
-
|
248 |
-
|
249 |
-
|
250 |
-
|
251 |
-
|
252 |
-
|
253 |
-
|
254 |
-
|
255 |
-
|
256 |
-
|
257 |
-
def generate_check_gigachat_pro(prompt):
|
258 |
-
try:
|
259 |
-
messages = [SystemMessage(content=prompt)]
|
260 |
-
res2 = chat_pro_check(messages)
|
261 |
-
cleaned_message = clean_message(res2.content.strip())
|
262 |
-
return cleaned_message
|
263 |
-
except Exception as e:
|
264 |
-
return f"Ошибка при обращении к GigaChat-Pro: {e}"
|
265 |
|
266 |
# Функция для замены сокращений с 'k' или 'К' на тысячи
|
267 |
def replace_k_with_thousands(message):
|
@@ -270,7 +242,7 @@ def replace_k_with_thousands(message):
|
|
270 |
return message
|
271 |
|
272 |
def correct_dash_usage(text):
|
273 |
-
morph =
|
274 |
# Step 1: Replace any dash with long dash if surrounded by spaces
|
275 |
text = re.sub(r'\s[-–—]\s', ' — ', text)
|
276 |
|
@@ -323,7 +295,7 @@ def correct_dash_usage(text):
|
|
323 |
text = ' '.join(sentences)
|
324 |
|
325 |
def restore_yo(text):
|
326 |
-
morph =
|
327 |
words = text.split()
|
328 |
restored_words = []
|
329 |
|
@@ -363,9 +335,13 @@ def correct_dash_usage(text):
|
|
363 |
text = re.sub(r'\bповышьте\b', 'повысьте', text, flags=re.IGNORECASE)
|
364 |
|
365 |
text = re.sub(r'\bСбербизнес\b', 'СберБизнес', text, flags=re.IGNORECASE)
|
|
|
|
|
366 |
text = re.sub(r'\bСбербанк\b', 'СберБанк', text, flags=re.IGNORECASE)
|
367 |
text = re.sub(r'\bвашего ООО\b', 'вашей компании', text, flags=re.IGNORECASE)
|
368 |
-
|
|
|
|
|
369 |
# Step 11: Replace all forms of "рублей", "рубля", "руб." with "р"
|
370 |
# Используем два отдельных регулярных выражения для точности
|
371 |
# 1. Заменяем "руб." на "р", учитывая, что "руб." может быть перед символом "/" или другим несловесным символом
|
@@ -377,7 +353,8 @@ def correct_dash_usage(text):
|
|
377 |
text = re.sub(r'(\d+)\s+тысяч(?:а|и)?(?:\s+рублей)?', r'\1 000 р', text, flags=re.IGNORECASE)
|
378 |
text = re.sub(r'(\d+)\s*тыс\.\s*руб\.', r'\1 000 р', text, flags=re.IGNORECASE)
|
379 |
text = re.sub(r'(\d+)\s*тыс\.\s*р\.', r'\1 000 р', text, flags=re.IGNORECASE)
|
380 |
-
|
|
|
381 |
# Replace millions with "млн"
|
382 |
text = re.sub(r'(\d+)\s+миллиона\b|\bмиллионов\b', r'\1 млн', text, flags=re.IGNORECASE)
|
383 |
text = re.sub(r'(\d+)\s*млн\s*руб\.', r'\1 млн р', text, flags=re.IGNORECASE)
|
@@ -500,41 +477,12 @@ def notify_failed_checks(checks):
|
|
500 |
gr.Warning("ВСЕ ПРОВЕРКИ ПРОЙДЕНЫ")
|
501 |
|
502 |
# Модифицированная функция перегенерации сообщений с уведомлениями о номере попытки
|
503 |
-
def generate_message_gigachat_pro_with_retry(prompt, current_prefix, description, key_message):
|
504 |
-
global approach_stats
|
505 |
-
last_message = None
|
506 |
-
for attempt in range(30):
|
507 |
-
gr.Info(f"Итерация {attempt + 1}: генерируется сообщение...")
|
508 |
-
message = generate_message_gigachat_pro(prompt)
|
509 |
-
message = replace_k_with_thousands(message)
|
510 |
-
message = correct_dash_usage(message)
|
511 |
-
message_length = len(message)
|
512 |
-
if not notify_failed_length(message_length):
|
513 |
-
last_message = message
|
514 |
-
time.sleep(1)
|
515 |
-
continue
|
516 |
-
checks = perform_checks(message, description, key_message)
|
517 |
-
last_message = message
|
518 |
-
approach_stats[current_prefix]["total_attempts"] += 1
|
519 |
-
for check_name, passed in checks.items():
|
520 |
-
if passed is False:
|
521 |
-
approach_stats[current_prefix]["failed_checks"][check_name] += 1
|
522 |
-
break
|
523 |
-
notify_failed_checks(checks) # Вызываем функцию независимо от результата проверок
|
524 |
-
if all(checks.values()):
|
525 |
-
return message
|
526 |
-
prompt = append_errors_to_prompt(prompt, checks)
|
527 |
-
time.sleep(1)
|
528 |
-
gr.Info("Не удалось сгенерировать сообщение, соответствующее требованиям, за 20 итераций. Возвращаем последнее сгенерированное сообщение.")
|
529 |
-
return last_message
|
530 |
-
|
531 |
-
|
532 |
-
#def generate_message_mistral_with_retry(prompt, current_prefix, description, key_message):
|
533 |
# global approach_stats
|
534 |
# last_message = None
|
535 |
-
# for attempt in range(
|
536 |
# gr.Info(f"Итер��ция {attempt + 1}: генерируется сообщение...")
|
537 |
-
# message =
|
538 |
# message = replace_k_with_thousands(message)
|
539 |
# message = correct_dash_usage(message)
|
540 |
# message_length = len(message)
|
@@ -558,6 +506,35 @@ def generate_message_gigachat_pro_with_retry(prompt, current_prefix, description
|
|
558 |
# return last_message
|
559 |
|
560 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
561 |
# Функция для создания задания для копирайтера
|
562 |
def generate_standard_prompt(description, advantages, key_message, *selected_values):
|
563 |
prompt = (
|
@@ -695,7 +672,7 @@ def generate_all_messages(desc, benefits, key_message, gender, generation, psych
|
|
695 |
yield standard_prompt, display_personalization_prompt, None, None, None, None, None, None
|
696 |
flag += 1
|
697 |
prompt = add_prefix_suffix(standard_prompt, prefixes[i], suffixes[i])
|
698 |
-
non_personalized_message =
|
699 |
non_personalized_length = len(non_personalized_message)
|
700 |
non_personalized_display = f"{non_personalized_message}\n------\nКоличество знаков: {non_personalized_length}"
|
701 |
non_personalized_messages.append(non_personalized_display)
|
@@ -713,7 +690,7 @@ def generate_all_messages(desc, benefits, key_message, gender, generation, psych
|
|
713 |
|
714 |
# Генерация персонализированного сообщения
|
715 |
full_personalized_prompt = f"{personalization_prompt}\n\nТекст для адаптации: {non_personalized_message}"
|
716 |
-
personalized_message =
|
717 |
personalized_length = len(personalized_message)
|
718 |
personalized_display = f"{personalized_message}\n------\nКоличество знаков: {personalized_length}"
|
719 |
personalized_messages.append(personalized_display)
|
@@ -739,7 +716,7 @@ def generate_all_messages(desc, benefits, key_message, gender, generation, psych
|
|
739 |
# 1. Запрещенные слова
|
740 |
|
741 |
def check_forbidden_words(message):
|
742 |
-
morph =
|
743 |
|
744 |
# Перечень запрещённых слов и фраз
|
745 |
forbidden_patterns = [
|
@@ -773,7 +750,7 @@ def check_forbidden_words(message):
|
|
773 |
# 2 и #3. Обращение к клиенту и приветствие клиента
|
774 |
|
775 |
def check_no_greeting(message):
|
776 |
-
morph =
|
777 |
# Список типичных обращений и приветствий
|
778 |
greeting_patterns = [
|
779 |
r"привет\b", r"здравствуй", r"добрый\s(день|вечер|утро)",
|
@@ -793,7 +770,7 @@ def check_no_greeting(message):
|
|
793 |
# 4. Обещания и гарантии
|
794 |
|
795 |
def check_no_promises(message):
|
796 |
-
morph =
|
797 |
promise_patterns = [
|
798 |
"обещать", "обещание", "гарантировать", "обязаться", "обязать", "обязательство", "обязательный"
|
799 |
]
|
@@ -809,7 +786,7 @@ def check_no_promises(message):
|
|
809 |
# 5. Составные конструкции из двух глаголов
|
810 |
|
811 |
def check_no_double_verbs(message):
|
812 |
-
morph =
|
813 |
# Разделяем текст по пробелам и знакам препинания
|
814 |
words = re.split(r'\s+|[.!?]', message)
|
815 |
morphs = [morph.parse(word)[0] for word in words]
|
@@ -827,7 +804,7 @@ def check_no_double_verbs(message):
|
|
827 |
# 6. Причастия и причастные обороты
|
828 |
|
829 |
def check_no_participles(message):
|
830 |
-
morph =
|
831 |
words = message.split()
|
832 |
exceptions = {"повышенный", "увеличенный", "пониженный", "сниженный"}
|
833 |
|
@@ -841,7 +818,7 @@ def check_no_participles(message):
|
|
841 |
# 7. Деепричастия и деепричастные обороты
|
842 |
|
843 |
def check_no_adverbial_participles(message):
|
844 |
-
morph =
|
845 |
words = message.split()
|
846 |
morphs = [morph.parse(word)[0] for word in words]
|
847 |
|
@@ -853,7 +830,7 @@ def check_no_adverbial_participles(message):
|
|
853 |
# 8. Превосходная степень прилагательных
|
854 |
|
855 |
def check_no_superlative_adjectives(message):
|
856 |
-
morph =
|
857 |
words = message.split()
|
858 |
morphs = [morph.parse(word)[0] for word in words]
|
859 |
|
@@ -865,7 +842,7 @@ def check_no_superlative_adjectives(message):
|
|
865 |
# 9. Страдательный залог
|
866 |
|
867 |
def check_no_passive_voice(message):
|
868 |
-
morph =
|
869 |
words = message.split()
|
870 |
morphs = [morph.parse(word)[0] for word in words]
|
871 |
|
@@ -877,7 +854,7 @@ def check_no_passive_voice(message):
|
|
877 |
# 10. Порядковые числительные от 10 прописью
|
878 |
|
879 |
def check_no_written_out_ordinals(message):
|
880 |
-
morph =
|
881 |
ordinal_words = [
|
882 |
"десятый", "одиннадцатый", "двенадцатый", "тринадцатый", "четырнадцатый", "пятнадцатый",
|
883 |
"шестнадцатый", "семнадцатый", "восемнадцатый", "девятнадцатый", "двадцатый"
|
@@ -966,7 +943,7 @@ def check_no_time_parasites(message):
|
|
966 |
def check_no_multiple_nouns(message):
|
967 |
noun_count = 0
|
968 |
words = re.split(r'\s+|[.!?]', message) # Разбиваем по пробелам и знакам препинания
|
969 |
-
morph =
|
970 |
|
971 |
for word in words:
|
972 |
parsed_word = morph.parse(word)[0]
|
@@ -1036,7 +1013,7 @@ def check_no_dates_written_out(message):
|
|
1036 |
# Доп правило. Повторы слов
|
1037 |
|
1038 |
def check_no_word_repetitions(message):
|
1039 |
-
morph =
|
1040 |
|
1041 |
# Список союзов и предлогов, которые мы будем игнорировать
|
1042 |
ignore_words = set([
|
@@ -1135,9 +1112,9 @@ def check_disconnected_sentences(message):
|
|
1135 |
если таких предложений **нет**, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1136 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь **только** в формате JSON с закрывающими кавычками и скобками.**'''
|
1137 |
|
1138 |
-
response =
|
1139 |
time.sleep(3) # Задержка в 3 секунды между запросами
|
1140 |
-
print("
|
1141 |
result = parse_json_response(response)
|
1142 |
if result is not None:
|
1143 |
decision = result.get("decision", False)
|
@@ -1162,9 +1139,9 @@ def check_synonymous_members(message):
|
|
1162 |
если таких слов или выражений нет, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1163 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1164 |
|
1165 |
-
response =
|
1166 |
time.sleep(3)
|
1167 |
-
print("
|
1168 |
result = parse_json_response(response)
|
1169 |
if result is not None:
|
1170 |
decision = result.get("decision", False)
|
@@ -1195,9 +1172,9 @@ def check_clickbait_phrases(message):
|
|
1195 |
|
1196 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1197 |
|
1198 |
-
response =
|
1199 |
time.sleep(3)
|
1200 |
-
print("
|
1201 |
result = parse_json_response(response)
|
1202 |
if result is not None:
|
1203 |
decision = result.get("decision", False)
|
@@ -1241,9 +1218,9 @@ def check_abstract_claims(message):
|
|
1241 |
|
1242 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1243 |
|
1244 |
-
response =
|
1245 |
time.sleep(3)
|
1246 |
-
print("
|
1247 |
result = parse_json_response(response)
|
1248 |
if result is not None:
|
1249 |
decision = result.get("decision", False)
|
@@ -1278,9 +1255,9 @@ def check_specialized_terms(message):
|
|
1278 |
|
1279 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1280 |
|
1281 |
-
response =
|
1282 |
time.sleep(3)
|
1283 |
-
print("
|
1284 |
result = parse_json_response(response)
|
1285 |
if result is not None:
|
1286 |
decision = result.get("decision", False)
|
@@ -1307,9 +1284,9 @@ def check_offensive_phrases(message):
|
|
1307 |
если таких фраз нет, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1308 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1309 |
|
1310 |
-
response =
|
1311 |
time.sleep(3)
|
1312 |
-
print("
|
1313 |
result = parse_json_response(response)
|
1314 |
if result is not None:
|
1315 |
decision = result.get("decision", False)
|
@@ -1350,9 +1327,9 @@ def check_cliches_and_bureaucratese(message):
|
|
1350 |
если в тексте **есть** такие выражения, **верни только** JSON {{"decision": true, "explanation": "<пояснение>"}}.
|
1351 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1352 |
|
1353 |
-
response =
|
1354 |
time.sleep(3)
|
1355 |
-
print("
|
1356 |
result = parse_json_response(response)
|
1357 |
if result is not None:
|
1358 |
decision = result.get("decision", False)
|
@@ -1377,9 +1354,9 @@ def check_no_contradictions(message, description):
|
|
1377 |
Если сообщение содержит факты, которые отсутствуют в описании предложения, **верни только** JSON {{"decision": true, "explanation": "<описание противоречий>"}}.
|
1378 |
**Не добавляй никакого дополнительного текста. Отвечай только в формате JSON с закрывающими кавычками и скобками.**'''
|
1379 |
|
1380 |
-
response =
|
1381 |
time.sleep(3)
|
1382 |
-
print("
|
1383 |
result = parse_json_response(response)
|
1384 |
if result is not None:
|
1385 |
decision = result.get("decision", False)
|
@@ -1404,9 +1381,9 @@ def check_contains_key_message(message, key_message):
|
|
1404 |
Если сообщение **не содержит всю** информацию из ключевого текста, **верни только** JSON {{"decision": true, "explanation": "Ключевое текст отсутствует."}}.
|
1405 |
**Не добавляй никакого дополнительного текста. Отвечай только в формате JSON с закрывающими кавычками и скобками.**'''
|
1406 |
|
1407 |
-
response =
|
1408 |
time.sleep(3)
|
1409 |
-
print("
|
1410 |
result = parse_json_response(response)
|
1411 |
if result is not None:
|
1412 |
decision = result.get("decision", False)
|
@@ -1429,7 +1406,7 @@ def safe_check(func, *args):
|
|
1429 |
def perform_checks(message, description, key_message):
|
1430 |
checks = {}
|
1431 |
|
1432 |
-
# 2. Morphological checks using
|
1433 |
morphological_checks = [
|
1434 |
("forbidden_words", check_forbidden_words),
|
1435 |
("client_addressing", check_no_greeting),
|
@@ -1542,74 +1519,6 @@ def format_checks(checks):
|
|
1542 |
return " \n".join(formatted_results)
|
1543 |
|
1544 |
|
1545 |
-
# Функция для обработки нажатия кнопки "Проверить"
|
1546 |
-
def perform_all_checks_and_show_results(personalized_message_1, personalized_message_2, personalized_message_3):
|
1547 |
-
# Моментально показываем все персонализированные сообщения
|
1548 |
-
yield (
|
1549 |
-
personalized_message_1, None, # Первое сообщение без проверки
|
1550 |
-
personalized_message_2, None, # Второе сообщение без проверки
|
1551 |
-
personalized_message_3, None, # Третье сообщение без проверки
|
1552 |
-
None, None, None # Пустые графики для спидометров
|
1553 |
-
)
|
1554 |
-
|
1555 |
-
# Выполняем и показываем проверки с задержкой 1 секунда
|
1556 |
-
checks_1 = perform_checks(personalized_message_1)
|
1557 |
-
formatted_checks_1 = format_checks(checks_1)
|
1558 |
-
time.sleep(1) # Задержка 1 секунда перед выводом первого результата проверки
|
1559 |
-
yield (
|
1560 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1561 |
-
personalized_message_2, None, # Второе сообщение без проверки
|
1562 |
-
personalized_message_3, None, # Третье сообщение без проверки
|
1563 |
-
None, None, None # Пустые графики для спидометров
|
1564 |
-
)
|
1565 |
-
|
1566 |
-
checks_2 = perform_checks(personalized_message_2)
|
1567 |
-
formatted_checks_2 = format_checks(checks_2)
|
1568 |
-
time.sleep(1) # Задержка 1 секунда перед выводом второго результата проверки
|
1569 |
-
yield (
|
1570 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1571 |
-
personalized_message_2, formatted_checks_2, # Проверка для второго сообщения
|
1572 |
-
personalized_message_3, None, # Третье сообщение без проверки
|
1573 |
-
None, None, None # Пустые графики для спидометров
|
1574 |
-
)
|
1575 |
-
|
1576 |
-
checks_3 = perform_checks(personalized_message_3)
|
1577 |
-
formatted_checks_3 = format_checks(checks_3)
|
1578 |
-
time.sleep(1) # Задержка 1 секунда перед выводом третьего результата проверки
|
1579 |
-
yield (
|
1580 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1581 |
-
personalized_message_2, formatted_checks_2, # Проверка для второго сообщения
|
1582 |
-
personalized_message_3, formatted_checks_3, # Проверка для третьего сообщения
|
1583 |
-
None, None, None # Пустые графики для спидометров
|
1584 |
-
)
|
1585 |
-
|
1586 |
-
# Генерация и показ графиков спидометров с задержкой 2 секунды
|
1587 |
-
time.sleep(2)
|
1588 |
-
gauges = generate_random_gauges()
|
1589 |
-
yield (
|
1590 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1591 |
-
personalized_message_2, formatted_checks_2, # Проверка для второго сообщения
|
1592 |
-
personalized_message_3, formatted_checks_3, # Проверка для третьего сообщения
|
1593 |
-
gauges[0], None, None # Первый график спидометра
|
1594 |
-
)
|
1595 |
-
|
1596 |
-
time.sleep(2)
|
1597 |
-
yield (
|
1598 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1599 |
-
personalized_message_2, formatted_checks_2, # Проверка для второго сообщения
|
1600 |
-
personalized_message_3, formatted_checks_3, # Проверка для третьего сообщения
|
1601 |
-
gauges[0], gauges[1], None # Первый и второй графики спидометра
|
1602 |
-
)
|
1603 |
-
|
1604 |
-
time.sleep(2)
|
1605 |
-
yield (
|
1606 |
-
personalized_message_1, formatted_checks_1, # Проверка для первого сообщения
|
1607 |
-
personalized_message_2, formatted_checks_2, # Проверка для второго сообщения
|
1608 |
-
personalized_message_3, formatted_checks_3, # Проверка для третьего сообщения
|
1609 |
-
gauges[0], gauges[1], gauges[2] # Все три графика спидометра
|
1610 |
-
)
|
1611 |
-
|
1612 |
-
|
1613 |
# Интерфейс Gradio
|
1614 |
with gr.Blocks() as demo:
|
1615 |
# Твой интерфейс
|
@@ -1681,11 +1590,6 @@ with gr.Blocks() as demo:
|
|
1681 |
non_personalized_messages = gr.Textbox(label="Стандартные сообщения", lines=12, interactive=False)
|
1682 |
personalized_messages = gr.Textbox(label="Персонализированные сообщения", lines=12, interactive=False)
|
1683 |
|
1684 |
-
# Четвертый ряд
|
1685 |
-
with gr.Row():
|
1686 |
-
btn_check = gr.Button("Проверить", elem_id="check3")
|
1687 |
-
btn_check.click(fn=change_tab, inputs=[gr.Number(value=3, visible=False)], outputs=tabs)
|
1688 |
-
|
1689 |
# Сначала переключаем вкладку, потом запускаем генерацию сообщений
|
1690 |
btn_to_prompts.click(
|
1691 |
fn=change_tab,
|
|
|
8 |
from openpyxl import load_workbook
|
9 |
import plotly.graph_objects as go
|
10 |
import random
|
11 |
+
import pymorphy3
|
12 |
import string
|
13 |
import json
|
14 |
from mistralai import Mistral
|
|
|
17 |
import base64
|
18 |
import io
|
19 |
|
20 |
+
MISTRAL_API_KEY = os.getenv('MISTRAL_API_KEY')
|
21 |
token = os.getenv('GITHUB_TOKEN')
|
22 |
|
23 |
# Клиент для генерации сообщений
|
24 |
+
client_mistral_generate = Mistral(api_key=MISTRAL_API_KEY)
|
25 |
|
26 |
# Клиент для выполнения проверок
|
27 |
+
client_mistral_check = Mistral(api_key=MISTRAL_API_KEY)
|
28 |
|
29 |
+
morph = pymorphy3.MorphAnalyzer()
|
30 |
|
31 |
# Авторизация в GigaChat Pro
|
32 |
+
#gc_key = os.getenv('GC_KEY')
|
33 |
#chat_pro = GigaChat(credentials=gc_key, model='GigaChat', max_tokens=68, temperature=1.15, verify_ssl_certs=False)
|
34 |
+
#chat_pro = GigaChat(
|
35 |
+
# credentials=gc_key,
|
36 |
+
# model='GigaChat-Pro-preview',
|
37 |
+
# base_url='https://gigachat-preview.devices.sberbank.ru/api/v1/',
|
38 |
+
# max_tokens=68,
|
39 |
+
# temperature=1.15,
|
40 |
+
# verify_ssl_certs=False
|
41 |
+
#)
|
42 |
+
|
43 |
+
#chat_pro_check = GigaChat(
|
44 |
+
# credentials=gc_key,
|
45 |
+
# model='GigaChat-Pro-preview',
|
46 |
+
# base_url='https://gigachat-preview.devices.sberbank.ru/api/v1/',
|
47 |
+
# max_tokens=3000,
|
48 |
+
# temperature=0.8,
|
49 |
+
# verify_ssl_certs=False
|
50 |
+
#)
|
51 |
|
52 |
approach_stats = {
|
53 |
"Начни сообщение с призыва к действию с продуктом.": {"failed_checks": defaultdict(int), "total_attempts": 0},
|
|
|
74 |
print(f"Ошибка при обработке данных листа {sheet_name}: {e}")
|
75 |
features[sheet_name] = {}
|
76 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
77 |
def save_statistics_to_github(approach_stats):
|
78 |
repo = "fruitpicker01/Storage_dev"
|
79 |
timestamp = int(time.time())
|
|
|
146 |
else:
|
147 |
print(f"Ошибка при сохранении CSV-файла: {response.status_code} {response.text}")
|
148 |
|
|
|
|
|
|
|
|
|
149 |
# Функция для смены вкладки
|
150 |
def change_tab(id):
|
151 |
return gr.Tabs(selected=id)
|
|
|
163 |
return message
|
164 |
|
165 |
# Функция для генерации сообщения с GigaChat Pro
|
166 |
+
#def generate_message_gigachat_pro(prompt):
|
167 |
+
# try:
|
168 |
+
# messages = [SystemMessage(content=prompt)]
|
169 |
+
# res = chat_pro(messages)
|
170 |
+
# cleaned_message = clean_message(res.content.strip())
|
171 |
+
# return cleaned_message
|
172 |
+
# except Exception as e:
|
173 |
+
# return f"Ошибка при обращении к GigaChat-Pro: {e}"
|
174 |
+
|
175 |
+
def generate_message_mistral_generate(prompt, max_retries=5):
|
176 |
+
retries = 0
|
177 |
+
while retries < max_retries:
|
178 |
+
try:
|
179 |
+
chat_response = client_mistral_generate.chat.complete(
|
180 |
+
model="mistral-large-latest",
|
181 |
+
messages=[
|
182 |
+
{
|
183 |
+
"role": "user",
|
184 |
+
"content": prompt,
|
185 |
+
"max_tokens": 68,
|
186 |
+
"temperature": 0.8
|
187 |
+
},
|
188 |
+
]
|
189 |
+
)
|
190 |
+
cleaned_message = clean_message(chat_response.choices[0].message.content.strip())
|
191 |
+
return cleaned_message
|
192 |
+
except Exception as e:
|
193 |
+
if "Status 429" in str(e):
|
194 |
+
wait_time = 3 # Можно установить фиксированную задержку
|
195 |
+
print(f"Превышен лимит запросов. Ожидание {wait_time} секунд перед повторной попыткой...")
|
196 |
+
time.sleep(wait_time)
|
197 |
+
retries += 1
|
198 |
+
else:
|
199 |
+
print(f"Ошибка при обращении к Mistral: {e}")
|
200 |
+
return None
|
201 |
+
|
202 |
+
def generate_message_mistral_check(prompt, max_retries=5):
|
203 |
+
retries = 0
|
204 |
+
while retries < max_retries:
|
205 |
+
try:
|
206 |
+
chat_response = client_mistral_check.chat.complete(
|
207 |
+
model="mistral-large-latest",
|
208 |
+
messages=[
|
209 |
+
{
|
210 |
+
"role": "user",
|
211 |
+
"content": prompt,
|
212 |
+
"max_tokens": 3000,
|
213 |
+
"temperature": 0.2
|
214 |
+
},
|
215 |
+
]
|
216 |
+
)
|
217 |
+
cleaned_message = clean_message(chat_response.choices[0].message.content.strip())
|
218 |
+
return cleaned_message
|
219 |
+
except Exception as e:
|
220 |
+
if "Status 429" in str(e):
|
221 |
+
wait_time = 3 # Можно установить фиксированную задержку
|
222 |
+
print(f"Превышен лимит запросов. Ожидание {wait_time} секунд перед повторной попыткой...")
|
223 |
+
time.sleep(wait_time)
|
224 |
+
retries += 1
|
225 |
+
else:
|
226 |
+
print(f"Ошибка при обращении к Mistral: {e}")
|
227 |
+
return None
|
228 |
+
|
229 |
+
#def generate_check_gigachat_pro(prompt):
|
230 |
+
# try:
|
231 |
+
# messages = [SystemMessage(content=prompt)]
|
232 |
+
# res2 = chat_pro_check(messages)
|
233 |
+
# cleaned_message = clean_message(res2.content.strip())
|
234 |
+
# return cleaned_message
|
235 |
+
# except Exception as e:
|
236 |
+
# return f"Ошибка при обращении к GigaChat-Pro: {e}"
|
237 |
|
238 |
# Функция для замены сокращений с 'k' или 'К' на тысячи
|
239 |
def replace_k_with_thousands(message):
|
|
|
242 |
return message
|
243 |
|
244 |
def correct_dash_usage(text):
|
245 |
+
morph = pymorphy3.MorphAnalyzer()
|
246 |
# Step 1: Replace any dash with long dash if surrounded by spaces
|
247 |
text = re.sub(r'\s[-–—]\s', ' — ', text)
|
248 |
|
|
|
295 |
text = ' '.join(sentences)
|
296 |
|
297 |
def restore_yo(text):
|
298 |
+
morph = pymorphy3.MorphAnalyzer()
|
299 |
words = text.split()
|
300 |
restored_words = []
|
301 |
|
|
|
335 |
text = re.sub(r'\bповышьте\b', 'повысьте', text, flags=re.IGNORECASE)
|
336 |
|
337 |
text = re.sub(r'\bСбербизнес\b', 'СберБизнес', text, flags=re.IGNORECASE)
|
338 |
+
text = re.sub(r'\bСбербизнеса\b', 'СберБизнес', text, flags=re.IGNORECASE)
|
339 |
+
text = re.sub(r'\bСбербизнесе\b', 'СберБизнес', text, flags=re.IGNORECASE)
|
340 |
text = re.sub(r'\bСбербанк\b', 'СберБанк', text, flags=re.IGNORECASE)
|
341 |
text = re.sub(r'\bвашего ООО\b', 'вашей компании', text, flags=re.IGNORECASE)
|
342 |
+
text = re.sub(r'\b0₽\b', '0 р', text, flags=re.IGNORECASE)
|
343 |
+
text = re.sub(r'\b₽\b', 'р', text, flags=re.IGNORECASE)
|
344 |
+
|
345 |
# Step 11: Replace all forms of "рублей", "рубля", "руб." with "р"
|
346 |
# Используем два отдельных регулярных выражения для точности
|
347 |
# 1. Заменяем "руб." на "р", учитывая, что "руб." может быть перед символом "/" или другим несловесным символом
|
|
|
353 |
text = re.sub(r'(\d+)\s+тысяч(?:а|и)?(?:\s+рублей)?', r'\1 000 р', text, flags=re.IGNORECASE)
|
354 |
text = re.sub(r'(\d+)\s*тыс\.\s*руб\.', r'\1 000 р', text, flags=re.IGNORECASE)
|
355 |
text = re.sub(r'(\d+)\s*тыс\.\s*р\.', r'\1 000 р', text, flags=re.IGNORECASE)
|
356 |
+
text = re.sub(r'(\d+)\s*тыс\.\s*р', r'\1 000 р', text, flags=re.IGNORECASE)
|
357 |
+
|
358 |
# Replace millions with "млн"
|
359 |
text = re.sub(r'(\d+)\s+миллиона\b|\bмиллионов\b', r'\1 млн', text, flags=re.IGNORECASE)
|
360 |
text = re.sub(r'(\d+)\s*млн\s*руб\.', r'\1 млн р', text, flags=re.IGNORECASE)
|
|
|
477 |
gr.Warning("ВСЕ ПРОВЕРКИ ПРОЙДЕНЫ")
|
478 |
|
479 |
# Модифицированная функция перегенерации сообщений с уведомлениями о номере попытки
|
480 |
+
#def generate_message_gigachat_pro_with_retry(prompt, current_prefix, description, key_message):
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
481 |
# global approach_stats
|
482 |
# last_message = None
|
483 |
+
# for attempt in range(30):
|
484 |
# gr.Info(f"Итер��ция {attempt + 1}: генерируется сообщение...")
|
485 |
+
# message = generate_message_gigachat_pro(prompt)
|
486 |
# message = replace_k_with_thousands(message)
|
487 |
# message = correct_dash_usage(message)
|
488 |
# message_length = len(message)
|
|
|
506 |
# return last_message
|
507 |
|
508 |
|
509 |
+
def generate_message_mistral_with_retry(prompt, current_prefix, description, key_message):
|
510 |
+
global approach_stats
|
511 |
+
last_message = None
|
512 |
+
for attempt in range(20):
|
513 |
+
gr.Info(f"Итерация {attempt + 1}: генерируется сообщение...")
|
514 |
+
message = generate_message_mistral_generate(prompt)
|
515 |
+
message = replace_k_with_thousands(message)
|
516 |
+
message = correct_dash_usage(message)
|
517 |
+
message_length = len(message)
|
518 |
+
if not notify_failed_length(message_length):
|
519 |
+
last_message = message
|
520 |
+
time.sleep(1)
|
521 |
+
continue
|
522 |
+
checks = perform_checks(message, description, key_message)
|
523 |
+
last_message = message
|
524 |
+
approach_stats[current_prefix]["total_attempts"] += 1
|
525 |
+
for check_name, passed in checks.items():
|
526 |
+
if passed is False:
|
527 |
+
approach_stats[current_prefix]["failed_checks"][check_name] += 1
|
528 |
+
break
|
529 |
+
notify_failed_checks(checks) # Вызываем функцию независимо от результата проверок
|
530 |
+
if all(checks.values()):
|
531 |
+
return message
|
532 |
+
prompt = append_errors_to_prompt(prompt, checks)
|
533 |
+
time.sleep(1)
|
534 |
+
gr.Info("Не удалось сгенерировать сообщение, соответствующее требованиям, за 20 итераций. Возвращаем последнее сгенерированное сообщение.")
|
535 |
+
return last_message
|
536 |
+
|
537 |
+
|
538 |
# Функция для создания задания для копирайтера
|
539 |
def generate_standard_prompt(description, advantages, key_message, *selected_values):
|
540 |
prompt = (
|
|
|
672 |
yield standard_prompt, display_personalization_prompt, None, None, None, None, None, None
|
673 |
flag += 1
|
674 |
prompt = add_prefix_suffix(standard_prompt, prefixes[i], suffixes[i])
|
675 |
+
non_personalized_message = generate_message_mistral_with_retry(prompt, current_prefix, desc, key_message)
|
676 |
non_personalized_length = len(non_personalized_message)
|
677 |
non_personalized_display = f"{non_personalized_message}\n------\nКоличество знаков: {non_personalized_length}"
|
678 |
non_personalized_messages.append(non_personalized_display)
|
|
|
690 |
|
691 |
# Генерация персонализированного сообщения
|
692 |
full_personalized_prompt = f"{personalization_prompt}\n\nТекст для адаптации: {non_personalized_message}"
|
693 |
+
personalized_message = generate_message_mistral_with_retry(full_personalized_prompt, current_prefix, desc, key_message)
|
694 |
personalized_length = len(personalized_message)
|
695 |
personalized_display = f"{personalized_message}\n------\nКоличество знаков: {personalized_length}"
|
696 |
personalized_messages.append(personalized_display)
|
|
|
716 |
# 1. Запрещенные слова
|
717 |
|
718 |
def check_forbidden_words(message):
|
719 |
+
morph = pymorphy3.MorphAnalyzer()
|
720 |
|
721 |
# Перечень запрещённых слов и фраз
|
722 |
forbidden_patterns = [
|
|
|
750 |
# 2 и #3. Обращение к клиенту и приветствие клиента
|
751 |
|
752 |
def check_no_greeting(message):
|
753 |
+
morph = pymorphy3.MorphAnalyzer()
|
754 |
# Список типичных обращений и приветствий
|
755 |
greeting_patterns = [
|
756 |
r"привет\b", r"здравствуй", r"добрый\s(день|вечер|утро)",
|
|
|
770 |
# 4. Обещания и гарантии
|
771 |
|
772 |
def check_no_promises(message):
|
773 |
+
morph = pymorphy3.MorphAnalyzer()
|
774 |
promise_patterns = [
|
775 |
"обещать", "обещание", "гарантировать", "обязаться", "обязать", "обязательство", "обязательный"
|
776 |
]
|
|
|
786 |
# 5. Составные конструкции из двух глаголов
|
787 |
|
788 |
def check_no_double_verbs(message):
|
789 |
+
morph = pymorphy3.MorphAnalyzer()
|
790 |
# Разделяем текст по пробелам и знакам препинания
|
791 |
words = re.split(r'\s+|[.!?]', message)
|
792 |
morphs = [morph.parse(word)[0] for word in words]
|
|
|
804 |
# 6. Причастия и причастные обороты
|
805 |
|
806 |
def check_no_participles(message):
|
807 |
+
morph = pymorphy3.MorphAnalyzer()
|
808 |
words = message.split()
|
809 |
exceptions = {"повышенный", "увеличенный", "пониженный", "сниженный"}
|
810 |
|
|
|
818 |
# 7. Деепричастия и деепричастные обороты
|
819 |
|
820 |
def check_no_adverbial_participles(message):
|
821 |
+
morph = pymorphy3.MorphAnalyzer()
|
822 |
words = message.split()
|
823 |
morphs = [morph.parse(word)[0] for word in words]
|
824 |
|
|
|
830 |
# 8. Превосходная степень прилагательных
|
831 |
|
832 |
def check_no_superlative_adjectives(message):
|
833 |
+
morph = pymorphy3.MorphAnalyzer()
|
834 |
words = message.split()
|
835 |
morphs = [morph.parse(word)[0] for word in words]
|
836 |
|
|
|
842 |
# 9. Страдательный залог
|
843 |
|
844 |
def check_no_passive_voice(message):
|
845 |
+
morph = pymorphy3.MorphAnalyzer()
|
846 |
words = message.split()
|
847 |
morphs = [morph.parse(word)[0] for word in words]
|
848 |
|
|
|
854 |
# 10. Порядковые числительные от 10 прописью
|
855 |
|
856 |
def check_no_written_out_ordinals(message):
|
857 |
+
morph = pymorphy3.MorphAnalyzer()
|
858 |
ordinal_words = [
|
859 |
"десятый", "одиннадцатый", "двенадцатый", "тринадцатый", "четырнадцатый", "пятнадцатый",
|
860 |
"шестнадцатый", "семнадцатый", "восемнадцатый", "девятнадцатый", "двадцатый"
|
|
|
943 |
def check_no_multiple_nouns(message):
|
944 |
noun_count = 0
|
945 |
words = re.split(r'\s+|[.!?]', message) # Разбиваем по пробелам и знакам препинания
|
946 |
+
morph = pymorphy3.MorphAnalyzer()
|
947 |
|
948 |
for word in words:
|
949 |
parsed_word = morph.parse(word)[0]
|
|
|
1013 |
# Доп правило. Повторы слов
|
1014 |
|
1015 |
def check_no_word_repetitions(message):
|
1016 |
+
morph = pymorphy3.MorphAnalyzer()
|
1017 |
|
1018 |
# Список союзов и предлогов, которые мы будем игнорировать
|
1019 |
ignore_words = set([
|
|
|
1112 |
если таких предложений **нет**, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1113 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь **только** в формате JSON с закрывающими кавычками и скобками.**'''
|
1114 |
|
1115 |
+
response = generate_message_mistral_check(prompt)
|
1116 |
time.sleep(3) # Задержка в 3 секунды между запросами
|
1117 |
+
print("Mistral response:", response) # Выводим полный ответ модели
|
1118 |
result = parse_json_response(response)
|
1119 |
if result is not None:
|
1120 |
decision = result.get("decision", False)
|
|
|
1139 |
если таких слов или выражений нет, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1140 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1141 |
|
1142 |
+
response = generate_message_mistral_check(prompt)
|
1143 |
time.sleep(3)
|
1144 |
+
print("Mistral response:", response)
|
1145 |
result = parse_json_response(response)
|
1146 |
if result is not None:
|
1147 |
decision = result.get("decision", False)
|
|
|
1172 |
|
1173 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1174 |
|
1175 |
+
response = generate_message_mistral_check(prompt)
|
1176 |
time.sleep(3)
|
1177 |
+
print("Mistral response:", response)
|
1178 |
result = parse_json_response(response)
|
1179 |
if result is not None:
|
1180 |
decision = result.get("decision", False)
|
|
|
1218 |
|
1219 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1220 |
|
1221 |
+
response = generate_message_mistral_check(prompt)
|
1222 |
time.sleep(3)
|
1223 |
+
print("Mistral response:", response)
|
1224 |
result = parse_json_response(response)
|
1225 |
if result is not None:
|
1226 |
decision = result.get("decision", False)
|
|
|
1255 |
|
1256 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1257 |
|
1258 |
+
response = generate_message_mistral_check(prompt)
|
1259 |
time.sleep(3)
|
1260 |
+
print("Mistral response:", response)
|
1261 |
result = parse_json_response(response)
|
1262 |
if result is not None:
|
1263 |
decision = result.get("decision", False)
|
|
|
1284 |
если таких фраз нет, **верни только** JSON {{"decision": false, "explanation": "<пояснение>"}}.
|
1285 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1286 |
|
1287 |
+
response = generate_message_mistral_check(prompt)
|
1288 |
time.sleep(3)
|
1289 |
+
print("Mistral response:", response)
|
1290 |
result = parse_json_response(response)
|
1291 |
if result is not None:
|
1292 |
decision = result.get("decision", False)
|
|
|
1327 |
если в тексте **есть** такие выражения, **верни только** JSON {{"decision": true, "explanation": "<пояснение>"}}.
|
1328 |
**Не добавляй никакого дополнительного текста. Перед ответом убедись, что отвечаешь только в формате JSON с закрывающими кавычками и скобками.**'''
|
1329 |
|
1330 |
+
response = generate_message_mistral_check(prompt)
|
1331 |
time.sleep(3)
|
1332 |
+
print("Mistral response:", response)
|
1333 |
result = parse_json_response(response)
|
1334 |
if result is not None:
|
1335 |
decision = result.get("decision", False)
|
|
|
1354 |
Если сообщение содержит факты, которые отсутствуют в описании предложения, **верни только** JSON {{"decision": true, "explanation": "<описание противоречий>"}}.
|
1355 |
**Не добавляй никакого дополнительного текста. Отвечай только в формате JSON с закрывающими кавычками и скобками.**'''
|
1356 |
|
1357 |
+
response = generate_message_mistral_check(prompt)
|
1358 |
time.sleep(3)
|
1359 |
+
print("Mistral response:", response)
|
1360 |
result = parse_json_response(response)
|
1361 |
if result is not None:
|
1362 |
decision = result.get("decision", False)
|
|
|
1381 |
Если сообщение **не содержит всю** информацию из ключевого текста, **верни только** JSON {{"decision": true, "explanation": "Ключевое текст отсутствует."}}.
|
1382 |
**Не добавляй никакого дополнительного текста. Отвечай только в формате JSON с закрывающими кавычками и скобками.**'''
|
1383 |
|
1384 |
+
response = generate_message_mistral_check(prompt)
|
1385 |
time.sleep(3)
|
1386 |
+
print("Mistral response:", response)
|
1387 |
result = parse_json_response(response)
|
1388 |
if result is not None:
|
1389 |
decision = result.get("decision", False)
|
|
|
1406 |
def perform_checks(message, description, key_message):
|
1407 |
checks = {}
|
1408 |
|
1409 |
+
# 2. Morphological checks using pymorphy3
|
1410 |
morphological_checks = [
|
1411 |
("forbidden_words", check_forbidden_words),
|
1412 |
("client_addressing", check_no_greeting),
|
|
|
1519 |
return " \n".join(formatted_results)
|
1520 |
|
1521 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1522 |
# Интерфейс Gradio
|
1523 |
with gr.Blocks() as demo:
|
1524 |
# Твой интерфейс
|
|
|
1590 |
non_personalized_messages = gr.Textbox(label="Стандартные сообщения", lines=12, interactive=False)
|
1591 |
personalized_messages = gr.Textbox(label="Персонализированные сообщения", lines=12, interactive=False)
|
1592 |
|
|
|
|
|
|
|
|
|
|
|
1593 |
# Сначала переключаем вкладку, потом запускаем генерацию сообщений
|
1594 |
btn_to_prompts.click(
|
1595 |
fn=change_tab,
|