Diposting pada 14 Jan 2025 oleh Claudio Santini
Kokoro v0.19 adalah model text-to-speech yang baru diterbitkan dengan hanya 82 juta parameter dan keluaran berkualitas sangat tinggi. Ini dirilis di bawah lisensi Apache dan dilatih dengan audio <100 jam. Saat ini mendukung bahasa Amerika, Inggris, Perancis, Korea, Jepang dan mandarin, dalam banyak suara yang sangat bagus.
Contoh kualitasnya:
Saya selalu bermimpi mengubah perpustakaan eBook saya menjadi buku audio. Apalagi untuk buku niche yang tidak bisa Anda temukan dalam format audiobook. Karena Kokoro cukup cepat, kupikir ini akhirnya bisa dilakukan. Saya telah membuat alat kecil bernama Audiblez (untuk menghormati platform buku audio populer) yang mem-parsing file .epub dan mengubah isi buku menjadi file audio yang dinarasikan dengan baik.
Di M2 MacBook Pro saya, dibutuhkan sekitar 2 jam untuk mengkonversi ke mp3 Selfish Gene oleh Richard Dawkinsyaitu sekitar 100.000 kata (atau 600.000 karakter), dengan kecepatan sekitar 80 karakter per detik.
Cara menginstal dan menjalankannya
Jika Anda memiliki Python 3 di komputer Anda, Anda dapat menginstalnya dengan pip. Ketahuilah bahwa ini tidak akan berfungsi dengan Python 3.13.
Kemudian Anda juga perlu mengunduh beberapa file tambahan di folder yang sama, yaitu sekitar ~360MB:
pip install audiblez
wget
wget
Kemudian, untuk mengonversi file epub menjadi buku audio, jalankan saja:
audiblez book.epub -l en-gb -v af_sky
Ini pertama-tama akan membuat banyak book_chapter_1.wav
, book_chapter_2.wav
dll. file dalam direktori yang sama, dan pada akhirnya akan menghasilkan a book.m4b
file dengan keseluruhan buku yang dapat Anda dengarkan dengan VLC atau pemutar buku audio apa pun. Itu hanya akan menghasilkan .m4b
file jika Anda punya ffmpeg
diinstal pada mesin Anda.
Bahasa yang Didukung
Menggunakan -l
pilihan untuk menentukan bahasa, kode bahasa yang tersedia adalah: 🇮🇩 en-us
🇮🇩 en-gb
🇫🇷 fr-fr
🇯ق ja
Korea Selatan kr
dan CNY cmn
.
Suara yang Didukung
Menggunakan -v
pilihan untuk menentukan suara: suara yang tersedia adalah af
, af_bella
, af_nicole
, af_sarah
, af_sky
, am_adam
, am_michael
, bf_emma
, bf_isabella
, bm_george
, bm_lewis
. Anda dapat mencobanya di sini:
Deteksi Bab
Deteksi bab agak tersendat-sendat, tetapi berhasil menemukan bab inti di sebagian besar .epub yang saya coba, melewatkan sampul, indeks, lampiran, dll.
Jika ternyata bab tersebut tidak menyertakan bab yang Anda minati, cobalah bermain-main dengan bab tersebut is_chapter
fungsi dalam kode. Seringkali kata pengantar atau intro dilewati, dan saya tidak yakin apakah itu bug atau fitur.
Sumber
Melihat Proyek Audiblez di GitHub.
Masih ada beberapa sisi yang kasar, tetapi itu berfungsi cukup baik bagi saya. Perbaikan di masa depan dapat mencakup:
- Deteksi bab yang lebih baik, atau izinkan pengguna untuk memasukkan/mengecualikan bab.
- Tambahkan navigasi bab ke file m4b (kelihatannya sulit, karena ffmpeg tidak melakukannya)
- Tambahkan narasi untuk gambar menggunakan beberapa model gambar-ke-teks
Kode cukup pendek untuk dimasukkan di sini:
#!/usr/bin/env python3
# audiblez - A program to convert e-books into audiobooks using
# Kokoro-82M model for high-quality text-to-speech synthesis.
# by Claudio Santini 2025 -
import argparse
import sys
import time
import shutil
import subprocess
import soundfile as sf
import ebooklib
import warnings
import re
from pathlib import Path
from string import Formatter
from bs4 import BeautifulSoup
from kokoro_onnx import Kokoro
from ebooklib import epub
from pydub import AudioSegment
def main(kokoro, file_path, lang, voice):
filename = Path(file_path).name
with warnings.catch_warnings():
book = epub.read_epub(file_path)
title = book.get_metadata('DC', 'title')(0)(0)
creator = book.get_metadata('DC', 'creator')(0)(0)
intro = f'{title} by {creator}'
print(intro)
chapters = find_chapters(book)
print('Found chapters:', (c.get_name() for c in chapters))
texts = extract_texts(chapters)
has_ffmpeg = shutil.which('ffmpeg') is not None
if not has_ffmpeg:
print('\033(91m' + 'ffmpeg not found. Please install ffmpeg to create mp3 and m4b audiobook files.' + '\033(0m')
total_chars = sum((len
print('Started at:', time.strftime('%H:%M:%S'))
print(f'Total characters: {total_chars:,}')
print('Total words:', len(' '.join(texts).split(' ')))
i = 1
chapter_mp3_files = ()
for text in texts:
chapter_filename = filename.replace('.epub', f'_chapter_{i}.wav')
chapter_mp3_files.append(chapter_filename)
if Path(chapter_filename).exists():
print(f'File for chapter {i} already exists. Skipping')
i += 1
continue
print(f'Reading chapter {i} ({len(text):,} characters)...')
if i == 1:
text = intro + '.\n\n' + text
start_time = time.time()
samples, sample_rate = kokoro.create(text, voice=voice, speed=1.0, lang=lang)
sf.write(f'{chapter_filename}', samples, sample_rate)
end_time = time.time()
delta_seconds = end_time - start_time
chars_per_sec = len(text) / delta_seconds
remaining_chars = sum((len
remaining_time = remaining_chars / chars_per_sec
print(f'Estimated time remaining: {strfdelta(remaining_time)}')
print('Chapter written to', chapter_filename)
print(f'Chapter {i} read in {delta_seconds:.2f} seconds ({chars_per_sec:.0f} characters per second)')
progress = int((total_chars - remaining_chars) / total_chars * 100)
print('Progress:', f'{progress}%')
i += 1
if has_ffmpeg:
create_m4b(chapter_mp3_files, filename)
def extract_texts(chapters):
texts = ()
for chapter in chapters:
xml = chapter.get_body_content()
soup = BeautifulSoup(xml, features="lxml")
chapter_text=""
html_content_tags = ('title', 'p', 'h1', 'h2', 'h3', 'h4')
for child in soup.find_all(html_content_tags):
inner_text = child.text.strip() if child.text else ""
if inner_text:
chapter_text += inner_text + '\n'
texts.append(chapter_text)
return texts
def is_chapter(c):
name = c.get_name().lower()
part = r"part\d{1,3}"
if re.search(part, name):
return True
ch = r"ch\d{1,3}"
if re.search(ch, name):
return True
if 'chapter' in name:
return True
def find_chapters(book, verbose=True):
chapters = (c for c in book.get_items() if c.get_type() == ebooklib.ITEM_DOCUMENT and is_chapter(c))
if verbose:
for item in book.get_items():
if item.get_type() == ebooklib.ITEM_DOCUMENT:
# print(f"'{item.get_name()}'" + ', #' + str(len(item.get_body_content())))
print(f'{item.get_name()}'.ljust(60), str(len(item.get_body_content())).ljust(15), 'X' if item in chapters else '-')
return chapters
def strfdelta(tdelta, fmt="{D:02}d {H:02}h {M:02}m {S:02}s"):
remainder = int(tdelta)
f = Formatter()
desired_fields = (field_tuple(1) for field_tuple in f.parse(fmt))
possible_fields = ('W', 'D', 'H', 'M', 'S')
constants = {'W': 604800, 'D': 86400, 'H': 3600, 'M': 60, 'S': 1}
values = {}
for field in possible_fields:
if field in desired_fields and field in constants:
values(field), remainder = divmod(remainder, constants(field))
return f.format(fmt, **values)
def create_m4b(chaptfer_files, filename):
tmp_filename = filename.replace('.epub', '.tmp.m4a')
if not Path(tmp_filename).exists():
combined_audio = AudioSegment.empty()
for wav_file in chaptfer_files:
audio = AudioSegment.from_wav(wav_file)
combined_audio += audio
print('Converting to Mp4...')
combined_audio.export(tmp_filename, format="mp4", codec="aac", bitrate="64k")
final_filename = filename.replace('.epub', '.m4b')
print('Creating M4B file...')
proc = subprocess.run(('ffmpeg', '-i', f'{tmp_filename}', '-c', 'copy', '-f', 'mp4', f'{final_filename}'))
Path(tmp_filename).unlink()
if proc.returncode == 0:
print(f'{final_filename} created. Enjoy your audiobook.')
print('Feel free to delete the intermediary .wav chapter files, the .m4b is all you need.')
def cli_main():
if not Path('kokoro-v0_19.onnx').exists() or not Path('voices.json').exists():
print('Error: kokoro-v0_19.onnx and voices.json must be in the current directory. Please download them with:')
print('wget ')
print('wget ')
sys.exit(1)
kokoro = Kokoro('kokoro-v0_19.onnx', 'voices.json')
voices = list(kokoro.get_voices())
voices_str=", ".join(voices)
epilog = 'example:\n' +
' audiblez book.epub -l en-us -v af_sky'
default_voice="af_sky" if 'af_sky' in voices else voices(0)
parser = argparse.ArgumentParser(epilog=epilog, formatter_class=argparse.RawDescriptionHelpFormatter)
parser.add_argument('epub_file_path', help='Path to the epub file')
parser.add_argument('-l', '--lang', default="en-gb", help='Language code: en-gb, en-us, fr-fr, ja, ko, cmn')
parser.add_argument('-v', '--voice', default=default_voice, help=f'Choose narrating voice: {voices_str}')
if len(sys.argv) == 1:
parser.print_help(sys.stderr)
sys.exit(1)
args = parser.parse_args()
main(kokoro, args.epub_file_path, args.lang, args.voice)
if __name__ == '__main__':
cli_main()