File size: 20,862 Bytes
47c50bf
 
 
 
 
 
 
 
 
 
 
c348b7e
47c50bf
 
 
 
c348b7e
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
47c50bf
 
c348b7e
 
47c50bf
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c348b7e
 
f0c0344
c348b7e
f0c0344
 
c348b7e
f0c0344
c348b7e
 
 
47c50bf
 
 
 
c348b7e
 
 
 
47c50bf
 
c348b7e
f0c0344
 
 
 
 
 
c348b7e
f0c0344
 
 
 
c348b7e
f0c0344
c348b7e
f0c0344
c348b7e
 
f0c0344
c348b7e
f0c0344
 
 
 
c348b7e
 
 
f0c0344
 
c348b7e
 
 
 
f0c0344
c348b7e
f0c0344
c348b7e
f1b96f9
f0c0344
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c348b7e
f0c0344
47c50bf
 
 
263c6bc
47c50bf
 
c348b7e
 
47c50bf
c348b7e
 
 
47c50bf
 
c348b7e
 
 
47c50bf
c348b7e
47c50bf
 
 
 
c348b7e
47c50bf
 
c348b7e
47c50bf
 
 
c348b7e
 
f1b96f9
c348b7e
 
f1b96f9
 
47c50bf
f0c0344
c348b7e
 
 
 
 
 
 
 
 
 
 
 
 
 
f0c0344
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
f1b96f9
 
559e0f6
f0c0344
 
 
 
 
 
 
 
 
 
 
 
f1b96f9
f0c0344
 
 
f1b96f9
f0c0344
f1b96f9
 
f0c0344
 
 
 
 
 
 
 
 
 
 
 
 
f1b96f9
f0c0344
 
 
f1b96f9
f0c0344
f1b96f9
 
f0c0344
 
 
 
 
 
 
 
 
 
 
 
 
f1b96f9
f0c0344
 
 
f1b96f9
f0c0344
f1b96f9
c348b7e
f0c0344
 
 
 
 
 
 
 
 
 
 
 
 
f1b96f9
f0c0344
 
47c50bf
 
263c6bc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
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
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
import gradio as gr
import requests
import os
import json
import pandas as pd
import time
from langchain.schema import SystemMessage
from langchain_community.chat_models.gigachat import GigaChat
from openpyxl import load_workbook
import base64

# Установка ключей API для OpenAI, GigaChat и GitHub
openai_api_key = os.getenv('GPT_KEY')
gc_key = os.getenv('GC_KEY')
token = os.getenv('GITHUB_TOKEN')

# Класс конфигурации модели
class ModelConfig:
    def __init__(self, temperature):
        self.max_tokens = 68  # фиксированное значение
        self.temperature = temperature

# Класс для генерации сообщений
class MessageGenerator:
    def __init__(self, config: ModelConfig):
        self.config = config
        self.chat_pro = self.authenticate_gigachat('GigaChat-Pro')
        self.chat_lite = self.authenticate_gigachat('GigaChat')
        self.chat_plus = self.authenticate_gigachat('GigaChat-Plus')

    def authenticate_gigachat(self, model):
        return GigaChat(credentials=gc_key, model=model, max_tokens=self.config.max_tokens, temperature=self.config.temperature, verify_ssl_certs=False)

    def clean_message(self, message):
        if not message.endswith(('.', '!', '?')):
            last_period = message.rfind('.')
            if last_period != -1:
                message = message[:last_period + 1]
        return message

    def generate_message_gpt4o(self, prompt):
        headers = {
            "Content-Type": "application/json",
            "Authorization": f"Bearer {openai_api_key}"
        }
        data = {
            "model": "chatgpt-4o-latest",
            "messages": [{"role": "system", "content": prompt}],
            "max_tokens": self.config.max_tokens,
            "temperature": self.config.temperature
        }
        response = requests.post("https://api.openai.com/v1/chat/completions", json=data, headers=headers)
        response_data = response.json()
        return self.clean_message(response_data["choices"][0]["message"]["content"].strip())

    def generate_message_gigachat(self, prompt, model):
        messages = [SystemMessage(content=prompt)]
        if model == 'GigaChat-Pro':
            res = self.chat_pro(messages)
        elif model == 'GigaChat':
            res = self.chat_lite(messages)
        elif model == 'GigaChat-Plus':
            res = self.chat_plus(messages)
        return self.clean_message(res.content.strip())

# Загрузка данных из Excel-файла
data = pd.read_excel('Признаки.xlsx', sheet_name=None)
features = {sheet_name: df.set_index(df.columns[0]).to_dict()[df.columns[1]] for sheet_name, df in data.items()}

# Функция для генерации стандартного промпта
def generate_standard_prompt(description, advantages, *selected_values):
    prompt = (
        "Сгенерируй смс-сообщение для клиента.\n"
        "Начни сообщение с призыва к действию с продуктом.\n"
        f"Описание предложения: {description}\n"
        f"Преимущества: {advantages}\n"
        "Вклад на короткий срок.\n"
        "В тексте смс запрещено использование:\n"
        "- Запрещенные слова: № один, номер один, № 1, вкусный, дешёвый, продукт, спам, доступный, банкротство, долги, займ, срочно, сейчас, лучший, главный, номер 1, гарантия, успех, лидер;\n"
        "- Обращение к клиенту;\n"
        "- Приветствие клиента;\n"
        "- Обещания и гарантии;\n"
        "- Использовать составные конструкции из двух глаголов;\n"
        "- Причастия и причастные обороты;\n"
        "- Деепричастия и деепричастные обороты;\n"
        "- Превосходная степень прилагательных;\n"
        "- Страдательный залог;\n"
        "- Порядковые числительные от 10 прописью;\n"
        "- Цепочки с придаточными предложениями;\n"
        "- Разделительные повторяющиеся союзы;\n"
        "- Вводные конструкции;\n"
        "- Усилители;\n"
        "- Паразиты времени;\n"
        "- Несколько существительных подряд, в том числе отглагольных;\n"
        "- Производные предлоги;\n"
        "- Сложные предложения, в которых нет связи между частями;\n"
        "- Сложноподчинённые предложения;\n"
        "- Даты прописью;\n"
        "- Близкие по смыслу однородные члены предложения;\n"
        "- Шокирующие, экстравагантные, кликбейтные фразы;\n"
        "- Абстрактные заявления без поддержки фактами и отсутствие доказательства пользы для клиента;\n"
        "- Гарантирующие фразы;\n"
        "- Узкоспециализированные термины;\n"
        "- Фразы, способные создать двойственное ощущение, обидеть;\n"
        "- Речевые клише, рекламные штампы, канцеляризмы;\n"
        "Убедись, что в готовом тексте до 250 знаков с пробелами."
    )
    return prompt

# Функция для генерации сообщений с учетом повторных попыток
def generate_message_with_retry(generator, prompt, model):
    for _ in range(10):
        message = generator.generate_message_gigachat(prompt, model)
        if len(message) <= 250:
            return message
    return message  # Возвращаем последнее сгенерированное сообщение, если все попытки не удались

# Обновление интерфейса для отображения сообщений
def generate_messages(description, advantages, config, *selected_values):
    generator = MessageGenerator(config)
    standard_prompt = generate_standard_prompt(description, advantages, *selected_values)

    results = {
        "prompt": standard_prompt,
        "gpt4o": generator.generate_message_gpt4o(standard_prompt),
        "gigachat_pro": generate_message_with_retry(generator, standard_prompt, 'GigaChat-Pro'),
        "gigachat_lite": generate_message_with_retry(generator, standard_prompt, 'GigaChat'),
        "gigachat_plus": generate_message_with_retry(generator, standard_prompt, 'GigaChat-Plus')
    }

    return results["prompt"], results["gpt4o"], results["gigachat_pro"], results["gigachat_lite"], results["gigachat_plus"]

# Функция для генерации персонализированного промпта
def generate_personalization_prompt(*selected_values):
    prompt = "Адаптируй, не превышая длину сообщения в 250 знаков с пробелами, текст с учетом следующих особенностей:\n"
    for i, feature in enumerate(features.keys()):
        if selected_values[i]:
            prompt += f"{features[feature][selected_values[i]]}\n"
    prompt += "Убедись, что в готовом тексте до 250 знаков с пробелами."
    return prompt.strip()

# Функция для выполнения персонализации на основе сгенерированного промпта и сообщения
def perform_personalization(standard_message, personalization_prompt, generator):
    full_prompt = f"{personalization_prompt}\n\nТекст для адаптации:\n{standard_message}"
    return generator.generate_message_gpt4o(full_prompt)

# Функция для выполнения персонализации с учетом повторных попыток
def perform_personalization_with_retry(standard_message, personalization_prompt, generator, model):
    for _ in range(10):
        message = perform_personalization(standard_message, personalization_prompt, generator)
        if len(message) <= 250:
            return message
    return message

# Обновление интерфейса для персонализации сообщений
def personalize_messages(gpt4o_message, gigachat_pro_message, gigachat_lite_message, gigachat_plus_message, config, *selected_values):
    generator = MessageGenerator(config)
    personalization_prompt = generate_personalization_prompt(*selected_values)

    personalized_gpt4o = perform_personalization_with_retry(gpt4o_message, personalization_prompt, generator, "GPT-4o")
    personalized_gigachat_pro = perform_personalization_with_retry(gigachat_pro_message, personalization_prompt, generator, "GigaChat-Pro")
    personalized_gigachat_lite = perform_personalization_with_retry(gigachat_lite_message, personalization_prompt, generator, "GigaChat")
    personalized_gigachat_plus = perform_personalization_with_retry(gigachat_plus_message, personalization_prompt, generator, "GigaChat-Plus")

    return personalization_prompt, personalized_gpt4o, personalized_gigachat_pro, personalized_gigachat_lite, personalized_gigachat_plus

# Функция для сохранения данных на GitHub
def save_to_github(personalized_message, model_name, comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form):
    data_to_save = {
        "Модель": model_name,
        "Персонализированное сообщение": personalized_message,
        "Комментарий": comment,
        "Откорректированное сообщение": corrected_message,
        "Описание предложения": description,
        "Преимущества": advantages,
        "Неперсонализированный промпт": non_personalized_prompt,
        "Неперсонализированное сообщение": non_personalized_message,
        "Пол": gender,
        "Поколение": generation,
        "Психотип": psychotype,
        "Стадия бизнеса": business_stage,
        "Отрасль": industry,
        "ОПФ": legal_form
    }
    file_content_encoded = base64.b64encode(json.dumps(data_to_save).encode()).decode()
    repo = "fruitpicker01/Storage_1"
    path = f"file_{int(time.time())}.json"
    url = f"https://api.github.com/repos/{repo}/contents/{path}"
    headers = {
        "Authorization": f"token {token}",
        "Content-Type": "application/json"
    }
    data = {
        "message": f"Добавлен новый файл {path}",
        "content": file_content_encoded
    }
    requests.put(url, headers=headers, data=json.dumps(data))

# Создание интерфейса Gradio
with gr.Blocks() as demo:
    gr.Markdown("# Генерация SMS-сообщений по заданным признакам")
    
    with gr.Row():
        with gr.Column(scale=1):
            temperature_slider = gr.Slider(label="Температура", minimum=0, maximum=2, step=0.01, value=1)

            description_input = gr.Textbox(
                label="Описание предложения", 
                lines=5,
                value="Необходимо предложить клиенту оформить дебетовую премиальную бизнес-карту Mastercard Preffered..."
            )
            advantages_input = gr.Textbox(
                label="Преимущества", 
                lines=5,
                value="Предложение по бесплатному обслуживанию — бессрочное..."
            )

            selections = []
            for feature in features.keys():
                selections.append(gr.Dropdown(choices=[None] + list(features[feature].keys()), label=f"Выберите {feature}"))

            submit_btn = gr.Button("Создать неперсонализированное сообщение")

        with gr.Column(scale=2):
            prompt_display = gr.Textbox(label="Неперсонализированный промпт", lines=10, interactive=False)
            output_text_gpt4o = gr.Textbox(label="Неперсонализированное сообщение GPT-4o", lines=3, interactive=False)
            output_text_gigachat_pro = gr.Textbox(label="Неперсонализированное сообщение GigaChat-Pro", lines=3, interactive=False)
            output_text_gigachat_lite = gr.Textbox(label="Неперсонализированное сообщение GigaChat-Lite", lines=3, interactive=False)
            output_text_gigachat_plus = gr.Textbox(label="Неперсонализированное сообщение GigaChat-Plus", lines=3, interactive=False)

            submit_btn.click(
                fn=generate_messages, 
                inputs=[description_input, advantages_input, gr.State(lambda: ModelConfig(temperature_slider.value))] + selections, 
                outputs=[prompt_display, output_text_gpt4o, output_text_gigachat_pro, output_text_gigachat_lite, output_text_gigachat_plus]
            )

    with gr.Row():
        personalize_btn = gr.Button("Выполнить персонализацию")

        with gr.Column(scale=2):
            personalization_prompt_display = gr.Textbox(label="Промпт для персонализации", lines=6, interactive=False)
            personalized_output_text_gpt4o = gr.Textbox(label="Персонализированное сообщение GPT-4o", lines=6, interactive=False)
            personalized_output_text_gigachat_pro = gr.Textbox(label="Персонализированное сообщение GigaChat-Pro", lines=6, interactive=False)
            personalized_output_text_gigachat_lite = gr.Textbox(label="Персонализированное сообщение GigaChat-Lite", lines=6, interactive=False)
            personalized_output_text_gigachat_plus = gr.Textbox(label="Персонализированное сообщение GigaChat-Plus", lines=6, interactive=False)

            personalize_btn.click(
                fn=personalize_messages,
                inputs=[output_text_gpt4o, output_text_gigachat_pro, output_text_gigachat_lite, output_text_gigachat_plus, gr.State(lambda: ModelConfig(temperature_slider.value))] + selections,
                outputs=[personalization_prompt_display, personalized_output_text_gpt4o, personalized_output_text_gigachat_pro, personalized_output_text_gigachat_lite, personalized_output_text_gigachat_plus]
            )

    with gr.Row():
        gr.Markdown("*Комментарий (опционально):*")
        comment_gpt4o = gr.Textbox(label="", lines=3)
        comment_gigachat_pro = gr.Textbox(label="", lines=3)
        comment_gigachat_lite = gr.Textbox(label="", lines=3)
        comment_gigachat_plus = gr.Textbox(label="", lines=3)

    with gr.Row():
        gr.Markdown("*Откорректированное сообщение (опционально):*")
        corrected_gpt4o = gr.Textbox(label="", lines=3)
        corrected_gigachat_pro = gr.Textbox(label="", lines=3)
        corrected_gigachat_lite = gr.Textbox(label="", lines=3)
        corrected_gigachat_plus = gr.Textbox(label="", lines=3)

    with gr.Row():
        gr.Button("Жми 👍 для сохранения удачного SMS в базу =>")
        save_gpt4o_btn = gr.Button("👍")
        save_gigachat_pro_btn = gr.Button("👍")
        save_gigachat_lite_btn = gr.Button("👍")
        save_gigachat_plus_btn = gr.Button("👍")

        save_gpt4o_btn.click(
            fn=lambda personalized_message, comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form: 
                save_to_github(personalized_message, "GPT-4o", comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form),
            inputs=[
                personalized_output_text_gpt4o,
                comment_gpt4o,
                corrected_gpt4o,
                description_input,
                advantages_input,
                prompt_display,
                output_text_gpt4o,
                selections[0],  # Пол
                selections[1],  # Поколение
                selections[2],  # Психотип
                selections[3],  # Стадия бизнеса
                selections[4],  # Отрасль
                selections[5]   # ОПФ
            ],
            outputs=None
        )

        save_gigachat_pro_btn.click(
            fn=lambda personalized_message, comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form: 
                save_to_github(personalized_message, "GigaChat-Pro", comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form),
            inputs=[
                personalized_output_text_gigachat_pro,
                comment_gigachat_pro,
                corrected_gigachat_pro,
                description_input,
                advantages_input,
                prompt_display,
                output_text_gigachat_pro,
                selections[0],  # Пол
                selections[1],  # Поколение
                selections[2],  # Психотип
                selections[3],  # Стадия бизнеса
                selections[4],  # Отрасль
                selections[5]   # ОПФ
            ],
            outputs=None
        )

        save_gigachat_lite_btn.click(
            fn=lambda personalized_message, comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form: 
                save_to_github(personalized_message, "GigaChat-Lite", comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form),
            inputs=[
                personalized_output_text_gigachat_lite,
                comment_gigachat_lite,
                corrected_gigachat_lite,
                description_input,
                advantages_input,
                prompt_display,
                output_text_gigachat_lite,
                selections[0],  # Пол
                selections[1],  # Поколение
                selections[2],  # Психотип
                selections[3],  # Стадия бизнеса
                selections[4],  # Отрасль
                selections[5]   # ОПФ
            ],
            outputs=None
        )

        save_gigachat_plus_btn.click(
            fn=lambda personalized_message, comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form: 
                save_to_github(personalized_message, "GigaChat-Plus", comment, corrected_message, description, advantages, non_personalized_prompt, non_personalized_message, gender, generation, psychotype, business_stage, industry, legal_form),
            inputs=[
                personalized_output_text_gigachat_plus,
                comment_gigachat_plus,
                corrected_gigachat_plus,
                description_input,
                advantages_input,
                prompt_display,
                output_text_gigachat_plus,
                selections[0],  # Пол
                selections[1],  # Поколение
                selections[2],  # Психотип
                selections[3],  # Стадия бизнеса
                selections[4],  # Отрасль
                selections[5]   # ОПФ
            ],
            outputs=None
        )

    demo.launch()