2024-02-27 23:18:46 +08:00
|
|
|
|
# -*- coding: utf-8 -*-
|
2024-02-29 23:03:04 +08:00
|
|
|
|
import collections
|
2024-02-27 23:18:46 +08:00
|
|
|
|
import dataclasses
|
|
|
|
|
import enum
|
|
|
|
|
import logging
|
|
|
|
|
import threading
|
|
|
|
|
from typing import *
|
|
|
|
|
|
|
|
|
|
import pyttsx3.voice
|
|
|
|
|
|
|
|
|
|
import config
|
|
|
|
|
|
|
|
|
|
logger = logging.getLogger('text-to-speech.' + __name__)
|
|
|
|
|
|
|
|
|
|
_tts: Optional['Tts'] = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Priority(enum.IntEnum):
|
|
|
|
|
HIGH = 0
|
|
|
|
|
NORMAL = 1
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclasses.dataclass
|
|
|
|
|
class TtsTask:
|
|
|
|
|
priority: Priority
|
2024-02-29 23:03:04 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def tts_text(self):
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
def merge(self, task: 'TtsTask'):
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclasses.dataclass
|
|
|
|
|
class TextTtsTask(TtsTask):
|
2024-02-27 23:18:46 +08:00
|
|
|
|
text: str
|
|
|
|
|
|
2024-02-29 23:03:04 +08:00
|
|
|
|
@property
|
|
|
|
|
def tts_text(self):
|
|
|
|
|
return self.text
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclasses.dataclass
|
|
|
|
|
class GiftTtsTask(TtsTask):
|
|
|
|
|
author_name: str
|
|
|
|
|
num: int
|
|
|
|
|
gift_name: str
|
|
|
|
|
price: float
|
|
|
|
|
total_coin: int
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def tts_text(self):
|
|
|
|
|
cfg = config.get_config()
|
|
|
|
|
is_paid_gift = self.price > 0.
|
|
|
|
|
template = cfg.template_paid_gift if is_paid_gift else cfg.template_free_gift
|
|
|
|
|
text = template.format(
|
|
|
|
|
author_name=self.author_name,
|
|
|
|
|
num=self.num,
|
|
|
|
|
gift_name=self.gift_name,
|
|
|
|
|
price=self.price,
|
|
|
|
|
total_coin=self.total_coin,
|
|
|
|
|
)
|
|
|
|
|
return text
|
|
|
|
|
|
|
|
|
|
def merge(self, task: 'TtsTask'):
|
|
|
|
|
if not isinstance(task, GiftTtsTask):
|
|
|
|
|
return False
|
|
|
|
|
if task.author_name != self.author_name or task.gift_name != self.gift_name:
|
|
|
|
|
return False
|
|
|
|
|
self.num += task.num
|
|
|
|
|
self.price += task.price
|
|
|
|
|
self.total_coin += task.total_coin
|
|
|
|
|
return True
|
|
|
|
|
|
2024-02-27 23:18:46 +08:00
|
|
|
|
|
|
|
|
|
def init():
|
|
|
|
|
global _tts
|
|
|
|
|
_tts = Tts()
|
|
|
|
|
return _tts.init()
|
|
|
|
|
|
|
|
|
|
|
2024-02-29 23:03:04 +08:00
|
|
|
|
def say_text(text, priority: Priority = Priority.NORMAL):
|
|
|
|
|
task = TextTtsTask(priority=priority, text=text)
|
|
|
|
|
return say(task)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def say(task: TtsTask):
|
|
|
|
|
logger.debug('%s', task.tts_text)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
res = _tts.push_task(task)
|
|
|
|
|
if not res:
|
|
|
|
|
if task.priority == Priority.HIGH:
|
2024-02-29 23:03:04 +08:00
|
|
|
|
logger.info('Dropped high priority task: %s', task.tts_text)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
else:
|
2024-02-29 23:03:04 +08:00
|
|
|
|
logger.debug('Dropped task: %s', task.tts_text)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Tts:
|
|
|
|
|
def __init__(self):
|
|
|
|
|
self._worker_thread = threading.Thread(target=self._worker_thread_func, daemon=True)
|
|
|
|
|
# COM组件必须在使用它的线程里初始化,否则使用时会有问题
|
|
|
|
|
self._engine: Optional[pyttsx3.Engine] = None
|
|
|
|
|
self._thread_init_event = threading.Event()
|
|
|
|
|
|
|
|
|
|
cfg = config.get_config()
|
2024-02-29 23:03:04 +08:00
|
|
|
|
self._task_queues = TaskQueue(cfg.max_tts_queue_size)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
|
|
|
|
|
def init(self):
|
|
|
|
|
self._worker_thread.start()
|
|
|
|
|
res = self._thread_init_event.wait(10)
|
|
|
|
|
if not res:
|
|
|
|
|
logger.error('Initializing TTS engine timed out')
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
|
def _init_in_worker_thread(self):
|
|
|
|
|
logger.info('Initializing TTS engine')
|
|
|
|
|
self._engine = pyttsx3.init()
|
|
|
|
|
|
|
|
|
|
voices = cast(List[pyttsx3.voice.Voice], self._engine.getProperty('voices'))
|
|
|
|
|
logger.info('Available voices:\n%s', '\n'.join(map(str, voices)))
|
|
|
|
|
|
|
|
|
|
cfg = config.get_config()
|
2024-03-01 21:58:18 +08:00
|
|
|
|
if cfg.tts_voice_id != '':
|
2024-02-27 23:18:46 +08:00
|
|
|
|
self._engine.setProperty('voice', cfg.tts_voice_id)
|
|
|
|
|
self._engine.setProperty('rate', cfg.tts_rate)
|
|
|
|
|
self._engine.setProperty('volume', cfg.tts_volume)
|
|
|
|
|
|
|
|
|
|
self._thread_init_event.set()
|
|
|
|
|
|
|
|
|
|
def push_task(self, task: TtsTask):
|
2024-02-29 23:03:04 +08:00
|
|
|
|
return self._task_queues.push(task)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
|
|
|
|
|
def _worker_thread_func(self):
|
|
|
|
|
self._init_in_worker_thread()
|
|
|
|
|
|
|
|
|
|
logger.info('Running TTS worker')
|
|
|
|
|
while True:
|
2024-02-29 23:03:04 +08:00
|
|
|
|
task = self._task_queues.pop()
|
|
|
|
|
self._engine.say(task.tts_text)
|
2024-02-27 23:18:46 +08:00
|
|
|
|
self._engine.runAndWait()
|
2024-02-29 23:03:04 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TaskQueue:
|
|
|
|
|
def __init__(self, max_size=None):
|
|
|
|
|
self._max_size: Optional[int] = max_size
|
|
|
|
|
self._queues: List[collections.deque[TtsTask]] = [
|
|
|
|
|
collections.deque(maxlen=self._max_size) for _ in Priority
|
|
|
|
|
]
|
|
|
|
|
"""任务队列,索引是优先级"""
|
|
|
|
|
self._lock = threading.Lock()
|
|
|
|
|
self._not_empty_condition = threading.Condition(self._lock)
|
|
|
|
|
|
|
|
|
|
def push(self, task: TtsTask):
|
|
|
|
|
with self._lock:
|
|
|
|
|
q = self._queues[task.priority]
|
|
|
|
|
|
|
|
|
|
# 尝试合并
|
|
|
|
|
for old_task in reversed(q):
|
|
|
|
|
if old_task.merge(task):
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
# 没满直接push
|
|
|
|
|
if (
|
|
|
|
|
self._max_size is None
|
|
|
|
|
or sum(len(q_) for q_ in self._queues) < self._max_size
|
|
|
|
|
):
|
|
|
|
|
q.append(task)
|
|
|
|
|
self._not_empty_condition.notify()
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
if task.priority != Priority.HIGH:
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
# 高优先级的尝试挤掉低优先级的任务
|
|
|
|
|
lower_q = self._queues[Priority.NORMAL]
|
|
|
|
|
try:
|
|
|
|
|
old_task = lower_q.popleft()
|
|
|
|
|
except IndexError:
|
|
|
|
|
return False
|
|
|
|
|
logger.debug('Dropped task: %s', old_task.tts_text)
|
|
|
|
|
|
|
|
|
|
q.append(task)
|
|
|
|
|
self._not_empty_condition.notify()
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
def pop(self) -> TtsTask:
|
|
|
|
|
with self._lock:
|
|
|
|
|
while True:
|
|
|
|
|
# 按优先级遍历查找任务
|
|
|
|
|
for q in self._queues:
|
|
|
|
|
try:
|
|
|
|
|
return q.popleft()
|
|
|
|
|
except IndexError:
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
self._not_empty_condition.wait()
|