syncabook
syncabook
Overview
syncabook is a set of tools for creating ebooks with synchronized text and audio (a.k.a. read along, read aloud; like Amazon’s Whispersync). You start with a list of text files (plaintext or XTML) and a list of audio files (.wav or .mp3) and get an ebook in the open EPUB3 with Media Overlays format. Here is a video that demonstrates what reading an ebook produced with syncabook looks like.
The synchronization is done automatically using the afaligner library. It is a forced aligner that works by synthesizing text and then aligning synthesized and recorded audio using a variation of the DTW (Dynamic Time Warping) algorithm. For alignment details, please refer to the afaligner repository.
Requirements
- Python (>= 3.6)
- afaligner (optional – required only for synchronization step)
-
Python packages:
beautifulsoup4
,Jinja2
,lxml
,progressbar2
Installation
-
Get the source code:
git clone https://github.com/r4victor/syncabook/ && cd syncabook
-
Install syncabook:
pip install .
Now you can run syncabook from the command line:
syncabook -h
- [optional] If you need to do the synchronization, you must also install the afaligner library. It’s not necessary if you already have SMIL files that contain synchronization information for the book. For example, you can download such files for LibriVox recordings from the synclibrivox repository.
Installation via Docker
Installing afaligner with all its dependencies may seem tedious. In that case, consider using syncabook as a Docker container.
-
Get the source code::
git clone https://github.com/r4victor/syncabook/ && cd syncabook
-
Create a Docker image:
docker build -t syncabook .
Now you can run syncabook as a Docker container. The only difference with the native installation is that you have to mount the book’s directory as a volume with the -v
option. So if you’re currently in the book’s directory, then the command will look like this:
docker run -v "$(PWD)":/books/mybook syncabook sync /books/mybook
Running tests
-
Install
pytest
andepubcheck
:pip install pytest epubcheck
-
Run tests:
python -m pytest -s tests/
Ebook production
The ebook is assembled from source files in the book’s root directory that includes:
-
The
audio/
directory containing a list of audiobook’s audio files. -
The
sync_text/
directory containing a list of XHTML files synchronized with audio files. -
The
no_sync_text/
directory containing a list of XHTML files NOT synchronized with audio files (table of contents, colophon and any other files). -
The
smil/
directory containing SMIL files (synchronization info). -
The
metadata.json
file which contains information about the book such as title, author, narrator, etc.
In order to prepare such a structure syncabook provides a set of tools. Here’s a brief outline of a typical usage; see a concrete example below.
Initially we have an empty directory named ebooks/my_ebook/
. Then we somehow get the audio files and save them to ebooks/my_ebook/audio/
. Next we get the text and save it, for example, as ebooks/my_ebook/text.txt
. Now we need to convert this plain text to a list of XHTML files. We use the split_text
and to_xhtml
commands. The split_text
command splits one plain text file into a list of plain text files, and the to_xhtml
command converts a list of plain text files to a list of XHTML files. What’s left is to synchronize the text and the audio. We may use the sync
command that performs the synchronization and produces a list of SMIL files, or we may just use the create
command that performs the synchronization and then creates an ebook. The create
command automatically creates a nav.xhtml
file containing a table of contents and a colophon.xhtml
file to credit contributors. It ask us for all the necessary information in the process and saves it as ebooks/my_ebook/metadata.json
.
If you want to create an ebook for a LibriVox recording, the download_files
command lets you automatically download the audio files from librivox.org and the transcribed text from gutenberg.org. Moreover, if someone has produced an ebook for that recording and contributed the prepared XHTML and SMIL files to the
synclibrivox repository, the download_files
command gets them as well and all you are left to do is to run the create
command.
Usage example
We will create an ebook for On the Duty of Civil Disobedience by Henry David Thoreau based on the LibriVox recording by Bob Neufeld.
-
Download the text and the audio:
syncabook download_files https://librivox.org/civil-disobedience-by-henry-david-thoreau/ civil_disobedience
-
The audio is recorded in two parts, thus we create two files in
civil_disobedience/plainext/
in which we respectively copy the contents of the first and second parts. This is a little bit of manual labor. If a book is long and recording is made in units like chapters, thesplit_text
command can help you automate this process. -
Convert the plain text files into the XHTML files:
syncabook to_xhtml civil_disobedience/plaintext/ civil_disobedience/sync_text/
-
Sync the text and the audio to produce the SMIL files:
syncabook sync civil_disobedience/
-
Create the EPUB3 ebook:
syncabook create civil_disobedience/
<p> It asks us for the book’s title, book’s author and other information. Then it generates the <code>nav.xhtml</code> file containing a table of contents and the <code>colophon.xhtml</code> file to credit contributors and places them in <code>civil_disobedience/no_sync_text/</code>. We make some changes in <code>nav.xhtml</code> and proceed. Congrats! Our ebook is created and saved in <code>civil_disobedience/out/</code>. </li> </ol> <p> See the synclibrivox repository for this and other ebooks. </p> <h2 dir="auto"> <a rel="nofollow noopener" target="_blank" id="user-content-more-features" class="anchor" aria-hidden="true" href="#more-features"></a>More features </h2> <ol dir="auto"> <li> To add a cover to the produced ebook, put a JPEG image named <code>cover.jpg</code> in the <code>images/</code> directory before running <code>syncabook create</code>. </li> </ol> <h2 dir="auto"> <a rel="nofollow noopener" target="_blank" id="user-content-how-to-read-and-listen" class="anchor" aria-hidden="true" href="#how-to-read-and-listen"></a>How to read and listen </h2> <p> The ebooks produced are in the EPUB3 format and can be opened in any EPUB3 reader. Unfortunately, the Read Aloud feature is not well supported. Here’s a list of apps, which I know of, that support it: </p> <ul dir="auto"> <li> <a rel="nofollow noopener" target="_blank" href="https://chrome.google.com/webstore/detail/readium/fepbnnnkkadjhjahcafoaglimekefifl">Readium</a> (Chrome App) – great read & listen experience. Unfortunately, Google is going to deprecate Chrome Apps. </li> <li> <a rel="nofollow noopener" target="_blank" href="https://www.edrlab.org/software/thorium-reader/">Thorium Reader</a> (Windows, MacOS and Linux) – Readium’s successor, a desktop app in active development. </li> <li> <a rel="nofollow noopener" target="_blank" href="https://www.adobe.com/la/solutions/ebook/digital-editions/download.html">Adobe Digital Editions</a> (Windows, MacOS, iOS, Android) – fully supports EPUB3 standard. Not the best reading experience, though: text and audio seem out of sync. </li> <li> Menestrello (iOS, Android) – the best app to read & listen that was developed for this specific purpose. Unfortunately, no longer maintained and not even available on AppStore or Google Play. Still, .apk can be installed on Android. </li> </ul> <p> Please let me know if you know of other apps that support EPUB3 with Media Overlays. </p> <h2 dir="auto"> <a rel="nofollow noopener" target="_blank" id="user-content-notes" class="anchor" aria-hidden="true" href="#notes"></a>Notes </h2> <ul dir="auto"> <li> While it is not required to have a one-to-one correspondence<br /> between text and audio files (i.e. the splitting can be done differently), as the practice shows, it’s not always possible to achieve a satisfying quality of synchronization and if it is possible, one may need to know the appropriate alignment parameters. Therefore, it is recommended to split text in such a way as to match audio. </li> <li> <strong>syncabook</strong> identifies text fragments in XHTML files by looking for tags with attributes of the form <code>id="f[0-9]+"</code> (see an example). If you use the <code>to_xhtml</code> command to produce XHTML files from plaintext, they will contain the proper tags automatically. If you use your own XHTML files, you’ll need to modify them to contain the tags with <code>id</code> attributes. </li> </ul>