UMFUCHI
@UMFUCHI
НедоПайтонист

Можете помочь с пониманием кода?

Есть часть кода которую мне здесь и подсказали, но я не сильно понимаю как я должен её задействовать, задача кода заключалась в том чтобы бот реагировал на 2 команды 1 ответом. Вот сама часть кода, которую подсказали:
if isinstance(condition, list):
            conditions = [el.lower() for el in condition]
            predicate = lambda text: any(condition in text for condition in conditions)
        elif isinstance(condition, str):
            condition = condition.lower()
            predicate = lambda text: condition in text
        elif callable(condition):
            predicate = condition
        else:
            raise TypeError('Condition must be either string or function!')

А вот и весь код:
import speech_recognition as sr
from gtts import gTTS
import random
import time
import datetime
import playsound

#Распознаватель речи
def listen_command(): 

    r = sr.Recognizer()
    with sr.Microphone() as source:
        print("Скажите вашу команду: ")
        audio = r.listen(source)

    try:
        our_speech = r.recognize_google(audio, language="ru")
        print("Вы сказали: " + our_speech)
        return our_speech
    except sr.UnknownValueError:
        return "ошибка"
    except sr.RequestError:
        return "ошибка"


class VoiceCommandList:
    def __init__(self):
        self.actions = list()

    def on(self, condition):
        if isinstance(condition, str):
            condition = condition.lower()
            predicate = lambda text: condition in text
        elif callable(condition):
            predicate = condition
        else:
            raise TypeError('Condition must be either string or function!')

        def decorator(command_func):
            self.actions.append((predicate, command_func))
            return command_func

        return decorator

    def run_command(self, text):
        text = text.lower()
        for predicate, command in self.actions:
            if predicate(text):
                try:
                    response = command(text)
                    if response is None:
                        response = "Команда выполнена"
                except Exception as err:
                    response = "Ошибка при выполнении команды"
                    print(err)
                if response:
                    m.say_message(response)
                break
        else:
            m.say_message("Неизвестная команда")


vcl = VoiceCommandList()

#сама команда
@vcl.on('привет')
def hello(text):
    return "Здравствуйте, сэр!"

#То что говорит бот
def say_message(message):
    voice = gTTS(message, lang="ru")
    file_voice_name = "_audio_" + str(time.time()) + "_" + str(random.randint(0, 100000)) + ".mp3"
    voice.save(file_voice_name)
    playsound.playsound(file_voice_name)
    print("Голосовой ассистент: " + message)


if __name__ == '__main__':
    while True:
        command = listen_command()
        vcl.run_command(command)


Я так понимаю, что должно работать как-то так:

@vcl.on('привет',  "здравствуй")
def hello(text):
    return "Здравствуйте, сэр!"


Как мне задействовать эту часть?
Прошу объяснить как чайнику, желательно показать как должны выглядеть видоизменённые части кода
  • Вопрос задан
  • 153 просмотра
Пригласить эксперта
Ответы на вопрос 1
Vindicar
@Vindicar
RTFM!
> Я так понимаю, что должно работать как-то так:
> @vcl.on('привет', "здравствуй")
> def hello(text):

Неправильно понимаешь. Это могло бы сработать, если бы ты передал
@vcl.on(['привет', "здравствуй"]) #обрати внимание на список!
Тогда, совместно с твоим предложением по изменению on(), сработало бы. Это один способ.

Однако ты не обратил внимание вот на какой вариант:
@vcl.on('привет')
@vcl.on("здравствуй")
def hello(text):

Это тоже должно сработать! Список имеет смысл, если ты загружаешь этот список откуда-то, а не записываешь его как константу в коде скрипта.

А вообще я бы посоветовал использовать регулярные выражения. Примерно так:
import re
#.............................................
    def on(self, condition):
        if isinstance(condition, re.Pattern):
            predicate = lambda text: condition.match(text)
        elif isinstance(condition, str):
            condition = condition.lower()
            predicate = lambda text: condition in text
        elif callable(condition):
            predicate = condition
        else:
            raise TypeError('Condition must be either string or function!')

        def decorator(command_func):
            self.actions.append((predicate, command_func))
            return command_func

        return decorator

Для пущего удобства можно исправить метод run_command(), чтобы обработчику команды был доступен результат сопоставления.
def run_command(self, text):
        text = text.lower()
        for predicate, command in self.actions:
            res = predicate(text) #сохраняем результат
            if res:
                try:
                    response = command(text, res) #теперь обработчик принимает два аргумента!
                    if response is None:
                        response = "Команда выполнена"
                    else:
                        response = str(response)
                except Exception as err:
                    response = "Ошибка при выполнении команды"
                    print(err)
                if response:
                    m.say_message(response)
                break
        else:
            m.say_message("Неизвестная команда")


И пример использования будет примерно таким:
#нужно знать синтаксис регулярных выражений
#совпадёт с любой строкой, начинающейся строго со слов "скажи" или "произнеси"
#всё что после этих слов попадёт в отдельную группу 1.
@vcl.on(re.compile('^(?:скажи|произнеси) (.+)$', re.I)) 
def hello(text, match): #все обработчики будут принимать два аргумента!
    #match будет содержать результат сопоставления входной строки с регуляркой
    #group(1) будет содержать то, что оказалось на месте (.+)
    return match.group(1) #пусть бот повторит то, что мы попросили его сказать
Ответ написан
Комментировать
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Похожие вопросы
28 нояб. 2024, в 18:46
3000 руб./за проект
28 нояб. 2024, в 17:46
10000 руб./за проект
28 нояб. 2024, в 17:38
12000 руб./за проект